1

当我SetBkMode(hdc, TRANSPARENT);在下面的代码中使用时,当我调整主窗口的大小时(因此当孩子收到 WM_PAINT 消息时),我得到了以下效果:

在此处输入图像描述

问题是:当我调整主窗口的大小时,“查找:”的旧区域应该被删除,我猜。但它仍然存在。

如果我不使用SetBkMode(hdc, TRANSPARENT);,我没有这个问题。看起来像:

在此处输入图像描述

,即它有白色背景。此外,如果我使用SetBkMode(hdc, TRANSPARENT);,在我调整主窗口大小之前,它看起来与上面相同。所以我不认为SetBkMode(hdc, TRANSPARENT);在这里工作。

hwnd一个有风格的静态孩子SS_BITMAP

你知道为什么会出现这个问题吗?

switch (message) {
    case WM_PAINT:
        PAINTSTRUCT ps;
        HDC hdc = BeginPaint(hwnd, &ps);

        SelectObject(hdc, gDefaultGuiFont);
        SetBkMode(hdc, TRANSPARENT);

        RECT rc;
        GetClientRect(hwnd, &rc);
        DrawText(hdc, _TR("Find:"), -1, &rc, DT_SINGLELINE | DT_CENTER | DT_VCENTER);

        EndPaint(hwnd, &ps);

        return 0;

     .............
}
4

2 回答 2

0

问题是windows没有更新静态控件后面的控件(及时),你现在负责它的内容。所以你想使用父母提供的背景。好吧,只需让父母在子窗口中为您绘制它:

RECT rc;
GetClientRectRelative(m_hWnd, GetParent(m_hWnd), &rc);
SetWindowOrgEx(m_mdc, rc.left, rc.top, NULL);
SendMessage(GetParent(m_hWnd), WM_PAINT, (WPARAM)(HDC)m_mdc);
SetWindowOrgEx(m_mdc, 0, 0, NULL);

其中

bool GetClientRectRelative(HWND hWnd, HWND hWndRelativeTo, RECT *pRect)
{
    RECT rcWnd, rcRelativeTo;
    if (!GetClientRect(hWnd, &rcWnd) ||
            !ClientToScreen(hWnd, (POINT*)&rcWnd) ||
            !ClientToScreen(hWnd, (POINT*)&rcWnd + 1) ||
            !GetClientRect(hWndRelativeTo, &rcRelativeTo) || 
            !ClientToScreen(hWndRelativeTo, (POINT*)&rcRelativeTo) ||
            !ClientToScreen(hWndRelativeTo, (POINT*)&rcRelativeTo + 1))
        return false;

    pRect->top = rcWnd.top - rcRelativeTo.top;
    pRect->left = rcWnd.left - rcRelativeTo.left;
    pRect->right = rcWnd.right - rcRelativeTo.left;
    pRect->bottom = rcWnd.bottom - rcRelativeTo.top;

    return true;
}

现在画你喜欢的任何东西,我建议你使用TRANSPARENT背景模式。

WS_CLIPCHILDREN请使用和样式创建您所有的子窗口WS_CLIPSIBLINGS,然后这些问题将立即变得明显并且您避免闪烁。

于 2012-08-06T16:29:39.400 回答
0

尝试使用“固定”矩形。例如

    RECT rc;
    GetClientRect(hwnd, &rc);
    rc.left += ...; rc.top += ...; // shift up-left point
    DrawText(hdc, _TR("Find:"), -1, &rc, DT_SINGLELINE | DT_LEFT | DT_TOP);

这个想法是你在错误的位置(一次)和正确的位置(两次)绘制文本,而背景只更新一次。关于部分代码不能说更多。

于 2012-08-05T09:24:41.893 回答