基于opencv的MFC多摄像机视场标定软件

一、项目需求
确定网络摄像机的监控视场在全景地图上的精确位置,并采用简单而高效的方法将其所在视场标定并投影到全景地图上,项目要求能够将多路网络摄像机的视场标定在全景地图上,并实现这样的功能:当调取当前摄像机视频播放时,其他摄像机处于暂停状态。

二、项目分析
1、如何将视频投影到全景地图上,实质上就是如何将一幅幅视频帧图像与全景图像进行配准,关于图像配准方面的研究已经有很多方法,本项目采用基于互信息的图像配准算法寻找四对精确特征匹配点对。
2、设计一个多摄像机视场标定的人机交互界面的软件系统,要求能够将多路网络摄像机的视场标定在全景地图上,并实现调取当前摄像机视频播放时,其他摄像机处于暂停状态。

三、最终软件界面设计如下:

四、部分代码如下:

// CamCalibrationDlg.h : 头文件
//

#pragma once
#include "cv.h"
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include <opencv2/core/core.hpp>
#include "opencv2/calib3d/calib3d.hpp"
#include "highgui.h"
#include "cxcore.h"
#include "CvvImage.h"
#include <vector>
#include "afxwin.h"

using namespace cv;
using namespace std;

// CCamCalibrationDlg 对话框
class CCamCalibrationDlg : public CDialogEx
{
// 构造
public:
    CCamCalibrationDlg(CWnd* pParent = NULL);   // 标准构造函数

// 对话框数据
    enum { IDD = IDD_CAMCALIBRATION_DIALOG };

    protected:
    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持


// 实现
protected:
    HICON m_hIcon;

    // 生成的消息映射函数
    virtual BOOL OnInitDialog();
    afx_msg void OnSysCommand(UINT nID, LPARAM lParam);
    afx_msg void OnPaint();
    afx_msg HCURSOR OnQueryDragIcon();
    DECLARE_MESSAGE_MAP()

private:

    //全屏显示
    BOOL bFullScreen;
    CRect rectFullScreen;
    WINDOWPLACEMENT m_struOldWndpl;//结构中包含了有关窗口在屏幕上位置的信息
    WINDOWPLACEMENT m_struOldWndpPic;//PICTURE控件在屏幕上位置的信息

    vector<Point2f> m_newpoints; // 需要利用互信息子函数更新得到的匹配点集(要求四个) 
    vector<Point2f> m_points1;// 视频帧图像匹配点集(要求四个)
    vector<Point2f> m_points2;// 全景图像匹配点集,初始匹配点集(要求四个)

    Mat TheImage; //全景图
    Mat CamImage; //视频帧图像

    CStatic m_TheImage;//是全景图界面的显示窗口
    CStatic m_CamImage;//是视频帧图像界面的显示窗口

    CRect m_TheImageRect;//显示全景窗口的矩形窗
    CRect m_CamImageRect;//显示视频窗口的矩形窗

    CString m_pt;    //与编辑框绑定的成员变量,将获取的点集中显示在编辑框中
    double  m_scale;    //全景图缩放比率

    CvCapture *capture;

    int g_m_run;  //视频暂停标志


    Mat H1; //投影变换矩阵1
    Mat H2; //投影变换矩阵2
    Mat H3; //投影变换矩阵3
    Mat H4; //投影变换矩阵4
    Mat H5; //投影变换矩阵5
    Mat H6; //投影变换矩阵6

    //存储投影后四个顶点的坐标
    vector<Point2f> m_pointsh1;
    vector<Point2f> m_pointsh2;
    vector<Point2f> m_pointsh3;
    vector<Point2f> m_pointsh4;
    vector<Point2f> m_pointsh5;
    vector<Point2f> m_pointsh6;


    int m_Radio; 

    int m_Pointdownflag;

public:

    void ShowMatImgToWnd(CWnd* pWnd, cv::Mat img);

    afx_msg void OnBnClickedPictureOpen();

    double Entropy(Mat img);//单幅图像信息熵计算

    double ComEntropy(Mat img1, Mat img2, double img1_entropy, double img2_entropy);// 两幅图像联合信息熵计算

    Point2f Refresh_MacthPoints(Point2f point1, Point2f point2);//查找全景图上四个精确匹配点集

    Point2f MousePointToImgPixel(Mat img, CRect rect, Point2f point);// 鼠标点击坐标转换为实际图像坐标《分辨率转换》



    afx_msg void OnBnClickedVideoPreview();
    afx_msg void OnBnClickedVideoPlay();
    afx_msg void OnBnClickedVideoPause();
    // afx_msg void OnTimer(UINT_PTR nIDEvent);


    afx_msg void OnBnClickedRadio1();
    afx_msg void OnBnClickedRadio2();
    afx_msg void OnBnClickedRadio3();
    afx_msg void OnBnClickedRadio4();
    afx_msg void OnBnClickedRadio5();
    afx_msg void OnBnClickedRadio6();

    afx_msg void OnBnClickedPictureChangeSize();
    afx_msg void OnBnClickedPicturePoints();
    afx_msg void OnBnClickedVideoPoints();
    afx_msg void OnBnClickedSavePoints();

    afx_msg void OnBnClickedAdjustPoints();

    afx_msg void OnBnClickedQuit();
    afx_msg void OnBnClickedHomgtaphyFind();
    afx_msg void OnBnClickedCalibration();

    Mat  showFinal(Mat src1, Mat src2); //将标定后的组合图显示出来
    afx_msg void OnMouseMove(UINT nFlags, CPoint point);
    afx_msg void OnLButtonDown(UINT nFlags, CPoint point);



    ////全屏显示
    afx_msg void OnLButtonDblClk(UINT nFlags, CPoint point);
    afx_msg void OnGetMinMaxInfo(MINMAXINFO* lpMMI);
    void drawpic(IplImage* img, unsigned int id);
    //绘图到&nbsp;MFC 的 Picture Control 控件相关函数
    //参数一为 OpenCV&nbsp;的图像数据结构类,参数二为 Picture Control 控件的id

};

// CamCalibrationDlg.cpp : 实现文件
//

#include "stdafx.h"
#include "CamCalibration.h"
#include "CamCalibrationDlg.h"
#include "afxdialogex.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif

#define WIDTHBYTES(bits) (((bits)+31)/32*4)//用于使图像宽度所占字节数为4byte的倍数 

// 用于应用程序“关于”菜单项的 CAboutDlg 对话框

class CAboutDlg : public CDialogEx
{
public:
    CAboutDlg();

// 对话框数据
    enum { IDD = IDD_ABOUTBOX };

    protected:
    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持

// 实现
protected:
    DECLARE_MESSAGE_MAP()
};

CAboutDlg::CAboutDlg() : CDialogEx(CAboutDlg::IDD)
{
}

