如果以WS_POPUP样式创建Window,则永远不会传递WM_PAINT消息

时间:2019-06-16 11:30:17

标签: c++ winapi

我正在尝试使用“ WS_POPUP” 样式创建一个没有标题栏的窗口,并且还需要在边框上进行一些绘制。

当我收到“ WM_PAINT” 消息时,我正在“ WndProc” 中实现绘图逻辑。但是,它从未交付。

我尝试过弄乱样式,但是每当使用WS_POPUP样式时,就不会始终交付WM_PAINT。

当您使用Visual Studio Community 2019创建一个新的“ Windows桌面应用程序”项目时,只需更改以下一项内容,即可从自动生成的代码中复制该内容:

WS_OVERLAPPEDWINDOW-> WS_POPUP

创建窗口:

HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_POPUP | WS_VISIBLE,
      CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr);

注册窗口类:

    WNDCLASSEXW wcex;

    wcex.cbSize = sizeof(WNDCLASSEX);

    wcex.style          = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc    = WndProc;
    wcex.cbClsExtra     = 0;
    wcex.cbWndExtra     = 0;
    wcex.hInstance      = hInstance;
    wcex.hIcon          = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_WINDOWSPROJECT2));
    wcex.hCursor        = LoadCursor(nullptr, IDC_ARROW);
    wcex.hbrBackground  = (HBRUSH)(COLOR_WINDOW+1);
    wcex.lpszMenuName   = MAKEINTRESOURCEW(IDC_WINDOWSPROJECT2);
    wcex.lpszClassName  = szWindowClass;
    wcex.hIconSm        = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));

    return RegisterClassExW(&wcex);

消息循环:

    while (GetMessage(&msg, nullptr, 0, 0))
    {
        if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
        {
            TranslateMessage(&msg);
            DispatchMessage(&msg);
        }
    }

处理窗口消息:

LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    switch (message)
    {
    case WM_PAINT:
        {
            PAINTSTRUCT ps; // NEVER CALLED !!!
            HDC hdc = BeginPaint(hWnd, &ps);
            // TODO: Add any drawing code that uses hdc here...
            EndPaint(hWnd, &ps);
        }
        break;
    case WM_DESTROY:
        PostQuitMessage(0);
        break;
    default:
        return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}

我希望传递WM_PAINT消息,以便进行绘画。但是,它从未交付。

1 个答案:

答案 0 :(得分:2)

如果您修改CreateWindow调用,将在启动时收到WM_PAINT事件,如下所示:

HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_POPUP | WS_VISIBLE, 
  0, 0, 800, 600, nullptr, nullptr, hInstance, nullptr);

您只需要输入一些有效的宽度和高度即可。 不过,下一次收到WM_PAINT的时间将是您InvalidateRect窗口时。 或者,例如,如果您单击“显示桌面”按钮(右下)并还原窗口(从最小化还原)。 请注意,当窗口重新获得焦点(alt-tab)时,您将不会收到WM_PAINT,除非您InvalidateRect如下所示:

case WM_ACTIVATE:
    InvalidateRect(hWnd, NULL, TRUE);
    break;