在真正回答你的问题之前,我有几个问题要问你:
为什么要使用全局低级键盘挂钩来全局执行此操作?
尽管有时它们是解决问题的唯一方法,但出于多种原因,通常强烈建议不要使用像这样的全局钩子。有许多更好的方法可以防止用户输入无效或不可接受的数据。例如,我会禁用特定控件或一组控件(例如所有文本框)不需要的键。这样,用户仍然可以使用键盘快捷键和其他非字母数字键与您的应用程序交互,这对于可访问性的原因至关重要。请记住,全局挂钩会影响机器上运行的所有其他线程,而不仅仅是您的应用程序——这可能不是您想要的。
即使您决定使用全局挂钩,您确定您真的需要禁用所有非字母数字键吗?退格和删除呢?用户不应该能够删除东西吗?选项卡呢?进入?那么修饰键呢,例如 Shift、Ctrl 和 Alt:是否允许使用这些修饰键?
在继续进行此设计之前,请认真重新考虑您是否真的需要一个低级挂钩。如果您在设计另一个解决方案时需要帮助,请提出一个新问题,描述您想要完成的任务(例如,我想阻止用户在文本框控件中输入任何非字母数字字符;这是我用来创建我的文本框...)。
但是如果你坚持无视我的建议,解决方案相当简单:调查传递给钩子过程的KBDLLHOOKSTRUCT
结构的成员。该vkCode
成员为您提供所按下键的虚拟键代码。很有可能,这就是您需要的所有信息。但以防万一,密钥的硬件扫描码也在scanCode
会员中提供。
不幸的是,您当前拥有的代码是错误的。钩子回调过程的第一个参数确实是一个int
,但它不是关键代码。相反,它是指示钩子过程应如何处理消息的代码。像这样使用它:
LRESULT CALLBACK LowLevelKeyboardProc(int nCode, WPARAM wParam, LPARAM lParam)
{
// If nCode is greater than or equal to HC_ACTION, process the message.
if (nCode >= HC_ACTION)
{
KBDLLHOOKSTRUCT* pkbhs = reinterpret_cast<KBDLLHOOKSTRUCT*>(lParam);
// Handle the keys as you wish here.
//
// Remember that pkbhs->vkCode gives you the virtual key code
// of the key that was pressed.
//
// To prevent a particular key from being processed, you should
// return a non-zero value (e.g. 1) immediately.
}
// Pass the message on.
return CallNextHookEx(m_hHook, nCode, wParam, lParam);
}
而且当你安装钩子时,绝对不需要强制转换函数指针。像这样毫无意义的强制转换只会隐藏潜在的编译时错误,从而导致运行时崩溃。简单写一下:
m_hHook = SetWindowsHookEx(WH_KEYBOARD_LL, lowLevelKeyboardProc, 0, 0);