2012-12-14 101 views
1

我在窗口中有2个区域,每个区域都有自己的工具提示。
这些工具提示是通过处理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_PAINT:

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; 

这些工具提示的问题是,角落(圆形矩形之外)应该是透明,但我似乎不能让他们消失。
我试过(天真地)使用HOLLOW_BRUSH来绘制背景矩形,但没有工作,正如你可以从我已经尝试使用分层窗口方法的例子看到的,再次无济于事。

任何人都可以帮助我获得我的工具提示背景的透明度?


这里是工具提示的不透明度的图片
[拐角已recoloured白色能见度 - 这些需要是透明的部件]
(文本显示为空白)
Tooltip without transparency

+0

呃。透明工具提示的概念非常奇怪。它应该清楚地看到它与任何重叠的部分,因此尖端易于阅读。我怀疑你需要SetWindowRgn()。 –

+0

工具提示是图片中的颜色,它只是需要透明的角落(它们是圆形的) - 授予图片不是很清楚,但您应该能够确定它们是红色的。他们是需要透明度的部分。 – Daboyzuk

+0

对于SetWindowRgn()来说,角落现在是白色以提高可视性 – Daboyzuk

回答

1

您可以使用SetWindowRgn来使部分窗口透明(使用CreateRoundRectRgn创建区域)。

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

+0

我不确定真正的阿尔法混合是我想要的,但我想如果我控制wm_paint我想我只是忽略窗口应该是透明的和事实上的事实...我会给它一个当我可以 – Daboyzuk

+0

SetWindowRgn和CreateRoundRectRgn已经完成了这项工作,谢谢! – Daboyzuk

2

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

对不起,没有测试任何这一点,但它适用于静态和组控制。到目前为止,我证明的唯一控制是对此方法无响应的复选框按钮,我无法为其设置背景透明度。

问候。

+0

感谢您的回复......我没有想过处理彩信......我真的应该有。是的,我已经尝试ws_ex_transparant已与其他样式,虽然我忽略提及它...将看看如果通过一个空心画笔可以帮助我,虽然 – Daboyzuk

+0

@Daboyzuk不客气。不要忘记设置后台模式 – mkey

+0

对于响应延迟感到抱歉,看起来(除非我做错了什么,或者阻止它们)工具提示实际上并没有收到任何WM_CTLCOLOR消息,我觉得有点奇怪。 – Daboyzuk