C 透明工具提示背景

C 透明工具提示背景,c,winapi,tooltip,transparency,C,Winapi,Tooltip,Transparency,我在一个窗口中有两个区域,每个区域都有自己的工具提示。 这些工具提示是通过处理WM_PAINT消息(防止闪烁)自定义绘制的 这是工具提示的创建过程: tooltips[MAIN_GRAPH_TT].tthWnd = CreateWindowEx(WS_EX_TOPMOST,TOOLTIPS_CLASS,0,WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP | TTS_NOFADE,CW_USEDEFAULT, CW_USEDEFAULT,CW_USEDEFAULT

我在一个窗口中有两个区域,每个区域都有自己的工具提示。
这些工具提示是通过处理WM_PAINT消息(防止闪烁)自定义绘制的

这是工具提示的创建过程:

tooltips[MAIN_GRAPH_TT].tthWnd =  CreateWindowEx(WS_EX_TOPMOST,TOOLTIPS_CLASS,0,WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP | TTS_NOFADE,CW_USEDEFAULT, CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,0,0,g_hInst,0);  
tooltips[SECONDARY_GRAPH_TT].tthWnd =   CreateWindowEx(WS_EX_TOPMOST,TOOLTIPS_CLASS,0,WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP | TTS_NOFADE,CW_USEDEFAULT, CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,0,0,g_hInst,0);  
这是工具提示的初始化:

if (tooltips[MAIN_GRAPH_TT].tthWnd)
{

    lpfnOldTTProc = (WNDPROC)SetWindowLong(tooltips[MAIN_GRAPH_TT].tthWnd,
        GWL_WNDPROC, (DWORD) TooltipProc);
    SetWindowLong(tooltips[MAIN_GRAPH_TT].tthWnd, GWL_EXSTYLE, WS_EX_LAYERED|WS_EX_TOOLWINDOW);
    SetLayeredWindowAttributes(tooltips[MAIN_GRAPH_TT].tthWnd,RGB(255,0,0),0,ULW_COLORKEY);
    SendMessage(tooltips[MAIN_GRAPH_TT].tthWnd,CWM_SETWNDPROC,0,(LPARAM)new WNDPROC(lpfnOldTTProc));
}

if (tooltips[SECONDARY_GRAPH_TT].tthWnd)
{

    lpfnOldTTProc = (WNDPROC)SetWindowLong(tooltips[SECONDARY_GRAPH_TT].tthWnd, GWL_WNDPROC, (DWORD) TooltipProc);
    SetWindowLong(tooltips[SECONDARY_GRAPH_TT].tthWnd, GWL_EXSTYLE, WS_EX_LAYERED|WS_EX_TOOLWINDOW);
    SetLayeredWindowAttributes(tooltips[SECONDARY_GRAPH_TT].tthWnd,RGB(255,0,0),0,ULW_COLORKEY);
    SendMessage(tooltips[SECONDARY_GRAPH_TT].tthWnd,CWM_SETWNDPROC,0,(LPARAM)new WNDPROC(lpfnOldTTProc));
}  
这是自定义工具提示WNDPROC的WM_绘制:

 case WM_PAINT:
     {

        const int FRAME_WIDTH = 1;
        const int CORNER_DIAMETER = 10;
        PAINTSTRUCT ps;
        HDC hdc = BeginPaint(hWnd,&ps);
        HDC hMemDC;
        RECT cr;
        GetClientRect(hWnd,&cr);
        hMemDC = CreateCompatibleDC(hdc);
        HBITMAP memBM = CreateCompatibleBitmap(hdc, cr.right-cr.left, cr.bottom-cr.top);
        HBITMAP hOldBM = (HBITMAP) SelectObject(hMemDC,memBM);
        //drawing start [draw to hMemDC]
        {
            FillSolidRect(hMemDC,0,0,cr.right-cr.left,cr.bottom-cr.top,RGB(255,0,0));               
            HPEN hFramePen = CreatePen(PS_SOLID,FRAME_WIDTH,BLACK);
            HBRUSH hBGBrush = GetSysColorBrush(COLOR_INFOBK);
            SetTextColor(hMemDC,GetSysColor(COLOR_INFOTEXT));
            SetBkColor(hMemDC,WHITENESS);
            SetBkMode(hMemDC,TRANSPARENT);
            HBRUSH hOldBrush = (HBRUSH) SelectObject(hMemDC,hBGBrush);
            HPEN hOldPen = (HPEN) SelectObject(hMemDC,hFramePen);
            HFONT hOldFont = SelectFont(hMemDC,g_hFonts[FONT_TOOLTIP]);
            RoundRect(hMemDC,cr.left,cr.top,cr.right,cr.bottom,CORNER_DIAMETER,CORNER_DIAMETER);
            RECT textRec = cr;
            textRec.left += FRAME_WIDTH*2;
            textRec.right -= FRAME_WIDTH*2;
            textRec.top += FRAME_WIDTH*2;
            textRec.bottom -= FRAME_WIDTH*2;
            if(hWnd == tooltips[MAIN_GRAPH_TT].tthWnd)
                DrawText(hMemDC,tttBuffer[MAIN_GRAPH_TT],sizeof(tttBuffer),&textRec,DT_LEFT|DT_TOP);
            else if(hWnd == tooltips[SECONDARY_GRAPH_TT].tthWnd)
                DrawText(hMemDC,tttBuffer[SECONDARY_GRAPH_TT],sizeof(tttBuffer),&textRec,DT_LEFT|DT_TOP);
            SelectObject(hMemDC,hOldBrush);
            SelectObject(hMemDC,hOldPen);
            SelectObject(hMemDC,hOldFont);
            DeleteObject(hFramePen);
            DeleteObject(hBGBrush);

        }
        //drawing end
        BitBlt(hdc,
                cr.left,
                cr.top,
                cr.right-cr.left, cr.bottom-cr.top,
                hMemDC,
                0,
                0,
                SRCCOPY);
        SelectObject(hdc,hOldBM);
        DeleteObject(memBM);
        DeleteDC(hMemDC);
        EndPaint(hWnd,&ps);
     }
    break;
这些工具提示的问题是,角点(圆形矩形之外)应该是透明的,但我似乎无法使它们消失。
我曾经尝试(天真地)使用空心画笔绘制背景矩形,但没有成功,正如您从示例中看到的,我尝试使用分层窗口方法,但同样没有成功

有人能帮我获得工具提示背景的透明度吗


下面是没有透明度的工具提示图片
[角落已重新涂成白色以便于查看--这些是需要透明的部分]
(文本空白)

在处理控件和透明度时,我发现WS_EX_TRANSPARENT exStyle可以为您节省一些时间。此外,您可能需要检查工具提示是否向主窗口发送WM_CTLCOLORSTATIC消息,如果发送,您可能需要返回一个空心(null)笔刷句柄作为响应。您可能还需要在该时间点通过调用WM_CTLCOLORSTATIC消息提供的hDC上的SetBkMode将后台模式设置为透明

很抱歉没有测试这些,但是它可以很好地与静态控件和组控件配合使用。到目前为止,我证明的唯一控件是复选框按钮,它对该方法没有响应,我无法为其设置背景透明度


注意。

您可以使用
SetWindowRgn
使窗口的某些部分透明(使用
CreateRoundRectRgn
创建区域)


或者,您可以使用
SetLayeredWindowAttributes
使用真实的alpha混合使窗口部分透明。

。透明工具提示的概念非常奇怪。无论重叠的部分是什么,都应该清晰可见,因此提示易于阅读。我怀疑您需要SetWindowRgn()。工具提示是图片中的颜色,只是需要透明的角(它们是圆形的)——尽管图片不是很清晰,但您应该能够分辨出它们是红色的。它们是需要透明度的部分。为了更好地显示,角现在是白色的。对于SetWindowRgn(),这似乎是个好主意,我必须试一试,看看我是否能用它从头开始创建工具提示。我更喜欢通过工具提示控件来完成,但它绝对值得一试。我不确定真正的alpha混合是我想要的,但我想如果我控制wm_绘制,我想我只是忽略了窗口应该是透明的,并且绘制。。。当我可以设置WindowRGN和CreateRoundRectRgn完成工作时,我会尝试一下,谢谢!谢谢你的回复。。。我没有想过要处理彩色信息。。。我真的应该这么做。是的,我已经尝试过其他风格的ws_ex_transparant,尽管我忘了提到它。。。会看看是否通过一个空心刷可以帮助我though@Daboyzuk不客气。不要忘了设置背景模式,也为响应延迟感到抱歉,似乎(除非我做错了什么,或者阻止了它们)工具提示实际上没有收到任何WM_CTLCOLOR消息,我觉得有点奇怪。@Daboyzuk没关系。艾德克,这并不奇怪。众所周知,winapi中的常见控件非常奇怪,这是一个令人悲哀的事实。就像我上面提到的复选框示例一样。一个愚蠢的问题,你确定你在正确的家长窗口等待他们吗?是的,我相当肯定。我不是绝对正确的,但我只是仔细检查了一下,仍然没有收到任何工具提示的CTLCOLOR(+friends)消息。