void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialogEx::DoDataExchange(pDX);


}

BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP()


// CCamCalibrationDlg 对话框



CCamCalibrationDlg::CCamCalibrationDlg(CWnd* pParent /*=NULL*/)
    : CDialogEx(CCamCalibrationDlg::IDD, pParent)

{
    m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void CCamCalibrationDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialogEx::DoDataExchange(pDX);

    DDX_Control(pDX, IDC_PICTURE, m_TheImage);//自定义控件和IDC界面控件绑定
    DDX_Control(pDX, IDC_VIDEO, m_CamImage);//自定义控件和IDC界面控件绑定
    DDX_Text(pDX, IDC_EDIT_POINTSHOW, m_pt);//自定义控件和IDC编辑框绑定
    DDX_Text(pDX, IDC_EDIT_RESIZE, m_scale);//自定义控件和IDC编辑框绑定


}

BEGIN_MESSAGE_MAP(CCamCalibrationDlg, CDialogEx)
    ON_WM_SYSCOMMAND()
    ON_WM_PAINT()
    ON_WM_QUERYDRAGICON()
    ON_BN_CLICKED(IDC_PICTURE_OPEN, &CCamCalibrationDlg::OnBnClickedPictureOpen)
    ON_BN_CLICKED(IDC_VIDEO_PREVIEW, &CCamCalibrationDlg::OnBnClickedVideoPreview)
    ON_BN_CLICKED(IDC_VIDEO_PLAY, &CCamCalibrationDlg::OnBnClickedVideoPlay)
    ON_BN_CLICKED(IDC_VIDEO_PAUSE, &CCamCalibrationDlg::OnBnClickedVideoPause)
    // ON_WM_TIMER()

    ON_BN_CLICKED(IDC_RADIO1, &CCamCalibrationDlg::OnBnClickedRadio1)
    ON_BN_CLICKED(IDC_RADIO2, &CCamCalibrationDlg::OnBnClickedRadio2)
    ON_BN_CLICKED(IDC_RADIO3, &CCamCalibrationDlg::OnBnClickedRadio3)
    ON_BN_CLICKED(IDC_RADIO4, &CCamCalibrationDlg::OnBnClickedRadio4)
    ON_BN_CLICKED(IDC_RADIO5, &CCamCalibrationDlg::OnBnClickedRadio5)
    ON_BN_CLICKED(IDC_RADIO6, &CCamCalibrationDlg::OnBnClickedRadio6)
    ON_BN_CLICKED(IDC_PICTURE_CHANGE_SIZE, &CCamCalibrationDlg::OnBnClickedPictureChangeSize)
    ON_BN_CLICKED(IDC_PICTURE_POINTS, &CCamCalibrationDlg::OnBnClickedPicturePoints)
    ON_BN_CLICKED(IDC_VIDEO_POINTS, &CCamCalibrationDlg::OnBnClickedVideoPoints)
    ON_BN_CLICKED(IDC_ADJUST_POINTS, &CCamCalibrationDlg::OnBnClickedAdjustPoints)
    ON_BN_CLICKED(IDC_QUIT, &CCamCalibrationDlg::OnBnClickedQuit)
    ON_BN_CLICKED(IDC_HOMGTAPHY_FIND, &CCamCalibrationDlg::OnBnClickedHomgtaphyFind)
    ON_BN_CLICKED(IDC_CALIBRATION, &CCamCalibrationDlg::OnBnClickedCalibration)

    ON_WM_MOUSEMOVE()
    ON_WM_LBUTTONDOWN()

    ON_WM_LBUTTONDBLCLK()
    ON_WM_GETMINMAXINFO()
    ON_BN_CLICKED(IDC_SAVE_POINTS, &CCamCalibrationDlg::OnBnClickedSavePoints)
END_MESSAGE_MAP()


// CCamCalibrationDlg 消息处理程序

BOOL CCamCalibrationDlg::OnInitDialog()
{
    CDialogEx::OnInitDialog();

    // 将“关于...”菜单项添加到系统菜单中。

    // IDM_ABOUTBOX 必须在系统命令范围内。
    ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
    ASSERT(IDM_ABOUTBOX < 0xF000);

    CMenu* pSysMenu = GetSystemMenu(FALSE);
    if (pSysMenu != NULL)
    {
        BOOL bNameValid;
        CString strAboutMenu;
        bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
        ASSERT(bNameValid);
        if (!strAboutMenu.IsEmpty())
        {
            pSysMenu->AppendMenu(MF_SEPARATOR);
            pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
        }
    }

    // 设置此对话框的图标。 当应用程序主窗口不是对话框时,框架将自动
    // 执行此操作
    SetIcon(m_hIcon, TRUE);         // 设置大图标
    SetIcon(m_hIcon, FALSE);        // 设置小图标

    ShowWindow(SW_MAXIMIZE);

    // TODO: 在此添加额外的初始化代码



    return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}

void CCamCalibrationDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
    if ((nID & 0xFFF0) == IDM_ABOUTBOX)
    {
        CAboutDlg dlgAbout;
        dlgAbout.DoModal();
    }
    else
    {
        CDialogEx::OnSysCommand(nID, lParam);
    }
}

// 如果向对话框添加最小化按钮,则需要下面的代码
// 来绘制该图标。 对于使用文档/视图模型的 MFC 应用程序,
// 这将由框架自动完成。

void CCamCalibrationDlg::OnPaint()
{
    if (IsIconic())
    {
        CPaintDC dc(this); // 用于绘制的设备上下文

        SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);

        // 使图标在工作区矩形中居中
        int cxIcon = GetSystemMetrics(SM_CXICON);
        int cyIcon = GetSystemMetrics(SM_CYICON);
        CRect rect;
        GetClientRect(&rect);
        int x = (rect.Width() - cxIcon + 1) / 2;
        int y = (rect.Height() - cyIcon + 1) / 2;

        // 绘制图标
        dc.DrawIcon(x, y, m_hIcon);
    }
    else
    {
        CDialogEx::OnPaint();
    }
}

//当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR CCamCalibrationDlg::OnQueryDragIcon()
{
    return static_cast<HCURSOR>(m_hIcon);
}



// 将图像显示到对应的图像框
void CCamCalibrationDlg::ShowMatImgToWnd(CWnd* pWnd, cv::Mat img)
{
    if (img.empty())
        return;
    CDC *pDC = pWnd->GetDC();
    HDC hDC = pDC->GetSafeHdc();
    CRect rect;
    pWnd->GetClientRect(&rect);
    IplImage Iimg = img;
    CvvImage cimg;
    cimg.CopyOf(&Iimg); // 复制图片
    cimg.DrawToHDC(hDC, &rect); // 将图片绘制到显示控件的指定区域内
    ReleaseDC(pDC);
}



