大家在使用某些软件的过程中,有没有注意到有些软件有一些很有趣的东西。比如说在<br>主窗口的标题栏上居然有一个按钮。在Internet中随处可见这样的小控件。按钮怎么可<br>以加入到非客户区(Client)呢? <br> 在这里,最关键的一点就是,大家不要被传统知识误导:真的认为它是一个按钮。有<br>名柄(handle)的控件当然不能放在标题栏上了。有经验的程序员用Spy++跟踪一下的话,<br>马上就会发现其中的秘密。它并不是一个按钮,只不过是处理成按钮的样子罢了。 <br>既然知道了所以然,那么我们为什么不能自己来做一个呢,当然没问题,下面我们就用<br>Delphi来实现它,讲注意我的注解。 <br><br>具体实例之前,我们应该知道几个关于标题栏的重要的消息: <br>WM_NCPAINT:重画标题栏消息。我们必须截住它,可以在这里重画按钮; <br>WM_NCLBUTTONDOWN:在标题栏上按下鼠标左键消息。我们可以截住它,在标题栏上画出<br>按钮按下的样子,并且可以在其中进行自已的单击事件的处理,使得它像一个按钮; <br><br>WM_NCLBUTTONUP:在标题栏上释放鼠标左键消息。我们可以截住它,在标题栏上画出按<br>钮弹起的样子; <br><br>WM_NCLBUTTONDBLCLK:在标题栏上双击鼠标左键消息。我们可以截住它,当在按钮区域<br>双击时,我们就该使其无效,从而避免窗体执行最大化和还原操作。 <br><br>WM_NCRBUTTONDOWN:在标题栏上按下鼠标右键消息。我们可以截住它,当在按钮区域<br>双击时,我们就该使其无效,从而避免弹出窗体按制菜单。 <br><br>WM_NCMOUSEMOVE:在标题栏上移动鼠标消息。我们可以截住它,当鼠标移出按钮区域时,<br>我们就必须画出按钮没有被按下,即凸起时的样子。 <br><br>WM_NCACTIVATE:当标题栏在激活与非激活之间切换时收到该消息。我们可以截住它,<br>当该窗口处理激活状态时,我们可以做一些事情,比如说将我们的标题栏按钮上的字体<br>变灰或变黑来指示该窗口的当前状态。下面我没有加入该项功能,如果大家感兴趣的话,<br>可以自己完成。 <br><br>(大家从这里可以发现,标题栏的消息都是WM_NC开头的) <br>例子:<br>unit main;<br><br>interface<br><br>uses<br> Windows, Messages, SysUtils, Classes, <br>Graphics, Controls, Forms, Dialogs,<br> StdCtrls, Menus;<br><br>type<br> TForm1 = class(TForm)<br> procedure FormCreate(Sender: TObject);<br> procedure FormDestroy(Sender: TObject);<br><br> private<br> { Private declarations }<br><br> CBBtnRect: TRect; // Caption Bar Button Rectangle<br> CBBtnFont: TFont; // Caption Bar Button Font<br> procedure DrawCaptionBtn(uEdge: UINT);<br> // 当在标题栏上按下鼠标左按钮时进入该过程<br>procedure WMNcLButtonDown(var m: TMessage); <br>message WM_NCLBUTTONDOWN;<br> // 当在标题栏上放开鼠标左按钮时进入该过程<br>procedure WMNcLButtonUp(var m: TMessage); <br>message WM_NCLBUTTONUP;<br> // 当在标题栏上移动鼠标时进入该过程<br>procedure WMNcMouseMove(var m: TMessage); <br>message WM_NCMOUSEMOVE;<br> // 当在标题栏上双击鼠标左铵钮时进入该过程<br>procedure WMNcLButtonDBLClk<br>(var m: TMessage); message WM_NCLBUTTONDBLCLK;<br> // 当在标题栏上按下鼠标右按钮时进入该过程<br>procedure WMNcRButtonDown(var m: TMessage); <br>message WM_NCRBUTTONDOWN;<br> // 当画标题栏时进入该过程<br>procedure WMNcPaint(var m: TMessage); <br>message WM_NCPAINT;<br> // 当标题栏在激活与非激活之间切换时进入该过程<br>procedure WMNcActivate(var m: TMessage); <br>message WM_NCACTIVATE;<br> public<br> { Public declarations }<br> end;<br><br>var<br> Form1: TForm1;<br><br>implementation<br><br>{$R *.DFM}<br><br>procedure TForm1.DrawCaptionBtn(uEdge: UINT);<br>var<br> hCaptionDC: HDC; // 标题条Device Context<br> hOldFont: HFONT; // 原来的字体<br> r: TRect;<br>begin<br> hCaptionDC := GetWindowDC(Self.Handle);<br> // 注意不能用GetDC,那样的话,将得不到标题栏<br> // 的设备上下文<br> //画按钮的样子,如果uEdge=EDGE_RAIS,<br>则画出的样子为凸起;如果<br>//uEdge=EDGE_SUNKEN,则画出的样子为凹下。<br> DrawEdge(hCaptionDC, CBBtnRect, uEdge, <br>BF_RECT or BF_MIDDLE or<br> BF_SOFT); <br><br> //设置标题栏的设备上下文为透明状态<br> SetBkMode(hCaptionDC, TRANSPARENT);<br><br> //设置标题栏设备上下文的字体<br> hOldFont:= SelectObject(hCaptionDC, CBBtnFont.Handle);<br><br> //画按钮<br> if uEdge = EDGE_RAISED then<br> DrawText(hCaptionDC, 'Caption Bar Button', <br>18, CBBtnRect, DT_CENTER)<br> else begin<br> r := CBBtnRect;<br> OffsetRect(r, 1, 1); <br> DrawText(hCaptionDC, 'Caption Bar Button', 18, r, DT_CENTER);<br> end;<br><br> //还原为原来的字体<br> SelectObject(hCaptionDC, hOldFont);<br>end;<br><br>procedure TForm1.WMNcActivate(var m: TMessage);<br>begin<br> inherited;<br> DrawCaptionBtn(EDGE_RAISED);<br>end;<br><br><br>procedure TForm1.WMNcPaint(var m: TMessage);<br>begin<br> inherited;<br> DrawCaptionBtn(EDGE_RAISED);<br>end;<br><br><br>procedure TForm1.WMNcLButtonDBLClk(var m: TMessage);<br>var<br> p: TPoint;<br>begin<br> p.x := LOWORD(m.lParam) - Self.Left;<br> p.y := HIWORD(m.lParam) - Self.Top;<br> if not PtInRect(CBBtnRect, p) then // 如果不在按钮区域内<br> inherited; // 执行默认的操作<br>end;<br><br>procedure TForm1.WMNcMouseMove(var m: TMessage);<br>var<br> p: TPoint;<br>begin<br> p.x := LOWORD(m.lParam) - Self.Left;<br> p.y := HIWORD(m.lParam) - Self.Top;<br> if not PtInRect(CBBtnRect, p) then // 如果不在按钮区域<br> DrawCaptionBtn(EDGE_RAISED)<br> else<br> inherited; // 执行默认的操作<br>end;<br><br><br>procedure TForm1.WMNcLButtonDown(var m: TMessage);<br>var<br> p: TPoint;<br>begin<br> p.x := LOWORD(m.lParam) - Self.Left;<br> p.y := HIWORD(m.lParam) - Self.Top;<br> if PtInRect(CBBtnRect, p) then // 如果按在了按钮区域<br> begin<br> Self.BringToFront;<br> DrawCaptionBtn(EDGE_SUNKEN);<br> end<br> else<br> inherited; // 执行默认的操作<br>end;<br><br><br>procedure TForm1.WMNcLButtonUp(var m: TMessage);<br>var<br> p: TPoint;<br>begin<br> p.x := LOWORD(m.lParam) - Self.Left;<br> p.y := HIWORD(m.lParam) - Self.Top;<br> if PtInRect(CBBtnRect, p) then // <br>如果在标题栏按钮区域释放鼠标<br> begin<br> DrawCaptionBtn(EDGE_RAISED);<br> end<br> else<br> inherited; // 执行默认的操作<br>end;<br><br><br>procedure TForm1.WMNcRButtonDown(var m: TMessage);<br>var<br> p: TPoint;<br>begin<br> p.x := LOWORD(m.lParam) - Self.Left;<br> p.y := HIWORD(m.lParam) - Self.Top;<br> if not PtInRect(CBBtnRect, p) then // 如果不在标题栏按钮区域<br> inherited; // 执行默认的操作<br>end;<br><br><br>procedure TForm1.FormCreate(Sender: TObject);<br>begin<br> // 这个大小大家可以得用GetSystemMetrics<br>函数来进行更精确的计算。这里<br> // 只是用来示例<br> with CBBtnRect do<br> begin<br> left := 100;<br> top := 6;<br> right := 450;<br> bottom := 20;<br> end;<br><br> // 标题栏按钮字体。<br> CBBtnFont:= TFont.Create;<br> with CBBtnFont do<br> begin<br> Name := '宋体';<br> Size := 9;<br> Color := clRed;<br> end;<br>end;<br><br><br>procedure TForm1.FormDestroy(Sender: TObject);<br>begin<br> CBBtnFont.Free;<br>end;<br><br>end.<br><br>