GDI GradientFill не работает на закадровом изображении - PullRequest
1 голос
/ 27 января 2010

Я пытаюсь использовать функцию GDI GradientFill для рисования за пределами экрана, затем BitBlt это на экран.

Но я всегда получаю черное растровое изображение ... если я GradientFill прямо к экрану, это работает.

Ниже приведен пример приложения, чтобы понять, что я имею в виду.

#pragma comment(lib, "msimg32.lib")
#include <windows.h>

const CHAR c_szWndClass[] = "GradientTestWnd";
const CHAR c_szWndTitle[] = "GradientTest";
const int c_nWndWidth = 1024;
const int c_nWndHeight = 768;

int WINAPI WinMain(      
                   HINSTANCE hInstance,
                   HINSTANCE hPrevInstance,
                   LPSTR lpCmdLine,
                   int nCmdShow
                   )
{
    WNDCLASSEX wcx;
    ZeroMemory(&wcx, sizeof(wcx));
    wcx.cbSize = sizeof(wcx);
    wcx.style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC;
    wcx.lpfnWndProc = DefWindowProc;
    wcx.hInstance = hInstance;
    wcx.lpszClassName = c_szWndClass;

    RegisterClassEx(&wcx);

    HWND hwndMain = CreateWindowEx(
        0,
        c_szWndClass,
        c_szWndTitle,
        WS_OVERLAPPEDWINDOW,
        CW_USEDEFAULT,
        CW_USEDEFAULT,
        c_nWndWidth,
        c_nWndHeight,
        NULL,
        NULL,
        hInstance,
        NULL);

    ShowWindow(hwndMain, SW_SHOW);

    HDC hdc;
    hdc = GetDC(hwndMain);

    HDC hdcOffscreen = CreateCompatibleDC(hdc);
    HBITMAP bitmap = CreateCompatibleBitmap(hdcOffscreen, c_nWndWidth, c_nWndHeight);
    HBITMAP old_bitmap = (HBITMAP) SelectObject(hdcOffscreen, bitmap);

    TRIVERTEX vertices[2];
    ZeroMemory(&vertices, sizeof(vertices));
    vertices[0].Red = 0xFF00;
    vertices[0].Green = 0x0000;
    vertices[0].Blue = 0x0000;
    vertices[0].x = 0;
    vertices[0].y = 0;

    vertices[1].Red = 0x0000;
    vertices[1].Green = 0x0000;
    vertices[1].Blue = 0xFF00;
    vertices[1].x = c_nWndWidth;
    vertices[1].y = c_nWndHeight;

    GRADIENT_RECT rects[1];
    ZeroMemory(&rects, sizeof(rects));
    rects[0].UpperLeft = 0;
    rects[0].LowerRight = 1;

    // This works
    //GradientFill(hdc, vertices, 2, rects, 1, GRADIENT_FILL_RECT_V);

    // This doesn't
    GradientFill(hdcOffscreen, vertices, 2, rects, 1, GRADIENT_FILL_RECT_V);
    BitBlt(hdc, 0, 0, c_nWndWidth, c_nWndHeight, hdcOffscreen, 0, 0, SRCCOPY);

    Sleep(5000);

    SelectObject(hdcOffscreen, old_bitmap);
    DeleteObject(bitmap);
    DeleteDC(hdcOffscreen);

    return 0;
}

Ответы [ 2 ]

4 голосов
/ 18 октября 2010

проблема здесь фактически связана с начальным состоянием контекста устройства, из которого вы создаете совместимое растровое изображение - в этой строке:

HBITMAP bitmap = CreateCompatibleBitmap(hdcOffscreen, c_nWndWidth, c_nWndHeight);

Вместо этого hdcOffscreen должен быть hdc - это потому, что контекст устройства, созданный здесь:

HDC hdcOffscreen = CreateCompatibleDC(hdc);

по умолчанию выбрал монохромное растровое изображение 1x1 - при попытке создать из него совместимое растровое изображение вы также получите монохромное растровое изображение. так что если вы сделаете это вместо:

HBITMAP bitmap = CreateCompatibleBitmap(hdc, c_nWndWidth, c_nWndHeight);

Вы должны увидеть свой градиент :) старый вопрос, на который, похоже, дан ответ, но я подумал, что id просто поможет уточнить, почему он не работает!

подробности / ссылки:

http://msdn.microsoft.com/en-us/library/dd183489%28VS.85%29.aspx

Когда создается память DC, ее поверхность дисплея ровно одна монохромный пиксель в ширину и один монохромный пиксель высотой

http://msdn.microsoft.com/en-us/library/dd183488%28v=VS.85%29.aspx

Цветовой формат созданного растрового изображения с помощью функции CreateCompatibleBitmap соответствует цветовому формату устройства определяется параметром hdc

чч:)

1 голос
/ 08 февраля 2010

Создайте DIB вместо совместимого растрового изображения.

Заменить

HBITMAP bitmap = CreateCompatibleBitmap(hdcOffscreen, c_nWndWidth, c_nWndHeight);

с

BITMAPINFO BitmapInfo;
memset(&BitmapInfo, 0, sizeof(BITMAPINFOHEADER));
BitmapInfo.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
BitmapInfo.bmiHeader.biWidth = c_nWndWidth;
BitmapInfo.bmiHeader.biHeight = c_nWndHeight;
BitmapInfo.bmiHeader.biPlanes = 1;
BitmapInfo.bmiHeader.biBitCount = 32;
BitmapInfo.bmiHeader.biCompression = BI_RGB;
HBITMAP bitmap = CreateDIBSection(hdcOffscreen, &BitmapInfo, DIB_RGB_COLORS, NULL, NULL, 0);
...