void CCamCalibrationDlg::OnBnClickedPictureOpen()
{
    // TODO: 在此添加控件通知处理程序代码
    CFileDialog dlg(
        TRUE, _T("*.bmp;*.jpg;*.jpeg"), NULL,
        OFN_FILEMUSTEXIST | OFN_PATHMUSTEXIST | OFN_HIDEREADONLY,
        _T("image files (*.bmp; *.jpg;*.jpeg) |*.bmp; *.jpg; *.jpeg | All Files (*.*) |*.*||"), NULL
        );                                        // 选项图片的约定
    dlg.m_ofn.lpstrTitle = _T("Open Image");    // 打开文件对话框的标题名
    if (dlg.DoModal() != IDOK)                    // 判断是否获得图片
        return;

    CString mPath = dlg.GetPathName();            // 获取图片路径

    TheImage = cvLoadImage(mPath, 1);    // 读取图片、缓存到一个局部变量 ipl 中
    // Mat TheImage = imread(mPath,1);
    // drawpic(TheImage, IDC_PICTURE);
    ShowMatImgToWnd(GetDlgItem(IDC_PICTURE), TheImage);  
    UpdateWindow();
}


//单幅图像信息熵计算
double CCamCalibrationDlg::Entropy(Mat img)
{
    double temp[256] = { 0.0 };

    // 计算每个像素的累积值
    for (int m = 0; m<img.rows; m++)
    {// 有效访问行列的方式
        const uchar* t = img.ptr<uchar>(m);
        for (int n = 0; n<img.cols; n++)
        {
            int i = t[n];
            temp[i] = temp[i] + 1;
        }
    }

    // 计算每个像素的概率
    for (int i = 0; i<256; i++)
    {
        temp[i] = temp[i] / (img.rows*img.cols);
    }

    double result = 0;
    // 计算图像信息熵
    for (int i = 0; i<256; i++)
    {
        if (temp[i] == 0.0)
            result = result;
        else
            result = result - temp[i] * (log(temp[i]) / log(2.0));
    }

    return result;

}




// 两幅图像联合信息熵计算
double CCamCalibrationDlg::ComEntropy(Mat img1, Mat img2, double img1_entropy, double img2_entropy)
{
    double temp[256][256] = { 0.0 };

    // 计算联合图像像素的累积值
    for (int m1 = 0, m2 = 0; m1 < img1.rows, m2 < img2.rows; m1++, m2++)
    {    // 有效访问行列的方式
        const uchar* t1 = img1.ptr<uchar>(m1);
        const uchar* t2 = img2.ptr<uchar>(m2);
        for (int n1 = 0, n2 = 0; n1 < img1.cols, n2 < img2.cols; n1++, n2++)
        {
            int i = t1[n1], j = t2[n2];
            temp[i][j] = temp[i][j] + 1;
        }
    }

    // 计算每个联合像素的概率
    for (int i = 0; i < 256; i++)
    {
        for (int j = 0; j < 256; j++)

        {
            temp[i][j] = temp[i][j] / (img1.rows*img1.cols);
        }
    }

    double result = 0.0;
    //计算图像联合信息熵
    for (int i = 0; i < 256; i++)
    {
        for (int j = 0; j < 256; j++)

        {
            if (temp[i][j] == 0.0)
                result = result;
            else
                result = result - temp[i][j] * (log(temp[i][j]) / log(2.0));
        }
    }

    //得到两幅图像的互信息熵
    img1_entropy = Entropy(img1);
    img2_entropy = Entropy(img2);
    result = img1_entropy + img2_entropy - result;

    return result;

}


Point2f CCamCalibrationDlg::Refresh_MacthPoints(Point2f point1, Point2f point2)
{

    int IMGSIDE = 10; //截取正方形子图的边长的一半
    int ScanSide = 5; //搜索半径

    Mat dst1;   // 视频帧图像上截取子图
    Mat dst2;   // 全景图像上截取子图


    Point2f CLSPoint, CRXPoint;
    //截取视频帧图像子图 
    CLSPoint.x = point1.x - IMGSIDE;
    CLSPoint.y = point1.y - IMGSIDE;
    CRXPoint.x = point1.x + IMGSIDE;
    CRXPoint.y = point1.y + IMGSIDE;
    Rect ROI0(CLSPoint, CRXPoint);
    CamImage(ROI0).copyTo(dst1);


    Point2f TLSPoint, TRXPoint;
    //截取全景图图像子图
    TLSPoint.x = point2.x - IMGSIDE;
    TLSPoint.y = point2.y - IMGSIDE;
    TRXPoint.x = point2.x + IMGSIDE;
    TRXPoint.y = point2.y + IMGSIDE;
    Rect ROI00(TLSPoint, TRXPoint);
    TheImage(ROI00).copyTo(dst2);


    //计算对应图像子块的互信息熵
    double OneImaEntropy1; //视频帧图像信息熵
    double OneImaEntropy2; //全景图信息熵
    double TwoImaEntropy; //初始互信息熵

    OneImaEntropy1 = Entropy(dst1);
    OneImaEntropy2 = Entropy(dst2);
    TwoImaEntropy = ComEntropy(dst1, dst2, OneImaEntropy1, OneImaEntropy2);


    Point2f NewPoints; // 更新的匹配点

    Point2f Points;
    Mat dst;

    double OneImaEntropy;
    double NewTwoEntropy;
    double MAX = TwoImaEntropy;

    for (int i = -ScanSide; i <= ScanSide; i++)
    {
        for (int j = -ScanSide; j <= ScanSide; j++)
        {
            NewPoints.x = point2.x + i;
            NewPoints.y = point2.y + j;

            Rect rect(NewPoints.x - IMGSIDE, NewPoints.y - IMGSIDE, 2 * IMGSIDE, 2 * IMGSIDE);
            TheImage(rect).copyTo(dst);


            OneImaEntropy = Entropy(dst);
            NewTwoEntropy = ComEntropy(dst, dst1, OneImaEntropy, OneImaEntropy1);


            if (NewTwoEntropy > MAX)   // 计算所有互信息熵,然后取最大值的坐标点
            {
                MAX = NewTwoEntropy;
                //Points = NewPoints;
                Points.x = point2.x + i;
                Points.y = point2.y + j;
            }

        }
    }

    return Points;
}



// 鼠标点击坐标转换为实际图像坐标
Point2f CCamCalibrationDlg::MousePointToImgPixel(Mat img, CRect rect, Point2f point)
{
    Point2f imgpix;
    imgpix.x = point.x / (rect.right - rect.left) * img.cols;
    imgpix.y = point.y / (rect.bottom - rect.top) * img.rows;

    return imgpix;
}







