隐藏窗口的PrintWindow和BitBlt

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)

我对上面的代码有三个问题:

  1. 我需要打电话吗GdiFlush()?我阅读了文档并做了一些谷歌研究,但我仍然不确定调用它是否有意义。
  2. UpdateWindow()在之前的调用有什么BitBlt()不同吗?这是否有帮助,使设备上下文内容“更新”?
  3. 我是否正确地假设PrintWindow()所有工作都是从目标应用程序内部完成的(增加了目标应用程序的 CPU 使用率),而BitBlt()完全是从调用线程内部执行的(从而增加了我自己的应用程序的 CPU 使用率)?
  4. 在什么情况下上述功能可能会失败?我知道,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

  • 不幸的是,这并不适用于所有窗口。设置“WS_EX_LAYERED”后,窗口不再更新。起初我以为是因为窗口使用了`CS_CLASSDC`或`CS_OWNDC`,但是这些样式没有设置。我尝试调用“SetLayeredWindowAttributes()”但没有成功。有任何想法吗? (2认同)