如何创建包含Windows密钥的全局热键组合?

时间:2013-10-14 09:05:53

标签: hotkeys keyboard-hook windows-key

我正在使用SetWindowsHookEx捕获键盘以显示正在运行的应用程序。我可以使用CTRL,ALT,SHIFT和常规键创建组合。但是我无法使用WINDOWS键创建组合(例如,CTRL + WINDOWS + A)。

我见过有关单独捕获WINDOWS键的文章(例如在游戏运行时阻止Windows 8启动屏幕),但从不创建组合。

我知道可以像AutoHotKey这样的软件捕获这些组合。

SetWindowsHookEx是错误的方法吗?

1 个答案:

答案 0 :(得分:0)

我发现了一个类似的问题,其中包含一个帮助here的答案。 这使我走上了解决问题的正确道路。

看来Keyboard.Modifiers无法检测到Windows键(至少在我的Win8实例上)。

相反,我必须使用Keyboard.IsKeyDown以不同的方式处理Windows密钥。这导致构建一种方法来检查按下的键(来自我的LowLevelKeyboardProc)和按下的当前修改键的组合是否与属性HotKeyHotKeyModifiers中定义的键和修饰符相同。

这是LowLevelKeyboardProc的C#:

/// <summary>
/// Called by windows when a keypress occurs.
/// </summary>
/// <param name="nCode">A code the hook procedure uses to determine how to process the message. If nCode is less than zero, the hook procedure must pass the message to the CallNextHookEx function without further processing and should return the value returned by CallNextHookEx.</param>
/// <param name="wParam">The identifier of the keyboard message. This parameter can be one of the following messages: WM_KEYDOWN, WM_KEYUP, WM_SYSKEYDOWN, or WM_SYSKEYUP. </param>
/// <param name="lParam">A pointer to a KBDLLHOOKSTRUCT structure. </param>
private IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
{
    if (nCode >= 0 && (wParam == (IntPtr)WM_KEYDOWN || wParam == (IntPtr)WM_SYSKEYDOWN))
    {
        int vkCode = Marshal.ReadInt32(lParam);
        var keyPressed = KeyInterop.KeyFromVirtualKey(vkCode);

        if (IsKeyCombinationPressed(keyPressed))
            OnKeyCombinationPressed(new EventArgs());
    }

    return CallNextHookEx(_hookId, nCode, wParam, lParam);
}

IsKeyCombinationPressed()方法的C#:

/// <summary>
/// Returns true if the registered key combination is pressed
/// </summary>
/// <remarks>
/// Keyboard.Modifiers doesn't pick up the windows key (at least on Windows 8) so we use Keyboard.IsKeyDown to detect it (if required).
/// </remarks>
bool IsKeyCombinationPressed(Key keyPressed)
{
    if (keyPressed != HotKey) return false;

    //Handle windows key
    bool isWindowsKeyRequired = (HotKeyModifiers & ModifierKeys.Windows) != 0;
    bool isWindowsKeyPressed = Keyboard.IsKeyDown(Key.LWin) || Keyboard.IsKeyDown(Key.RWin);

    //Remove windows key from modifiers (if required)
    ModifierKeys myModifierKeys = isWindowsKeyRequired ? HotKeyModifiers ^ ModifierKeys.Windows : HotKeyModifiers;
    bool isModifierKeysPressed = Keyboard.Modifiers == myModifierKeys;

    return isWindowsKeyRequired
        ? isWindowsKeyPressed && isModifierKeysPressed
        : isModifierKeysPressed;
}