void CCamCalibrationDlg::OnBnClickedVideoPreview()
{

    switch (m_Radio)
    { 
      case 1: 
          capture = cvCaptureFromAVI("51_20150115161213.avi");

          break;
      case 2:
          capture = cvCaptureFromAVI("53_20150115161348.avi");

          break;
      case 3:
          capture = cvCaptureFromAVI("54_20150115161515.avi");

          break;
      case 4: 
          capture = cvCaptureFromAVI("56_20150115161648.avi");

          break;
      case 5:
          capture = cvCaptureFromAVI("58_20150115161823.avi");

          break;
      case 6:
          capture = cvCaptureFromAVI("59_20150115161946.avi");

          break;
    default:
        break;
    }

    CamImage = cvQueryFrame(capture); //获取一帧图片 
    //CamImage = imread("22.jpg");
    ShowMatImgToWnd(GetDlgItem(IDC_VIDEO), CamImage);

}


void CCamCalibrationDlg::OnBnClickedVideoPlay()
{
    // TODO: 在此添加控件通知处理程序代码
    //if (!capture)
    //{
    // // capture = cvCaptureFromCAM(0);//调取摄像头
    // capture = cvCaptureFromAVI("53_20150115161348.avi");//调取视频
    //}
    Mat m_Frame;
    // m_Frame = cvQueryFrame(capture);
    g_m_run = 1;
    while(g_m_run == 1)
    {
        m_Frame = cvQueryFrame(capture);
        ShowMatImgToWnd(GetDlgItem(IDC_VIDEO), m_Frame);
        char c = cvWaitKey(33);
    }

    // 设置计时器,每30ms触发一次事件 
    // SetTimer(1, 1000 / 25, NULL);

}


void CCamCalibrationDlg::OnBnClickedVideoPause()
{
    // TODO: 在此添加控件通知处理程序代码
    g_m_run = 0;

}




//void CCamCalibrationDlg::OnTimer(UINT_PTR nIDEvent)
//{
// // TODO: 在此添加消息处理程序代码和/或调用默认值
// Mat m_Frame;
// m_Frame = cvQueryFrame(capture);
//
//
// if (g_m_run = 1)
// {
// ShowMatImgToWnd(GetDlgItem(IDC_VIDEO), m_Frame);
// }
//
// CDialogEx::OnTimer(nIDEvent);
//}





void CCamCalibrationDlg::OnBnClickedRadio1()
{
    // TODO: 在此添加控件通知处理程序代码
    m_Radio = 1;
}




void CCamCalibrationDlg::OnBnClickedRadio2()
{
    // TODO: 在此添加控件通知处理程序代码
    m_Radio = 2;
}


void CCamCalibrationDlg::OnBnClickedRadio3()
{
    // TODO: 在此添加控件通知处理程序代码
    m_Radio = 3;
}


void CCamCalibrationDlg::OnBnClickedRadio4()
{
    // TODO: 在此添加控件通知处理程序代码
    m_Radio = 4;
}


void CCamCalibrationDlg::OnBnClickedRadio5()
{
    // TODO: 在此添加控件通知处理程序代码
    m_Radio = 5;
}


void CCamCalibrationDlg::OnBnClickedRadio6()
{
    // TODO: 在此添加控件通知处理程序代码
    m_Radio = 6;
}


void CCamCalibrationDlg::OnBnClickedPictureChangeSize()
{
    // TODO: 在此添加控件通知处理程序代码
    Mat tmpImage = TheImage; //将全景图赋给临时图像
    Mat dstImage; //目标图像
    UpdateData(true); // 将控件数据传递给变量,更新显示 
    //double m_scale = 1.75; //全景图缩放比率
    resize(tmpImage, dstImage, Size(tmpImage.cols * m_scale, tmpImage.rows * m_scale), CV_INTER_LINEAR);

    //pyrUp(tmpImage, dstImage, Size(tmpImage.cols * 0.5 , tmpImage.rows * 0.5));
    imshow("1", dstImage);
    TheImage = dstImage;
}





void CCamCalibrationDlg::OnBnClickedPicturePoints()
{
    // TODO: 在此添加控件通知处理程序代码 
     m_Pointdownflag = 1;    
}


void CCamCalibrationDlg::OnBnClickedVideoPoints()
{
    // TODO: 在此添加控件通知处理程序代码 
    m_Pointdownflag = 2;     
}

void CCamCalibrationDlg::OnBnClickedSavePoints()
{
    // TODO: 在此添加控件通知处理程序代码
    m_Pointdownflag = 3;
}


void CCamCalibrationDlg::OnBnClickedAdjustPoints()
{
    // TODO: 在此添加控件通知处理程序代码
    vector<Point2f> Newpoints;// 需要利用互信息子函数更新得到的匹配点集(要求四个) 


    vector<Point2f> Points1;// 视频帧图像匹配点集(要求四个)
    vector<Point2f> Points2;// 全景图像匹配点集,初始匹配点集(要求四个)

    for (int i = 0; i < m_points1.size(); i++)
    {

        Point2f CamImagePix = MousePointToImgPixel(CamImage, m_CamImageRect, m_points1[i]);
        Points1.push_back(CamImagePix);
    }

    for (int i = 0; i < m_points2.size(); i++)
    {

        Point2f TheImagePix = MousePointToImgPixel(TheImage, m_TheImageRect, m_points2[i]);
        Points2.push_back(TheImagePix);
    }

    for (int i = 0; i < Points2.size(); i++)
    {
        Point2f Newpoint = Refresh_MacthPoints(Points1[i], Points2[i]);//查找全景图上四个精确匹配点集
        Newpoints.push_back(Newpoint);
    }

    m_newpoints = Newpoints;

    //清空临时容器
     Newpoints.clear();
     Points1.clear();
     Points2.clear();


}


void CCamCalibrationDlg::OnBnClickedQuit()
{
    // TODO: 在此添加控件通知处理程序代码
    CDialogEx::OnCancel();
}


