[标题]:在全局鼠标钩子中模拟鼠标右键单击
[时间]:2009-3-28
[摘要]:本文分为两部分,第一部分:使用SwapMouseButton()切换鼠标左右键功能。第二部分:在全局鼠标钩子中模拟鼠标右键单击。
[关键字]:鼠标失灵,鼠标右键,切换鼠标左右键,屏幕宽度,屏幕高度,SendInput,全局鼠标钩子,鼠标事件
[平台]:Windows XP SP3 ,VC6
[作者]:Winty (
[email protected])
[正文]:
我的鼠标左键失灵了。暂时只能使用鼠标右键。就只能把鼠标右键改成左键用(切换方法:控制面板->鼠标->切换主要和次要的按键)。当然,切换了之后,右键功能就没有了。可是平时用惯了鼠标右键,现在没有可不行。利用程序切换时,代码片段如下:
//bSwap:BOOL型
//为TRUE为切换左右键功能,为FALSE为恢复原左右键功能
::SwapMouseButton(bSwap);
可以使用RegisterHotKey()注册一个热键,在需要时切换。但是还得知道该传入TRUE还是FALSE到SwapMouseButton(),那么可以使用GetSystemMetrics(SM_SWAPBUTTON)查询鼠标当前左右键功能是否反转了:
//返回TRUE表示鼠标左右键功能反转了,FALSE表示正常
::GetSystemMetrics(SM_SWAPBUTTON);
最后可以让程序每次自动启动,需要右键的时候按一下热键就可以了,使用完了再切换回来。问题是,这样每次使用时,在左右键之间来回切换很麻烦。
下面在全局鼠标钩子中模拟鼠标右键单击。当按Ctrl+鼠标左键时,相当于点击右键。这里所说的鼠标左键与右键都是用SwapMouseButton(TRUE)切换过的,也就是说,我的真实的鼠标左键已坏,实际上只有鼠标右键可以使用,现在我把鼠标右键当左键用,但又添加右键本身的功能(就是加按Ctrl时相当于原来的右键)。未特殊说明,则下同。
1、使用SetWindowsHookEx安装钩子。
HHOOK CUtil::InstallMouseHook(HINSTANCE hInstance)
{
HHOOK hhkMouseHook;
hhkMouseHook = SetWindowsHookEx(
WH_MOUSE_LL, // hook type
CUtil::LowLevelMouseProc, // hook procedure
hInstance, // handle to application instance
0 // thread identifier:为0表示全局钩子
);
if(hhkMouseHook==NULL)
MessageBox(NULL,"安装MouseHook失败!","InstallMouseHook()",MB_OK);
return hhkMouseHook;
}
在应用程序的InitInstance()中调用:
//m_hInstance为应用程序实例句柄
HHOOK hhkLowLevelKybd = InstallMouseHook(m_hInstance):
2、编写全局鼠标钩子回调函数LowLevelMouseProc:
//当按下"Ctrl+鼠标左键" 或 "数字键盘减号键 + 鼠标左键"时,模拟鼠标右键
LRESULT CALLBACK
CUtil::LowLevelMouseProc(int nCode,
WPARAM wParam,
LPARAM lParam)
{
if(nCode == HC_ACTION){
//左Ctrl键按下
BOOL bLeftCtrlDown =
(GetAsyncKeyState(VK_LCONTROL) & 0x8000) != 0;
//数字键盘减号"-"键按下
BOOL bNumpadSubtractDown =
(GetAsyncKeyState(VK_SUBTRACT) & 0x8000) != 0;
//当按下"Ctrl+鼠标左键" 或 "数字键盘减号键+鼠标左键"时,模拟鼠标右键
if( (wParam == WM_LBUTTONDOWN) &&
(bLeftCtrlDown || bNumpadSubtractDown) )
{
const INPUT_SIZE = 2;
INPUT input[INPUT_SIZE];
ZeroMemory( &input, sizeof(INPUT)*INPUT_SIZE);//初始化INPUT结构体
//鼠标右键按下
input[0].type = INPUT_MOUSE;
input[0].mi.dwFlags = MOUSEEVENTF_LEFTDOWN;
//鼠标右键弹起
input[1].type = INPUT_MOUSE;
input[1].mi.dwFlags = MOUSEEVENTF_LEFTUP;
//发送鼠标右键单击
SendInput(
INPUT_SIZE, // count of input events
input, // array of input events
sizeof(INPUT) // size of structure
);
//发送模拟的鼠标右键单击后,不再响应鼠标左键单击消息
return 1;
}
}
//如果不是"Ctrl+鼠标左键"按下,则向后传递鼠标消息。
return CallNextHookEx(NULL,nCode,wParam,lParam);
}
LowLevelMouseProc()有几点要注意的地方:
a)、LowLevelMouseProc()应声明为static,因为在一个类中,只有static函数才能作为回调函数。
b)、wParam == WM_LBUTTONDOWN一句原来写的是WM_LBUTTONUP,结果点击后右键菜单出来了,但无法完成复制。所以应为WM_LBUTTONDOWN就不会有问题了。
c)、只用使用VK_LCONTROL而不是VK_CONTROL是因为Ctrl+鼠标左键单击在选择多个文件时已被使用。所以把VK_RCONTROL留给选择多个文件等其它用途时使用,而用数字键盘中的减号代替VK_RCONTROL。也就是说,当按下"Ctrl+鼠标左键" 或 "数字键盘减号键+鼠标左键"时,模拟鼠标右键。
d)、input[i].mi.dwFlags中不用MOUSEEVENTF_ABSOLUTE,代表input[i].mi.dx或input[i].mi.dx中的坐标是相对于上一次鼠标事件时的坐标,而dx和dy都已被初始化为0,即在原地显示鼠标右键菜单;
e)、也可以在input[i].mi.dwFlags中使用MOUSEEVENTF_ABSOLUTE,此时是绝对坐标,input[i].mi.dx和input[i].mi.dx的计算如下(而不是直接使用当前鼠标位置,具体原因请查看MSDN):
PMSLLHOOKSTRUCT pllh = (PMSLLHOOKSTRUCT)lParam;//鼠标坐标等信息
int cx=GetSystemMetrics(SM_CXSCREEN);//得到屏幕宽度
int cy=GetSystemMetrics(SM_CYSCREEN);//得到屏幕高度
LONG dx = pllh->pt.x * 65535 / cx;
LONG dy = pllh->pt.y * 65535 / cy;
input[0].mi.dx = dx;
input[0].mi.dy = dy;
input[1].mi.dx = dx;
input[1].mi.dy = dy;
f)、在发送完模拟的鼠标右键事件后,需要return 1;来阻止消息的继续传递,而不是return CallNextHookEx(NULL,nCode,wParam,lParam);,不然会出现不正确的右键行为。
g)、需要发送鼠标右键按下和弹起两个事件,不然会出现不正确的右键行为。
h)、WM_LBUTTONDOWN映射的是逻辑鼠标按键,wParam == WM_LBUTTONDOWN表示鼠标左键按下。而MOUSEEVENTF_LEFTDOWN映射的是物理鼠标按键,input[0].mi.dwFlags = MOUSEEVENTF_LEFTDOWN对应已切换功能的真实鼠标左键,所以实际表示发送的是鼠标右键消息。MOUSEEVENTF_LEFTUP意义相仿。
3、在应用程序的ExitInstance()中卸载全局鼠标钩子:
UnhookWindowsHookEx(hhkMouseHook);
这样就完成了所需要的功能。当按下"Ctrl+鼠标左键" 或 "数字键盘减号键 + 鼠标左键"时,相当于按了右键。