低级键盘挂钩:区分键代码

Low level keyboard hook: differentiate between key codes

本文关键字:代码 键盘      更新时间:2023-10-16

我试图限制程序用户对键盘的访问。为此,我定义了一个低级键盘挂钩:

LRESULT CALLBACK lowLevelKeyboardProc(int key, WPARAM wParam, LPARAM lParam)
{
    KBDLLHOOKSTRUCT* pkbhs = (KBDLLHOOKSTRUCT*)lParam;
    switch (key)
    {
        case HC_ACTION:
....

并钩住它:

m_hHook = SetWindowsHookEx(WH_KEYBOARD_LL, (HOOKPROC)lowLevelKeyboardProc, 0, 0);

我需要用户只能使用字母数字字符,~,@,#。。。只有密码中可以包含的字符(可打印字符)。使用低级键盘挂钩参数(int key、WPARAM WPARAM、LPARAM LPARAM)来区分这些字符和所有其他字符的最简单方法是什么?

该程序使用c++编写,并使用VC2010进行编译。

感谢所有的帮助!

在真正回答您的问题之前,我有几个问题要问您:

  1. 为什么要使用全局低级键盘挂钩全局执行此操作?

    尽管有时它们是解决问题的唯一方法,但由于多种原因,通常强烈不鼓励使用这样的全局钩子。有许多更好的方法可以防止用户输入无效或不可接受的数据。例如,我只会禁用某个特定控件或一组控件(例如,所有文本框)不需要的键。这样,用户仍然可以使用键盘快捷键和其他非字母数字键与应用程序交互,这对于可访问性来说至关重要。请记住,全局钩子会影响机器上运行的所有其他线程,而不仅仅是你的应用程序——这可能不是你想要的。

  2. 即使您确实决定使用全局挂钩,您确定真的需要禁用所有非字母数字键吗?Backspace和Delete怎么办?难道用户不应该删除东西吗?那Tab呢?进来那么修改键呢,比如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);