什么可能导致低级鼠标挂钩错过按钮启动事件?

What might cause a low-level mouse hook to miss button-up events?

本文关键字:错过 按钮 启动 事件 鼠标 什么      更新时间:2023-10-16

我编写了一个简单的程序将所有鼠标事件打印到调试视图:

#include <windows.h>
#include <stdio.h>

HHOOK g_mouseEventHook = NULL;

LRESULT CALLBACK mouseEventHookProc( int code, WPARAM wParam, LPARAM lParam )
{
if ( code != HC_ACTION ) {
return ::CallNextHookEx( g_mouseEventHook, code, wParam, lParam );
}
PMSLLHOOKSTRUCT mi = (PMSLLHOOKSTRUCT)lParam;
char buf[2048];
sprintf( buf, __FUNCTION__": caught event: wparam = 0x%08x, time = %d, x = %d, y = %d", wParam, mi->time, mi->pt.x, mi->pt.y );
OutputDebugStringA( buf );
return ::CallNextHookEx( g_mouseEventHook, code, wParam, lParam );
}

int main()
{
g_mouseEventHook = ::SetWindowsHookEx( WH_MOUSE_LL, mouseEventHookProc, ::GetModuleHandle( NULL ), 0 );
MSG msg;
while ( ::GetMessage( &msg, NULL, 0, 0 ) > 0 ) {
::TranslateMessage( &msg );
::DispatchMessage( &msg );
}
return 0;
}

代码不会费心清理(例如,取消注册钩子或干净地退出消息循环);您只需按 Ctrl+C 即可中止它。

我使用 Visual Studio 2013 通过运行

cl main.cpp user32.lib

运行时,调试器输出显示(使用例如 DebugView 可见)此输出进行"慢速"单击,我按下鼠标左键,稍等片刻,然后释放鼠标左键:

[1660] mouseEventHookProc: caught event: wparam = 0x00000201, time = 17367092, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17367092, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000202, time = 17368636, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17368636, x = 546, y = 90

wparam值对应于标准窗口消息:

  • 0x0200 =>WM_MOUSEMOVE
  • 0x0201 =>WM_LBUTTONDOWN
  • 0x0202 =>WM_LBUTTONUP

即在上面的跟踪中,我的程序看到鼠标向下,鼠标移动,鼠标向上,鼠标移动。似乎每个按钮事件总是与移动到同一坐标(并具有相同的时间戳)的移动事件一起进行。

当点击速度快一点时,就像我通常所做的那样,"up"事件通常会被删除。 即调试跟踪如下所示:

[1660] mouseEventHookProc: caught event: wparam = 0x00000201, time = 17371038, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17371054, x = 546, y = 90
[1660] mouseEventHookProc: caught event: wparam = 0x00000200, time = 17371132, x = 546, y = 90

即只有一个鼠标向下,然后有两个鼠标移动事件。时间戳在所有情况下都不同,并且完全缺少 up 事件。

我正在运行VMware Fusion的Windows 7 VM中进行这些实验。

有人能够重现和/或解释这种行为吗?我的测试程序中是否存在错误,或者这是我缺少的低级钩子的一些特殊行为?

我知道钩子可以错过某些事件的唯一方法是,如果钩子列表中的其他人在您面前并且他们通过不调用CallNextHookEx来吃掉事件。

如果您的挂钩功能太慢,那么 Windows 可能会在不告诉您的情况下取消挂钩,但这意味着您不会从该 HHOOK 实例获得进一步的事件。

VM可能会丢失主机和来宾计算机之间的某个位置的单击,尤其是在 VM 窗口和主机上的其他内容之间切换时。您可以通过创建一个小型测试应用程序来排除这种情况,该应用程序还打印其鼠标事件并在来宾中运行它。