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

Posted

技术标签:

【中文标题】什么可能导致低级鼠标钩子错过按钮事件?【英文标题】:What might cause a low-level mouse hook to miss button-up events? 【发布时间】:2018-02-02 13:44:30 【问题描述】:

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

#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' 事件通常会被丢弃。 IE。调试跟踪如下所示:

[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 中进行这些实验。

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

【问题讨论】:

代码可能不是最优的但是是正确的。猜测带有调试视图输出的东西。准确检查 - 使用 4 个全局或静态变量:static ULONG ld, lu, rd, ru; 并在每次调用 switch (wParam) case WM_LBUTTONDOWN: ld++; break; case WM_LBUTTONUP: lu++; break; case WM_RBUTTONDOWN: rd++; break; case WM_RBUTTONUP: ru++; break; 时增加对应的变量。每次都打印出来,或者在你删除钩子之后 @RbMm 有趣的想法 - 我试过了,但计数器与 DebugView 输出相匹配:除非我点击得很慢,否则“向上”事件比“向下”事件少。 在这种情况下不知道什么会产生这种情况。 i for test 现在尽可能快地点击多次 - 并且在测试结束时得到 - 相等 (ld==lu &amp;&amp; rd==ru`) 刚刚在 MSVC 2017 中尝试过(从带有 F5 的 IDE 运行) - 没有“额外”移动事件 并且每个都有一个 up 事件down,慢或快。不过,我确实做了一个微小的代码更改:在调试消息的末尾添加了一个新行 ;-) 我刚刚想到的一件事是,我在虚拟机中工作的事实可能会影响这一点(我相应地扩展了我的问题)。我会在真机上试试这个。 【参考方案1】:

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

如果你的钩子函数太慢,那么 Windows 可能会在不告诉你的情况下解开你,但这意味着你不会从该 HHOOK 实例中获得更多事件。

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

【讨论】:

以上是关于什么可能导致低级鼠标钩子错过按钮事件?的主要内容,如果未能解决你的问题,请参考以下文章

低级鼠标钩子和 DirectX

区分鼠标设备和低级鼠标挂钩

c++:使用全局鼠标钩子重新定位窗口

钩子编程(HOOK) 屏蔽全部按键鼠标及系统功能键

注册一个全局钩子来检测鼠标是不是拖动文件/文本

异步鼠标钩子?