void CCamCalibrationDlg::OnBnClickedHomgtaphyFind()
{
    // TODO: 在此添加控件通知处理程序代码

    vector <Point2f> TheImgPix;
    vector <Point2f> CamImgPix;


    for (int i = 0; i < m_points1.size(); i++)
    {

        Point2f CamImagePix = MousePointToImgPixel(CamImage, m_CamImageRect, m_points1[i]);
        CamImgPix.push_back(CamImagePix);
    }

    //校准后匹配
    //for (int i = 0; i < m_newpoints.size(); i++)
    //{
    // Point2f TheImagePix = MousePointToImgPixel(TheImage, m_TheImageRect, m_newpoints[i]);
    // TheImgPix.push_back(TheImagePix);
    //}

     // TheImgPix = m_newpoints;


    //校准前匹配
    for (int i = 0; i < m_points2.size(); i++)
    {
        Point2f TheImagePix = MousePointToImgPixel(TheImage, m_TheImageRect, m_points2[i]);
        TheImgPix.push_back(TheImagePix);
    }

     FileStorage fs1(".\\H1.xml", FileStorage::WRITE);
     FileStorage fs2(".\\H2.xml", FileStorage::WRITE);
     FileStorage fs3(".\\H3.xml", FileStorage::WRITE);
     FileStorage fs4(".\\H4.xml", FileStorage::WRITE);
     FileStorage fs5(".\\H5.xml", FileStorage::WRITE);
     FileStorage fs6(".\\H6.xml", FileStorage::WRITE);


     switch (m_Radio)
     {
     case 1:
         H1 = getPerspectiveTransform(CamImgPix, TheImgPix);//src ,dst
         fs1 << "H1" << H1;
         fs1.release();
         break;
     case 2:
         H2 = getPerspectiveTransform(CamImgPix, TheImgPix);//src ,dst
         fs2 << "H2" << H2;
         fs2.release();
         break;
     case 3:
         H3 = getPerspectiveTransform(CamImgPix, TheImgPix);//src ,dst
         fs3 << "H3" << H3;
         fs3.release();
         break;
     case 4:
         H4 = getPerspectiveTransform(CamImgPix, TheImgPix);//src ,dst
         fs4 << "H4" << H4;
         fs4.release();
         break;
     case 5:
         H5 = getPerspectiveTransform(CamImgPix, TheImgPix);//src ,dst
         fs5 << "H5" << H5;
         fs5.release();
         break;
     case 6:
         H6 = getPerspectiveTransform(CamImgPix, TheImgPix);//src ,dst
         fs6 << "H6" << H6;
         fs6.release();
         break;
     default:
         break;
     }

     TheImgPix.clear();//计算完H后清空
     CamImgPix.clear();
}


void CCamCalibrationDlg::OnBnClickedCalibration()
{
    // TODO: 在此添加控件通知处理程序代码

    //清空容器里的坐标点
     m_newpoints.clear(); 
     m_points1.clear();
     m_points2.clear();


    Mat CamImage1;
    CamImage1 = cvQueryFrame(capture);
    Mat logoWarped;

    if (true)
    {
        CamImage1 = cvQueryFrame(capture);
        switch (m_Radio)
        {
        case 1:
            warpPerspective(CamImage1, logoWarped, H1, TheImage.size(), INTER_LINEAR, BORDER_CONSTANT);
            break;
        case 2:
            warpPerspective(CamImage1, logoWarped, H2, TheImage.size(), INTER_LINEAR, BORDER_CONSTANT);
            break;
        case 3:
            warpPerspective(CamImage1, logoWarped, H3, TheImage.size(), INTER_LINEAR, BORDER_CONSTANT);
            break;
        case 4:
            warpPerspective(CamImage1, logoWarped, H4, TheImage.size(), INTER_LINEAR, BORDER_CONSTANT);
            break;
        case 5:
            warpPerspective(CamImage1, logoWarped, H5, TheImage.size(), INTER_LINEAR, BORDER_CONSTANT);
            break;
        case 6:
            warpPerspective(CamImage1, logoWarped, H6, TheImage.size(), INTER_LINEAR, BORDER_CONSTANT);
            break;
        default:
            break;
        }

        Mat TheImage1;
        TheImage1 = showFinal(TheImage, logoWarped);
        TheImage = TheImage1;

        //ShowMatImgToWnd(GetDlgItem(IDC_VIDEO), m_Frame);
    }
    // 设置计时器,每30ms触发一次事件 
    SetTimer(1, 1000 / 25, NULL);
}


Mat  CCamCalibrationDlg::showFinal(Mat src1, Mat src2)
{

    Mat gray, gray_inv, src1final, src2final;
    cvtColor(src2, gray, CV_BGR2GRAY);
    threshold(gray, gray, 0, 255, CV_THRESH_BINARY);
    //adaptiveThreshold(gray,gray,255,ADAPTIVE_THRESH_MEAN_C,THRESH_BINARY,5,4);
    bitwise_not(gray, gray_inv);
    src1.copyTo(src1final, gray_inv);
    src2.copyTo(src2final, gray);
    Mat finalImage = src1final + src2final;
    // src1 = finalImage;
    ShowMatImgToWnd(GetDlgItem(IDC_PICTURE), finalImage);

    return finalImage;
}


void CCamCalibrationDlg::OnMouseMove(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值 
        switch (m_Pointdownflag)
        {
        case 1:
            m_TheImage.GetWindowRect(m_TheImageRect);//获取显示全景图像所在矩形窗的坐标
            ScreenToClient(m_TheImageRect);         //转换为对话框上的坐标
            point.x -= m_TheImageRect.left;//point获取的是鼠标相对对话框客户区左上角的坐标,减去rect_ctr.left和
            point.y -= m_TheImageRect.top;//rect_ctr.top后,即为鼠标相对Picture控件左上角的坐标


            if (point.x>(m_TheImageRect.right - m_TheImageRect.left) || point.y>(m_TheImageRect.bottom - m_TheImageRect.top) || point.x<0 || point.y<0)
                ;
            else
            {               
                m_pt.Format("x=%f,y=%f", float(point.x), float(point.y));
            }
            UpdateData(false); // 将数据传给控件显示,更新显示
            break;
        case 2:
            m_CamImage.GetWindowRect(m_CamImageRect);//获取显示视频帧图像所在矩形窗
            ScreenToClient(m_CamImageRect);         //转换为对话框上的坐标
            point.x -= m_CamImageRect.left;//point获取的是鼠标相对对话框客户区左上角的坐标,减去rect_ctr.left和
            point.y -= m_CamImageRect.top;//rect_ctr.top后,即为鼠标相对Picture控件左上角的坐标

            if (point.x>(m_CamImageRect.right - m_CamImageRect.left) || point.y>(m_CamImageRect.bottom - m_CamImageRect.top) || point.x < 0 || point.y < 0)
                ;
            else
            {               
                m_pt.Format("x=%f,y=%f", float(point.x), float(point.y));
            }
            UpdateData(false); // 将数据传给控件显示,更新显示
            break;
        case 3:
            m_TheImage.GetWindowRect(m_TheImageRect);//获取显示全景图像所在矩形窗的坐标
            ScreenToClient(m_TheImageRect);         //转换为对话框上的坐标
            point.x -= m_TheImageRect.left;//point获取的是鼠标相对对话框客户区左上角的坐标,减去rect_ctr.left和
            point.y -= m_TheImageRect.top;//rect_ctr.top后,即为鼠标相对Picture控件左上角的坐标


            if (point.x>(m_TheImageRect.right - m_TheImageRect.left) || point.y>(m_TheImageRect.bottom - m_TheImageRect.top) || point.x<0 || point.y<0)
                ;
            else
            {
                m_pt.Format("x=%f,y=%f", float(point.x), float(point.y));
            }
            UpdateData(false); // 将数据传给控件显示,更新显示
            break;
        default:
            break;
        }

    CDialogEx::OnMouseMove(nFlags, point);
}


