0

我有一个使用 opencv 的相机应用程序,而我正在使用 MFC GUI。我需要将 CImage 转换为 IplImage 以供 Opencv 模块进行图像处理,并将其转换回 CImage 以便再次在窗口中显示。

我研究了这个主题,但没有足够的例子和解决方案。任何人有一些建议。谢谢

这是我的代码...

void CChildView::OnFileOpenimage(void)
{
    // TODO: Add your command handler code here
    CString strFilter;
    CSimpleArray<GUID> aguidFileTypes;
    HRESULT hResult;
hResult = imgOriginal.GetExporterFilterString(strFilter,aguidFileTypes);
if (FAILED(hResult)) {
    CString fmt;
    fmt.Format("GetExporterFilter failed:\n%x - %s", hResult, _com_error(hResult).ErrorMessage());
    ::AfxMessageBox(fmt);
    return;
}

CFileDialog dlg(TRUE, NULL, NULL, OFN_FILEMUSTEXIST, strFilter);
dlg.m_ofn.nFilterIndex = m_nFilterLoad;
hResult = (int)dlg.DoModal();
if(FAILED(hResult)) {
    return;
}

m_nFilterLoad = dlg.m_ofn.nFilterIndex;
imgOriginal.Destroy();
CString pathval =  dlg.GetPathName();
hResult = imgOriginal.Load(dlg.GetPathName());
if (FAILED(hResult)) {
    CString fmt;
    fmt.Format("Load image failed:\n%x - %s", hResult, _com_error(hResult).ErrorMessage());
    ::AfxMessageBox(fmt);
    return;
}
// IplImage *img from  imgOriginal; want to convert here for further processing.

m_nImageSize = SIZE_ORIGINAL;
Invalidate();
UpdateWindow();

}
4

3 回答 3

1

我自己有这个问题很长一段时间。我在这里找到了这段代码,我最终使用它效果很好。而不是研究“CImage to IplImage”,您应该搜索“HBITMAP to IplImage”

IplImage* hBitmap2Ipl(HBITMAP hBmp, bool flip)
{
    BITMAP bmp;
    ::GetObject(hBmp,sizeof(BITMAP),&bmp);

    int    nChannels = bmp.bmBitsPixel == 1 ? 1 : bmp.bmBitsPixel/8;
    int    depth     = bmp.bmBitsPixel == 1 ? IPL_DEPTH_1U : IPL_DEPTH_8U;

    IplImage *img=cvCreateImageHeader(cvSize(bmp.bmWidth, bmp.bmHeight), depth, nChannels);

    img->imageData = (char*)malloc(bmp.bmHeight*bmp.bmWidth*nChannels*sizeof(char));
    memcpy(img->imageData,(char*)(bmp.bmBits),bmp.bmHeight*bmp.bmWidth*nChannels);
    return img;

}

用法:

ATL::CImage image;
//whatever code you use to generate the image goes here
IplImage *convertedImage=hBitmap2Ipl(image.Detach());
image.Destroy();
于 2012-12-27T00:26:41.987 回答
0

找了半夜的复制粘贴内存泄露,终于找到了!

所以这是原始答案的更新代码片段。对我来说重要的是使用cvCreateImage而不是cvCreateImageHeader因为在调用hBitmap2IplImage之后我确实使用了cvReleaseImage。在我的情况下,翻转也是必要的。干杯!

IplImage* hBitmap2IplImage(HBITMAP hBmp, bool flip)
{
    BITMAP bmp;
    ::GetObject(hBmp,sizeof(BITMAP),&bmp);

    int    nChannels = bmp.bmBitsPixel == 1 ? 1 : bmp.bmBitsPixel/8;
    int    depth     = bmp.bmBitsPixel == 1 ? IPL_DEPTH_1U : IPL_DEPTH_8U;

    IplImage *img=cvCreateImage(cvSize(bmp.bmWidth, bmp.bmHeight), depth, nChannels);

    size_t imgSize = bmp.bmHeight*bmp.bmWidth*nChannels;
    memcpy_s(img->imageData, imgSize, (char*)(bmp.bmBits), imgSize);
    if (flip)
        cvFlip(img, NULL, 0);

    return img;
}

用法:

ATL::CImage image;
HBITMAP hBitmap = image.Detach(); // some handle to your bitmap
bool flip = true;
IplImage* pSomeIplImage = hBitmap2IplImage(hBitmap, flip);
image.Destroy();
//
// use "pSomeIplImage"
//
cvReleaseImage( &pSomeIplImage ); // don't forget
于 2017-09-28T00:05:56.253 回答
-1

如果位图图像在视图上,您​​可以使用设备上下文(CDC dcBuffer)来复制它。

IplImage *pCapture = cvCreateImage(cvSize(U16_IMAGE_WIDTH/2,U16_IMAGE_HEIGHT/2),8, 3);

::GetDIBits(dcBuffer.GetSafeHdc(), bitmapBuffer, 0, U16_IMAGE_HEIGHT/2, pCapture->imageData, &bitmapInfo, DIB_RGB_COLORS);

cvSaveImage("temp.bmp", pCapture);

cvReleaseImage(&pCapture);
于 2015-11-30T10:22:45.573 回答