【问题标题】:OPENCV desktop capture Part IIOPENCV 桌面捕获第二部分
【发布时间】:2016-04-19 05:37:20
【问题描述】:

从这里OPENCV Destop Capture 和这个hwnd2mat 函数继续上一篇文章,在opencv 中捕获桌面作为源的方法是使用这个hwnd2mat 函数从屏幕图像创建一个位图。完毕。 此功能已经从屏幕图像创建位图。但它会产生奇怪的效果,就像视频中的尾随一样,我只想要一个更可能像 this one 这样的普通视频源。我已经尝试找出是什么原因造成的,但我不知道了。

#include "opencv2/imgproc.hpp"
#include "opencv2/highgui.hpp"
#include <Windows.h>
#include <iostream>

using namespace std;
using namespace cv;

Mat hwnd2mat(HWND hwnd)
{
    HDC hwindowDC, hwindowCompatibleDC;

    int height, width, srcheight, srcwidth;
    HBITMAP hbwindow;
    Mat src;
    BITMAPINFOHEADER  bi;

    hwindowDC = GetDC(hwnd);
    hwindowCompatibleDC = CreateCompatibleDC(hwindowDC);
    SetStretchBltMode(hwindowCompatibleDC, COLORONCOLOR);

    RECT windowsize;    // get the height and width of the screen
    GetClientRect(hwnd, &windowsize);

    srcheight = windowsize.bottom;
    srcwidth = windowsize.right;
    height = windowsize.bottom / 1;  //change this to whatever size you want to resize to
    width = windowsize.right / 1;

    src.create(height, width, CV_8UC4);

    // create a bitmap
    hbwindow = CreateCompatibleBitmap(hwindowDC, width, height);
    bi.biSize = sizeof(BITMAPINFOHEADER);    //http://msdn.microsoft.com/en-us/library/windows/window/dd183402%28v=vs.85%29.aspx
    bi.biWidth = width;
    bi.biHeight = -height;  //this is the line that makes it draw upside down or not
    bi.biPlanes = 1;
    bi.biBitCount = 32;
    bi.biCompression = BI_RGB;
    bi.biSizeImage = 0;
    bi.biXPelsPerMeter = 0;
    bi.biYPelsPerMeter = 0;
    bi.biClrUsed = 0;
    bi.biClrImportant = 0;

    // use the previously created device context with the bitmap
    SelectObject(hwindowCompatibleDC, hbwindow);
    // copy from the window device context to the bitmap device context
    StretchBlt(hwindowCompatibleDC, 0, 0, width, height, hwindowDC, 0, 0, srcwidth, srcheight, SRCCOPY); //change SRCCOPY to NOTSRCCOPY for wacky colors !
    GetDIBits(hwindowCompatibleDC, hbwindow, 0, height, src.data, (BITMAPINFO *)&bi, DIB_RGB_COLORS);  //copy from hwindowCompatibleDC to hbwindow

    // avoid memory leak
    DeleteObject(hbwindow);
    DeleteDC(hwindowCompatibleDC);
    ReleaseDC(hwnd, hwindowDC);

    return src;
}

int main(int argc, char **argv)
{
    HWND hwndDesktop = GetDesktopWindow();
    namedWindow("output", CV_WINDOW_NORMAL);
    int key = 0;

    while (key != 30 )
    {
        Mat src = hwnd2mat(hwndDesktop);        
        // you can do some image processing here
        imshow("output", src);
        key = waitKey(30); // you can change wait time
    }
    ReleaseCapture();
    destroyAllWindows();
    return 0;
}

【问题讨论】:

  • 奇怪的效果是这样的:youtube.com/watch?v=RQePoFEUS3A&feature=youtu.be,你们能告诉我发生了什么吗
  • 所以您不想捕获整个屏幕而只捕获一个窗口?显然,如果您捕获整个屏幕,您确实会再次捕获输出图像,从而导致无限镜像效果。而不是GetDesktopWindow,您应该访问/定位您要捕获的窗口的句柄。
  • @Micka 好的,我明白了,我应该找到要捕获的窗口的句柄,但它是什么而不是 GetDesktopWindow
  • 您要捕获哪个窗口?如果您知道名称,请使用FindWindow msdn.microsoft.com/de-de/library/windows/desktop/… 或者如果您可以设法使所需窗口成为活动窗口,则可以使用GetActiveWindow。如果您想捕获整个屏幕,也许您可​​以在调用 GetDesktopWindow 之前隐藏输出窗口,然后再显示它。
  • @Micka 最后!直到现在凌晨 2 点 31 分,我明白这是怎么回事,你是对的,米卡!谢谢你!你是我的英雄。!我得到了我想要的结果!不仅如此,我现在理解了窗口句柄的逻辑! :)

标签: c++ opencv visual-studio-2013 bitmap video-capture


【解决方案1】:

使用 GetDesktopWindow() 将产生无限镜像效果,而不是使用您要捕获的窗口,而我,我正在使用 FindWindowEx(),并使用 Spy++ 来查找您要捕获的窗口,应该作品。但当然它会转到下一个错误:)。

【讨论】:

    猜你喜欢
    • 2016-03-31
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-05-16
    • 2023-04-08
    • 2016-01-04
    • 2013-09-30
    • 1970-01-01
    相关资源
    最近更新 更多