void CCamCalibrationDlg::OnLButtonDown(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值


    switch (m_Pointdownflag)
    {
    case 1:
        m_TheImage.GetWindowRect(m_TheImageRect);//获取显示全景图像所在矩形窗的坐标
        ScreenToClient(m_TheImageRect);         //转换为对话框上的坐标
        point.x -= m_TheImageRect.left;//point获取的是鼠标相对对话框客户区左上角的坐标,减去rect_ctr.left和
        point.y -= m_TheImageRect.top;//rect_ctr.top后,即为鼠标相对Picture控件左上角的坐标

        if (m_points2.size() <= 3)
        {
            m_points2.push_back(Point2f(float(point.x), float(point.y)));
        }
        else return;
        break;
    case 2:
        m_CamImage.GetWindowRect(m_CamImageRect);//获取显示视频帧图像所在矩形窗
        ScreenToClient(m_CamImageRect);         //转换为对话框上的坐标
        point.x -= m_CamImageRect.left;//point获取的是鼠标相对对话框客户区左上角的坐标,减去rect_ctr.left和
        point.y -= m_CamImageRect.top;//rect_ctr.top后,即为鼠标相对Picture控件左上角的坐标

        if (m_points1.size() <= 3)
        {
            m_points1.push_back(Point2f(float(point.x), float(point.y)));
        }
        else return;
        break;
    case 3:
        m_TheImage.GetWindowRect(m_TheImageRect);//获取显示全景图像所在矩形窗的坐标
        ScreenToClient(m_TheImageRect);         //转换为对话框上的坐标
        point.x -= m_TheImageRect.left;//point获取的是鼠标相对对话框客户区左上角的坐标,减去rect_ctr.left和
        point.y -= m_TheImageRect.top;//rect_ctr.top后,即为鼠标相对Picture控件左上角的坐标
        //Point2f imgpix;
        //imgpix.x = point.x / (m_TheImageRect.right - m_TheImageRect.left) * TheImage.cols;
        //imgpix.y = point.y / (m_TheImageRect.bottom - m_TheImageRect.top) * TheImage.rows;
        switch (m_Radio)
        {
        case 1:
            if (m_pointsh1.size() <= 3)
            {
                m_pointsh1.push_back(Point2f(float(point.x), float(point.y)));
            }
            else return;
            break;
        case 2:
            if (m_pointsh2.size() <= 3)
            {
                m_pointsh2.push_back(Point2f(float(point.x), float(point.y)));
            }
            else return;
            break;
        case 3:
            if (m_pointsh3.size() <= 3)
            {
                m_pointsh3.push_back(Point2f(float(point.x), float(point.y)));
            }
            else return;
            break;
        case 4:
            if (m_pointsh4.size() <= 3)
            {
                m_pointsh4.push_back(Point2f(float(point.x), float(point.y)));
            }
            else return;
            break;
        case 5:
            if (m_pointsh5.size() <= 3)
            {
                m_pointsh5.push_back(Point2f(float(point.x), float(point.y)));
            }
            else return;
            break;
        case 6:
            if (m_pointsh6.size() <= 3)
            {
                m_pointsh6.push_back(Point2f(float(point.x), float(point.y)));
            }
            else return;
            break;
        /*default: break;*/
        }       
    //default:
    // break;
    }

    CDialogEx::OnLButtonDown(nFlags, point);
}





void CCamCalibrationDlg::OnLButtonDblClk(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值
    if (!bFullScreen)
    {
        bFullScreen = true;
        //获取系统屏幕宽高
        int g_iCurScreenWidth = GetSystemMetrics(SM_CXSCREEN);
        int g_iCurScreenHeight = GetSystemMetrics(SM_CYSCREEN);

        //用m_struOldWndpl得到当前窗口的显示状态和窗体位置,以供退出全屏后使用
        GetWindowPlacement(&m_struOldWndpl);
        GetDlgItem(IDC_PICTURE)->GetWindowPlacement(&m_struOldWndpPic);

        //计算出窗口全屏显示客户端所应该设置的窗口大小,主要为了将不需要显示的窗体边框等部分排除在屏幕外
        CRect rectWholeDlg;
        CRect rectClient;
        GetWindowRect(&rectWholeDlg);//得到当前窗体的总的相对于屏幕的坐标
        RepositionBars(0, 0xffff, AFX_IDW_PANE_FIRST, reposQuery, &rectClient);//得到客户区窗口坐标
        ClientToScreen(&rectClient);//将客户区相对窗体的坐标转为相对屏幕坐标
        //GetDlgItem(IDC_STATIC_PICSHOW)->GetWindowRect(rectClient);//得到PICTURE控件坐标

        rectFullScreen.left = rectWholeDlg.left - rectClient.left;
        rectFullScreen.top = rectWholeDlg.top - rectClient.top;
        rectFullScreen.right = rectWholeDlg.right + g_iCurScreenWidth - rectClient.right;
        rectFullScreen.bottom = rectWholeDlg.bottom + g_iCurScreenHeight - rectClient.bottom;

        //设置窗口对象参数,为全屏做好准备并进入全屏状态
        WINDOWPLACEMENT struWndpl;
        struWndpl.length = sizeof(WINDOWPLACEMENT);
        struWndpl.flags = 0;
        struWndpl.showCmd = SW_SHOWNORMAL;
        struWndpl.rcNormalPosition = rectFullScreen;
        SetWindowPlacement(&struWndpl);//该函数设置指定窗口的显示状态和显示大小位置等,是我们该程序最为重要的函数

        //将PICTURE控件的坐标设为全屏大小
        GetDlgItem(IDC_PICTURE)->MoveWindow(CRect(0, 0, g_iCurScreenWidth, g_iCurScreenHeight));


    }
    else
    {
        GetDlgItem(IDC_PICTURE)->SetWindowPlacement(&m_struOldWndpPic);
        SetWindowPlacement(&m_struOldWndpl);
        bFullScreen = false;
    }
    CDialogEx::OnLButtonDblClk(nFlags, point);
}



void CCamCalibrationDlg::OnGetMinMaxInfo(MINMAXINFO* lpMMI)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值
    if (bFullScreen)
    {
        lpMMI->ptMaxSize.x = rectFullScreen.Width();
        lpMMI->ptMaxSize.y = rectFullScreen.Height();
        lpMMI->ptMaxPosition.x = rectFullScreen.left;
        lpMMI->ptMaxPosition.y = rectFullScreen.top;
        lpMMI->ptMaxTrackSize.x = rectFullScreen.Width();
        lpMMI->ptMaxTrackSize.y = rectFullScreen.Height();
    }
    CDialogEx::OnGetMinMaxInfo(lpMMI);
}




