8

我正在尝试使用 BitBlt 函数捕获屏幕截图。但是,每次我截取屏幕截图时,无论我做什么,非客户区都不会改变。就好像它正在获取它的一些缓存副本。客户区被正确捕获。

如果我关闭然后重新打开窗口并截屏,非客户区将被原样捕获。移动/调整窗口大小后的任何后续捕获都不会影响捕获的屏幕截图。同样,客户区将是正确的。

此外,CAPTUREBLT 标志似乎完全没有做任何事情。无论有没有它,我都没有注意到任何变化。这是我的捕获代码:

QPixmap WindowManagerUtils::grabWindow(WId windowId, GrabWindowFlags flags, int x, int y, int w, int h)
{
    RECT r;

    switch (flags)
    {
        case WindowManagerUtils::GrabWindowRect:
            GetWindowRect(windowId, &r);
            break;
        case WindowManagerUtils::GrabClientRect:
            GetClientRect(windowId, &r);
            break;
        case WindowManagerUtils::GrabScreenWindow:
            GetWindowRect(windowId, &r);
            return QPixmap::grabWindow(QApplication::desktop()->winId(), r.left, r.top, r.right - r.left, r.bottom - r.top);
        case WindowManagerUtils::GrabScreenClient:
            GetClientRect(windowId, &r);
            return QPixmap::grabWindow(QApplication::desktop()->winId(), r.left, r.top, r.right - r.left, r.bottom - r.top);
        default:
            return QPixmap();
    }

    if (w < 0)
    {
        w = r.right - r.left;
    }

    if (h < 0)
    {
        h = r.bottom - r.top;
    }

#ifdef Q_WS_WINCE_WM
    if (qt_wince_is_pocket_pc())
    {
        QWidget *widget = QWidget::find(winId);
        if (qobject_cast<QDesktopWidget*>(widget))
        {
            RECT rect = {0,0,0,0};
            AdjustWindowRectEx(&rect, WS_BORDER | WS_CAPTION, FALSE, 0);
            int magicNumber = qt_wince_is_high_dpi() ? 4 : 2;
            y += rect.top - magicNumber;
        }
    }
#endif

    // Before we start creating objects, let's make CERTAIN of the following so we don't have a mess
    Q_ASSERT(flags == WindowManagerUtils::GrabWindowRect || flags == WindowManagerUtils::GrabClientRect);

    // Create and setup bitmap
    HDC display_dc = NULL;
    if (flags == WindowManagerUtils::GrabWindowRect)
    {
        display_dc = GetWindowDC(NULL);
    }
    else if (flags == WindowManagerUtils::GrabClientRect)
    {
        display_dc = GetDC(NULL);
    }

    HDC bitmap_dc = CreateCompatibleDC(display_dc);
    HBITMAP bitmap = CreateCompatibleBitmap(display_dc, w, h);
    HGDIOBJ null_bitmap = SelectObject(bitmap_dc, bitmap);

    // copy data
    HDC window_dc = NULL;
    if (flags == WindowManagerUtils::GrabWindowRect)
    {
        window_dc = GetWindowDC(windowId);
    }
    else if (flags == WindowManagerUtils::GrabClientRect)
    {
        window_dc = GetDC(windowId);
    }

    DWORD ropFlags = SRCCOPY;
#ifndef Q_WS_WINCE
    ropFlags = ropFlags | CAPTUREBLT;
#endif

    BitBlt(bitmap_dc, 0, 0, w, h, window_dc, x, y, ropFlags);

    // clean up all but bitmap
    ReleaseDC(windowId, window_dc);
    SelectObject(bitmap_dc, null_bitmap);
    DeleteDC(bitmap_dc);

    QPixmap pixmap = QPixmap::fromWinHBITMAP(bitmap);

    DeleteObject(bitmap);
    ReleaseDC(NULL, display_dc);

    return pixmap;
}

大部分代码来自 Qt 的 QWidget::grabWindow 函数,因为我想进行一些更改以使其更加灵活。Qt 的文档指出:

grabWindow() 函数从屏幕而不是窗口中抓取像素,也就是说,如果在您抓取的窗口上部分或全部有另一个窗口,您也会从上层窗口中获取像素。

但是,我的经历完全相反……不管 CAPTUREBLT 标志如何。我已经尝试了我能想到的一切......没有任何效果。有任何想法吗?

4

2 回答 2

7

您对BitBltCAPTUREBLT行为的混淆来自于官方BitBlt文档不清楚和误导的事实。

它声明
“CAPTUREBLT - 在结果图像中包括任何叠加在窗口顶部的窗口。默认情况下,图像仅包含您的窗口。”

实际意味着什么(至少对于没有启用 Aero 的任何 Windows 操作系统) “CAPTUREBLT - 包括与您的窗口重叠的任何分层(!)窗口(请参阅 WS_EX_LAYERED 扩展窗口样式)。与您的窗口重叠的非分层窗口永远不会包括在内。”

不包含与窗口重叠的WS_EX_LAYERED扩展窗口样式的Windows不包含或不包含CAPTUREBLT标志(至少对于任何未启用 Aero 的 Windows 操作系统)。

QT 开发人员还误解了 BitBlt/CAPTUREBLT 文档,因此 QT 文档实际上是错误的关于 QPixmap::grabWindow 在没有启用 Aero 的 WIN32 平台上的行为。

添加:

如果要捕获屏幕上的窗口,则必须使用 CAPTUREBLT 标志捕获整个桌面,然后使用窗口提取矩形。(QT 开发人员应该做同样的事情)。它在两种情况下都可以正常工作:启用和不启用 Aero/可用。

于 2011-01-10T12:51:17.330 回答
0

我捕获所有屏幕并获得相同的结果... :(

const uint SRCCOPY = 0x00CC0020; //SRCCOPY
    const uint CAPTUREBLT = 0x00CC0020 | 0x40000000; //CAPTUREBLT

    bool dv = BitBlt(hCaptureDC, 0, 0, Bounds.Width, Bounds.Height,
             hDesktopDC, Bounds.Left, Bounds.Top, _with_tooltips ? CAPTUREBLT : SRCCOPY);
于 2013-11-05T09:57:11.860 回答