[MFC] 无边框窗口的Resize拖动效果的实现(下):子窗口覆盖,导致主窗口热点区无法响应WM_NCHITTEST

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/felicityWSH/article/details/71514562
这篇博文讲解了如何实现 无边框窗口的Resize拖动效果,文中提到,需要响应主窗口的WM_NCHITTEST消息,检测鼠标是否在resize热点区并返回HTTOPLEFT、HTBOTTOM等相应的结果。

但是,如果子窗口覆盖了主窗口的热点区,当鼠标在热点区移动时,主窗口是无法收到WM_NCHITTEST消息的。比如图中所示的窗口,主窗口只有蓝色的resize区域能响应WM_NCHITTEST,其他几个resize区域的鼠标消息均会被分发到子窗口中:

有什么解决办法?

1、子窗口消息分发
这种方法比较容易想到,在子窗口中响应到WM_NCHITTEST后,判断自己是否处于主窗口的resize热点区,是的话就把消息通知给主窗口。
但是这种方法很麻烦,会覆盖resize区的窗口类型太多了:CDialog,CWnd。既有窗口、也有控件。虽然都可以在基类中处理WM_NCHITTEST,但还是麻烦。

2、悬停Popup类型的半透明窗口
创建四个Popup类型的透明窗口,将四个窗口始终悬停在主窗口的四个边上,并使其窗口Z序高于主窗口。在这四个窗口中处理光标的检测(HTTOPLEFT、HTBOTTOM等对应的光标状态)、鼠标左键按下和resize的逻辑,如果用户在半透明窗口中按下左键并移动,则在计算移动的尺寸后,更新主窗口的桌面位置、宽高,即可实现主窗口的resize效果。四个窗口之所以使用半透明,是为了不让用户感受到其存在。但是注意,四个窗口不能完全透明,否则无法响应鼠标事件(可以将其alpha设为1)。

如何创建四个半透明的Popup窗口?关键词:WS_EX_LAYERED + SetLayeredWindowAttributes。如果还不懂WS_EX_LAYERED属性,可以参考 这篇博文

如上所述,方法2中,实际上提出了一种新的实现无边框窗口的Resize方法。

猜你喜欢

转载自blog.csdn.net/felicityWSH/article/details/71514562