BitBlt игнорирует CAPTUREBLT и, кажется, всегда захватывает кешированную копию цели.

Я пытаюсь сделать снимки экрана с помощью функции 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;
}

Большая часть этого кода взята из функции QWidget::grabWindow Qt, так как я хотел внести некоторые изменения, чтобы сделать ее более гибкой. В документации Qt указано, что:

Функция grabWindow() захватывает пиксели с экрана, а не из окна, т. е. если есть другое окно, частично или полностью поверх того, которое вы захватываете, вы также получаете пиксели из вышележащего окна.

Однако я испытываю прямо противоположное... независимо от флага CAPTUREBLT. Я пробовал все, что мог придумать... ничего не работает. Любые идеи?


person Jake Petroules    schedule 24.05.2010    source источник


Ответы (2)


Вы запутались в поведении BitBlt с поведением CAPTUREBLT из-за того, что официальная документация BitBlt неясна и вводит в заблуждение.

В нем указано, что
"CAPTUREBLT – включает все окна, расположенные поверх вашего окна, в результирующее изображение. По умолчанию изображение содержит только ваше окно".

Что на самом деле означает (по крайней мере, для любой ОС Windows без поддержки Aero) "CAPTUREBLT -- Включает все многослойные (!) окна (см. расширенный стиль окна WS_EX_LAYERED), которые перекрывают ваше окно. Неслоистые окна, которые перекрывают ваше окно, никогда не включены."

Windows без расширенного стиля окна WS_EX_LAYERED, перекрывающего ваше окно, не включается с флагом CAPTUREBLT или без него (по крайней мере, для любой ОС Windows без включенного Aero).

Разработчики QT также неправильно поняли документацию BitBlt/CAPTUREBLT, поэтому документация QT на самом деле неверна в отношении поведения QPixmap::grabWindow на платформе WIN32 без включенного Aero.

ДОБАВИТЬ:

Если вы хотите захватить свое окно, как оно есть на экране, вы должны захватить весь рабочий стол с флагом CAPTUREBLT, а затем извлечь прямоугольник с вашим окном. (Разработчики QT должны делать то же самое). Он будет корректно работать в обоих случаях: с включенным/доступным Aero и без него.

person Serge Dundich    schedule 10.01.2011

Я захватываю весь экран и получаю те же результаты... :(

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);
person Fernando Diaz Toledano    schedule 05.11.2013