7

我遇到了 Aero Snap 无法与我正在使用的应用程序(Windows 桌面、本机 C++ 应用程序)一起工作的问题,而且我对正在发生的事情有点困惑,因为它似乎应该可以正常工作,超出盒子。

我在一个最小的 win32 应用程序上使用了 Spy++,当我按下 Win-Left 时收到以下消息:

<00070> 00030D1C P WM_KEYDOWN nVirtKey:VK_LWIN cRepeat:1 ScanCode:5B fRepeat:0 fUp:0 <00071> 00030D1C P WM_KEYDOWN nVirtKey:VK_LWIN cRepeat:1 ScanCode:5B fRepeat:1 fUp:0 <00072> 00030D1C P WM_KEYDOWN nVirt VK_LWIN cRepeat:1 ScanCode:5B fRepeat:1 fUp:0 <00088> 00030D1C S WM_GETMINMAXINFO lpmmi:0043FCBC
<00089> 00030D1C R WM_GETMINMAXINFO lpmmi:0043FCBC
<00090> 00030D1C S WM_WINDOWPOSCHANGING lpwp:0043FCC4
<00091> 00030D1C S WM_GETMINMAXINFO lpmmi:0043F8E8
< 00092> 00030D1C R WM_GETMINMAXINFO lpmmi:0043F8E8
<00093> 00030D1C R WM_WINDOWPOSCHANGING
.. 等等

所以我可以看到左键的 WM_KEYDOWN 没有到达应用程序,但是我得到了 aero snap “resize window” 的东西。

当我 Spy++ 我的应用程序时,我可以看到左键没有被“拦截”,而是被传递给应用程序,所以我没有得到任何捕捉好处。

<00043> 000F0F12 P WM_KEYDOWN nVirtKey:VK_LWIN cRepeat:1 ScanCode:5B fRepeat:0 fUp:0
<00044> 000F0F12 P WM_KEYDOWN nVirtKey:VK_LWIN cRepeat:1 ScanCode:5B fRepeat:1 fUp:0
<00045> 000F0F12 P WM_KEYDOWN nVirt VK_LWIN cRepeat:1 ScanCode:5B fRepeat:1 fUp:0
<00060> 000F0F12 P WM_KEYUP nVirtKey:VK_LEFT cRepeat:1 ScanCode:4B fRepeat:0 fUp:1

我将深入研究消息处理的核心,看看发生了什么,但我会采用我能得到的所有提示:)

编辑我注意到 Win-Up 和 Win-Shift-Left/Right 实际上可以正常工作,所以只是 Win-Down 和 Win-Left/Right 没有被“气动”到正确的位置/大小。

编辑好吧,问题似乎是我的窗口不是用 WS_THICKFRAME 标志创建的。如果我添加标志,snap 工作。现在,我实际上并不想要那里的边界,但至少我知道是什么导致了奇怪的行为..

希望最后的编辑摆脱边框就像响应 WM_NCCALCSIZE 一样简单,并使客户端占据整个窗口。

4

2 回答 2

5

我不记得具体的消息,但是如果您处理主窗口的 WM_MOVING/WM_MOVE 和/或 WM_SIZING/WM_SIZE 消息,Aero Snap 将被禁用。如果这些没有达到DefWindowProcAero Snap 将无法工作。我猜 DefWindowProc 负责实现 Aero Snap,所以如果你确保这些消息到达它,那可能会有所帮助。

我发现了这个实现自定义窗口拖动代码,因此应用程序在拖动窗口时继续执行和更新屏幕,这意味着处理这些消息,但它禁用了 Aero Snap。

编辑:进一步检查,我提到的应用程序处理 WM_SYSCOMMAND 并检查(wParam & 0xFFF0) == SC_MOVE以指示窗口移动开始。然后它返回 0 并通过定期更新窗口位置来模拟窗口拖动,同时仍在运行应用程序、绘图等。这导致 Windows 认为窗口是不可移动的并且用户无法拖动它,但我的应用程序正在更新要制作的位置看起来它仍在被拖动,直到 WM_LBUTTONUP。显然,如果 Windows 认为窗口没有被拖动,它就不会尝试任何 Aero Snap 的东西。也许您的应用程序做了类似的事情(如果有人有更好的方法让应用程序在拖动过程中保持运行,我很想听听)。

于 2010-05-27T12:21:24.423 回答
2

我怀疑它是消息处理,消息循环永远不会看到 WM_KEYDOWN 消息。在尝试了各种失败后,我只能猜测 Windows 认为您的应用程序在某种程度上不兼容。例如,在您的程序中使用 SetWindowsHookEx()。

于 2010-05-27T12:26:38.343 回答