什么可能导致低级鼠标挂钩错过按钮启动事件?
What might cause a low-level mouse hook to miss button-up events?
我编写了一个简单的程序将所有鼠标事件打印到调试视图:
#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 窗口和主机上的其他内容之间切换时。您可以通过创建一个小型测试应用程序来排除这种情况,该应用程序还打印其鼠标事件并在来宾中运行它。
相关文章:
- 问:如何使用C++中的按钮从窗口打开窗口
- 如何将点击的信号和插槽添加到qt中的自定义按钮中
- QML按钮点击功能执行顺序
- 当用户超过按钮点击限制时报告
- 函数计算用户按下按钮的频率
- 热键/按钮,根据需要进行编译,但不运行(在F5和Ctrl+Shift+B之间)
- FLTK:按下哪个按钮 - 将数字传递给按钮的回调 (lambda)
- 如何通过按下第三个窗口中的按钮,将QString从一个窗口获取到另一个窗口
- 垂直方向的 Gtk3+ 旋转按钮 (c/c++)
- Qt自定义QPush按钮未显示在布局上
- SDL 操纵杆按钮按下/C++
- 如何使用按钮更新GTK3图像以使用C++从相机捕获图片
- Qt中来自不同功能的按钮引用
- WinAPI 在单击第一个对话框上的按钮控件并销毁第一个对话框后创建第二个对话框
- ImGui 在单击按钮后冻结
- WinAPI 无法创建按钮
- 按钮悬停在 QT 中垂直布局的选项卡小部件中不起作用
- 如何使用connect将qml按钮与同一类的cpp函数连接起来
- 按钮按下打开 OLED ESP32
- 什么可能导致低级鼠标挂钩错过按钮启动事件?