Joh*_*ker 5 c++ winapi screenshot dwm bitblt
我的程序正在截取其他应用程序窗口的屏幕截图,以自动执行其中的某些任务。这些窗口可能会隐藏在屏幕外或不时被其他窗口遮挡。
为了减少混乱,我从代码列表中删除了所有错误检查。我正在准备两种类型的屏幕截图
// Get size of the target window.
RECT clientRect;
GetClientRect(hwnd, &clientRect);
int width = clientRect.right - clientRect.left;
int height = clientRect.bottom - clientRect.top;
// Create a memory device context.
HDC windowDC = GetDC(hwnd);
HDC memoryDC = CreateCompatibleDC(windowDC);
// Create a bitmap for rendering.
BITMAPINFO bitmapInfo;
ZeroMemory(&bitmapInfo, sizeof(BITMAPINFO));
bitmapInfo.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
bitmapInfo.bmiHeader.biWidth = width;
bitmapInfo.bmiHeader.biHeight = -height;
bitmapInfo.bmiHeader.biPlanes = 1;
bitmapInfo.bmiHeader.biBitCount = 32;
bitmapInfo.bmiHeader.biCompression = BI_RGB;
bitmapInfo.bmiHeader.biSizeImage = width * height * 4;
RGBQUAD* buffer;
HBITMAP bitmap = CreateDIBSection(windowDC, &bitmapInfo, DIB_RGB_COLORS, (void**)&buffer, 0, 0);
HGDIOBJ previousObject = SelectOBject(memoryDC, bitmap);
Run Code Online (Sandbox Code Playgroud)
然后我用以下任一方法拍摄实际的屏幕截图
PrintWindow(hwnd, memoryDC, PW_CLIENTONLY);
GdiFlush();
Run Code Online (Sandbox Code Playgroud)
或者
UpdateWindow(hwnd);
BitBlt(memoryDC, 0, 0, width, height, windowDC, 0, 0, SRCCOPY);
GdiFlush();
Run Code Online (Sandbox Code Playgroud)
然后我将缓冲区内容复制到向量中
std::vector<RGBQUAD> pixels;
pixels.resize(width * height, { 0, 0, 0, 0 });
memcpy(&pixels[0], buffer, bitmapInfo.bmiHeader.biSizeImage);
Run Code Online (Sandbox Code Playgroud)
最后我清理了一切
ReleaseDC(hwnd, windowDC);
SelectObject(memoryDC, previousObject);
DeleteDC(memoryDC);
DeleteObject(bitmap);
Run Code Online (Sandbox Code Playgroud)
我对上面的代码有三个问题:
GdiFlush()?我阅读了文档并做了一些谷歌研究,但我仍然不确定调用它是否有意义。UpdateWindow()在之前的调用有什么BitBlt()不同吗?这是否有帮助,使设备上下文内容“更新”?PrintWindow()所有工作都是从目标应用程序内部完成的(增加了目标应用程序的 CPU 使用率),而BitBlt()完全是从调用线程内部执行的(从而增加了我自己的应用程序的 CPU 使用率)?BitBlt()如果启用了桌面组合 (DWM),则该功能仅适用于隐藏窗口,但在一小部分系统 (Windows 8/10) 上,BitBlt()并且PrintWindow()对于在其他系统上运行良好的窗口似乎会失败(即使 DWM已启用)。但我找不到任何模式来解释原因。任何信息表示赞赏,谢谢。
小智 5
最后,经过几个小时的调查,我找到了该问题的解决方案:在要成像的表单的 ACTIVATE 事件中调用以下命令就足够了(VB 编码中的示例):
Call SetWindowLong(me.hwnd, GWL_EXSTYLE, WS_EX_LAYERED)
Run Code Online (Sandbox Code Playgroud)
而该命令的定义如下:
Private Declare Function SetWindowLong Lib "User32" Alias "SetWindowLongA" (ByVal hwnd As Long, ByVal nIndex As Long, ByVal dwNewLong As Long) As Long
Private Const GWL_EXSTYLE = (-20)
Private Const WS_EX_LAYERED = &H80000
Run Code Online (Sandbox Code Playgroud)
请尝试这个!
最好的问候,斑比66