void CCamCalibrationDlg::drawpic(IplImage* img, unsigned int id)//CShowPictureInFullScreenDlg 为对话框类名
{
    BYTE *g_pBits;
    HDC g_hMemDC;
    HBITMAP g_hBmp, g_hOldBmp;
    CDC *pDC;
    CStatic *pic;
    int width, height;
    CRect rect;

    pDC = GetDlgItem(id)->GetDC();
    pic = (CStatic*)GetDlgItem(id);
    pic->GetClientRect(&rect);
    width = rect.Width();
    height = rect.Height();

    g_hMemDC = ::CreateCompatibleDC(pDC->m_hDC);//创建兼容DC

    BYTE bmibuf[sizeof(BITMAPINFO) + 256 * sizeof(RGBQUAD)];
    memset(bmibuf, 0, sizeof(bmibuf));
    BITMAPINFO *pbmi = (BITMAPINFO*)bmibuf;

    pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
    pbmi->bmiHeader.biWidth = img->width;
    pbmi->bmiHeader.biHeight = img->height;
    pbmi->bmiHeader.biPlanes = 1;
    pbmi->bmiHeader.biBitCount = 24;
    pbmi->bmiHeader.biCompression = BI_RGB;

    g_hBmp = ::CreateDIBSection(g_hMemDC, pbmi, DIB_RGB_COLORS, (void**)&g_pBits, 0, 0);//创建应用程序可以直接写入的、与设备无关的位图(DIB)
    g_hOldBmp = (HBITMAP)::SelectObject(g_hMemDC, g_hBmp);//复原兼容DC数据
    BitBlt(g_hMemDC, 0, 0, width, height, pDC->m_hDC, 0, 0, SRCCOPY);

    //修改图像内容:g_pBits
    int l_width = WIDTHBYTES(img->width* pbmi->bmiHeader.biBitCount);
    for (int row = 0; row < img->height; row++)
        memcpy(&g_pBits[row*l_width], &img->imageData[(img->height - row - 1)*l_width], l_width);

    TransparentBlt(pDC->m_hDC, 0, 0, width, height, g_hMemDC, 0, 0, img->width, img->height, RGB(0, 0, 0));
    SelectObject(g_hMemDC, g_hOldBmp);

    //释放内存资源
    ReleaseDC(pDC);
    DeleteDC(g_hMemDC);
    DeleteObject(pic);
    DeleteObject(g_hBmp);
    DeleteObject(g_hOldBmp);
}
#ifndef CVVIMAGE_CLASS_DEF 
#define CVVIMAGE_CLASS_DEF 

#ifndef RC_OPENCV_2_1_0 

#include <opencv/cv.h> 
#include <opencv/highgui.h> 

/* CvvImage class definition */
class  CvvImage
{
public:
    CvvImage();
    virtual ~CvvImage();

    /* Create image (BGR or grayscale) */
    virtual bool  Create(int width, int height, int bits_per_pixel, int image_origin = 0);

    /* Load image from specified file */
    virtual bool  Load(const char* filename, int desired_color = 1);

    /* Load rectangle from the file */
    virtual bool  LoadRect(const char* filename,
        int desired_color, CvRect r);

#if defined WIN32 || defined _WIN32 
    virtual bool  LoadRect(const char* filename,
        int desired_color, RECT r)
    {
        return LoadRect(filename, desired_color,
            cvRect(r.left, r.top, r.right - r.left, r.bottom - r.top));
    }
#endif 

    /* Save entire image to specified file. */
    virtual bool  Save(const char* filename);

    /* Get copy of input image ROI */
    virtual void  CopyOf(CvvImage& image, int desired_color = -1);
    virtual void  CopyOf(IplImage* img, int desired_color = -1);

    IplImage* GetImage() { return m_img; };
    virtual void  Destroy(void);

    /* width and height of ROI */
    int Width() { return !m_img ? 0 : !m_img->roi ? m_img->width : m_img->roi->width; };
    int Height() { return !m_img ? 0 : !m_img->roi ? m_img->height : m_img->roi->height; };
    int Bpp() { return m_img ? (m_img->depth & 255)*m_img->nChannels : 0; };

    virtual void  Fill(int color);

    /* draw to highgui window */
    virtual void  Show(const char* window);

#if defined WIN32 || defined _WIN32 
    /* draw part of image to the specified DC */
    virtual void  Show(HDC dc, int x, int y, int width, int height,
        int from_x = 0, int from_y = 0);
    /* draw the current image ROI to the specified rectangle of the destination DC */
    virtual void  DrawToHDC(HDC hDCDst, RECT* pDstRect);
#endif 

protected:

    IplImage*  m_img;
};

typedef CvvImage CImage;

#endif 

