如何在MFC视图上显示OpenCV Mat

How to display OpenCV Mat on MFC View

本文关键字:显示 OpenCV Mat 视图 MFC      更新时间:2023-10-16

我认为在MFC视图上显示OpenCV2 Mat很简单,但事实并非如此。这只是我在谷歌上找到的相关资料。请原谅我的无知,但我找不到任何其他材料显示如何使用一维数组"数据"成员返回SetDIBitsToDevice。更具体地说,我需要知道如何为函数指定BITMAPINFO。我回到旧的c风格的OpenCV与MFC工作吗?

更新:

我发现了一个SetDIBitsToDevice的例子,它实际上是旧的c风格的OpenCV。但是将它转换为OpenCV2是很简单的。为了使它起作用,我需要提到一些事情:

  1. Bpp方法不工作,因为Mat的深度返回0。我只是像这样修改:

    static int Bpp(cv::Mat img) { return 8 * img.channels(); } 
    
  2. Mat没有origin成员。但是对于FillBitmapInfo方法的origin参数,只要输入0就可以了

除此之外,下面的代码工作得很好。希望这对其他开发者也有帮助。

void COpenCVTestView::OnDraw(CDC* pDC)
{
COpenCVTestDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
    return;
if(pDoc->m_cvImage.empty()) return;
// TODO: add draw code for native data here
int height=pDoc->m_cvImage.rows;
int width=pDoc->m_cvImage.cols;
uchar buffer[sizeof( BITMAPINFOHEADER ) + 1024]; 
BITMAPINFO* bmi = (BITMAPINFO* )buffer; 
FillBitmapInfo(bmi,width,height,Bpp(pDoc->m_cvImage),0);
SetDIBitsToDevice(pDC->GetSafeHdc(), 0, 0, width,
    height, 0, 0, 0, height, pDoc->m_cvImage.data, bmi,
    DIB_RGB_COLORS);
}
void COpenCVTestView::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; 
            for (int i = 0; i < 256; i++) 
    { 
        palette[i].rgbBlue = palette[i].rgbGreen = palette[i].rgbRed = (BYTE)i; 
        palette[i].rgbReserved = 0; 
    } 
} 
}

这是在MFC中显示OpenCV数据的另一种可能的方法,我使用并且效果很好:

IplImage* image// <-- this contains the image you want to display
CvvImage tempdefault;
RECT myrect;   // <-- specifiy where on the screen you want it to be displayed
myrect.top    =  0;
myrect.bottom = _pictureh;
myrect.left   = _picturex;
myrect.right =  _picturew+_picturex;
tempdefault.Create(_pictureh,_picturew,32);
tempdefault.CopyOf(image);
tempdefault.DrawToHDC(pDC->GetSafeHdc(),&myrect);

From MSDN:

lpvBits[在]指向以字节数组形式存储的颜色数据的指针。有关更多信息,请参阅以下"备注"部分。

这个指针必须用Mat::data返回的数据初始化。

CvvImage在新版本的OpenCV中不可用。使用以下代码,您可以将Mat转换为CImage,然后在您想要的任何地方显示CImage:

int Mat2CImage(Mat *mat, CImage &img){
  if(!mat || mat->empty())
    return -1;
  int nBPP = mat->channels()*8;
  img.Create(mat->cols, mat->rows, nBPP);
  if(nBPP == 8)
  {
    static RGBQUAD pRGB[256];
    for (int i = 0; i < 256; i++)
        pRGB[i].rgbBlue = pRGB[i].rgbGreen = pRGB[i].rgbRed = i;
    img.SetColorTable(0, 256, pRGB);
  }
  uchar* psrc = mat->data;
  uchar* pdst = (uchar*) img.GetBits();
  int imgPitch = img.GetPitch();
  for(int y = 0; y < mat->rows; y++)
  {
    memcpy(pdst, psrc, mat->cols*mat->channels());//mat->step is incorrect for those images created by roi (sub-images!)
    psrc += mat->step;
    pdst += imgPitch;
  }
  return 0;
}