#endif 
#include "StdAfx.h"
#include "CvvImage.h"
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
CV_INLINE RECT NormalizeRect(RECT r);
CV_INLINE RECT NormalizeRect(RECT r)
{
    int t;
    if (r.left > r.right)
    {
        t = r.left;
        r.left = r.right;
        r.right = t;
    }
    if (r.top > r.bottom)
    {
        t = r.top;
        r.top = r.bottom;
        r.bottom = t;
    }

    return r;
}
CV_INLINE CvRect RectToCvRect(RECT sr);
CV_INLINE CvRect RectToCvRect(RECT sr)
{
    sr = NormalizeRect(sr);
    return cvRect(sr.left, sr.top, sr.right - sr.left, sr.bottom - sr.top);
}
CV_INLINE RECT CvRectToRect(CvRect sr);
CV_INLINE RECT CvRectToRect(CvRect sr)
{
    RECT dr;
    dr.left = sr.x;
    dr.top = sr.y;
    dr.right = sr.x + sr.width;
    dr.bottom = sr.y + sr.height;

    return dr;
}
CV_INLINE IplROI RectToROI(RECT r);
CV_INLINE IplROI RectToROI(RECT r)
{
    IplROI roi;
    r = NormalizeRect(r);
    roi.xOffset = r.left;
    roi.yOffset = r.top;
    roi.width = r.right - r.left;
    roi.height = r.bottom - r.top;
    roi.coi = 0;

    return roi;
}
void  FillBitmapInfo(BITMAPINFO* bmi, int width, int height, int bpp, int origin)
{
    assert(bmi && width >= 0 && height >= 0 && (bpp == 8 || bpp == 24 || bpp == 32));

    BITMAPINFOHEADER* bmih = &(bmi->bmiHeader);

    memset(bmih, 0, sizeof(*bmih));
    bmih->biSize = sizeof(BITMAPINFOHEADER);
    bmih->biWidth = width;
    bmih->biHeight = origin ? abs(height) : -abs(height);
    bmih->biPlanes = 1;
    bmih->biBitCount = (unsigned short)bpp;
    bmih->biCompression = BI_RGB;
    if (bpp == 8)
    {
        RGBQUAD* palette = bmi->bmiColors;
        int i;
        for (i = 0; i < 256; i++)
        {
            palette[i].rgbBlue = palette[i].rgbGreen = palette[i].rgbRed = (BYTE)i;
            palette[i].rgbReserved = 0;
        }
    }
}
CvvImage::CvvImage()
{
    m_img = 0;
}
void CvvImage::Destroy()
{
    cvReleaseImage(&m_img);
}
CvvImage::~CvvImage()
{
    Destroy();
}
bool  CvvImage::Create(int w, int h, int bpp, int origin)
{
    const unsigned max_img_size = 10000;

    if ((bpp != 8 && bpp != 24 && bpp != 32) ||
        (unsigned)w >= max_img_size || (unsigned)h >= max_img_size ||
        (origin != IPL_ORIGIN_TL && origin != IPL_ORIGIN_BL))
    {
        assert(0); // most probably, it is a programming error
        return false;
    }
    if (!m_img || Bpp() != bpp || m_img->width != w || m_img->height != h)
    {
        if (m_img && m_img->nSize == sizeof(IplImage))
            Destroy();
        /* prepare IPL header */
        m_img = cvCreateImage(cvSize(w, h), IPL_DEPTH_8U, bpp / 8);
    }
    if (m_img)
        m_img->origin = origin == 0 ? IPL_ORIGIN_TL : IPL_ORIGIN_BL;
    return m_img != 0;
}
void  CvvImage::CopyOf(CvvImage& image, int desired_color)
{
    IplImage* img = image.GetImage();
    if (img)
    {
        CopyOf(img, desired_color);
    }
}
#define HG_IS_IMAGE(img) \
   ((img) != 0 && ((const IplImage*)(img))->nSize == sizeof(IplImage) && \
   ((IplImage*)img)->imageData != 0)
void  CvvImage::CopyOf(IplImage* img, int desired_color)
{
    if (HG_IS_IMAGE(img))
    {
        int color = desired_color;
        CvSize size = cvGetSize(img);
        if (color < 0)
            color = img->nChannels > 1;
        if (Create(size.width, size.height,
            (!color ? 1 : img->nChannels > 1 ? img->nChannels : 3) * 8,
            img->origin))
        {
            cvConvertImage(img, m_img, 0);
        }
    }
}
bool  CvvImage::Load(const char* filename, int desired_color)
{
    IplImage* img = cvLoadImage(filename, desired_color);
    if (!img)
        return false;

    CopyOf(img, desired_color);
    cvReleaseImage(&img);

    return true;
}
bool  CvvImage::LoadRect(const char* filename,
    int desired_color, CvRect r)
{
    if (r.width < 0 || r.height < 0) return false;

    IplImage* img = cvLoadImage(filename, desired_color);
    if (!img)
        return false;
    if (r.width == 0 || r.height == 0)
    {
        r.width = img->width;
        r.height = img->height;
        r.x = r.y = 0;
    }
    if (r.x > img->width || r.y > img->height ||
        r.x + r.width < 0 || r.y + r.height < 0)
    {
        cvReleaseImage(&img);
        return false;
    }
    /* truncate r to source image */
    if (r.x < 0)
    {
        r.width += r.x;
        r.x = 0;
    }
    if (r.y < 0)
    {
        r.height += r.y;
        r.y = 0;
    }
    if (r.x + r.width > img->width)
        r.width = img->width - r.x;

    if (r.y + r.height > img->height)
        r.height = img->height - r.y;
    cvSetImageROI(img, r);
    CopyOf(img, desired_color);
    cvReleaseImage(&img);
    return true;
}
bool  CvvImage::Save(const char* filename)
{
    if (!m_img)
        return false;
    cvSaveImage(filename, m_img);
    return true;
}
void  CvvImage::Show(const char* window)
{
    if (m_img)
        cvShowImage(window, m_img);
}
void  CvvImage::Show(HDC dc, int x, int y, int w, int h, int from_x, int from_y)
{
    if (m_img && m_img->depth == IPL_DEPTH_8U)
    {
        uchar buffer[sizeof(BITMAPINFOHEADER) + 1024];
        BITMAPINFO* bmi = (BITMAPINFO*)buffer;
        int bmp_w = m_img->width, bmp_h = m_img->height;
        FillBitmapInfo(bmi, bmp_w, bmp_h, Bpp(), m_img->origin);
        from_x = MIN(MAX(from_x, 0), bmp_w - 1);
        from_y = MIN(MAX(from_y, 0), bmp_h - 1);
        int sw = MAX(MIN(bmp_w - from_x, w), 0);
        int sh = MAX(MIN(bmp_h - from_y, h), 0);
        SetDIBitsToDevice(
            dc, x, y, sw, sh, from_x, from_y, from_y, sh,
            m_img->imageData + from_y*m_img->widthStep,
            bmi, DIB_RGB_COLORS);
    }
}
void  CvvImage::DrawToHDC(HDC hDCDst, RECT* pDstRect)
{
    if (pDstRect && m_img && m_img->depth == IPL_DEPTH_8U && m_img->imageData)
    {
        uchar buffer[sizeof(BITMAPINFOHEADER) + 1024];
        BITMAPINFO* bmi = (BITMAPINFO*)buffer;
        int bmp_w = m_img->width, bmp_h = m_img->height;
        CvRect roi = cvGetImageROI(m_img);
        CvRect dst = RectToCvRect(*pDstRect);
        if (roi.width == dst.width && roi.height == dst.height)
        {
            Show(hDCDst, dst.x, dst.y, dst.width, dst.height, roi.x, roi.y);
            return;
        }
        if (roi.width > dst.width)
        {
            SetStretchBltMode(
                hDCDst,           // handle to device context
                HALFTONE);
        }
        else
        {
            SetStretchBltMode(
                hDCDst,           // handle to device context
                COLORONCOLOR);
        }
        FillBitmapInfo(bmi, bmp_w, bmp_h, Bpp(), m_img->origin);
        ::StretchDIBits(
            hDCDst,
            dst.x, dst.y, dst.width, dst.height,
            roi.x, roi.y, roi.width, roi.height,
            m_img->imageData, bmi, DIB_RGB_COLORS, SRCCOPY);
    }
}
void  CvvImage::Fill(int color)
{
    cvSet(m_img, cvScalar(color & 255, (color >> 8) & 255, (color >> 16) & 255, (color >> 24) & 255));
}

你可能感兴趣的:(视频,mfc,opencv)