如何处理可自定义的热键设置?

How can I handle a customizable hotkey setting?

我试图让我的应用程序的用户可以选择为某些功能设置键盘热键。

现在我正在使用 TextBoxKeyDown 事件,如下所示:

Key Hotkey;
private void SetHotKey(object sender, KeyEventArgs e)
{
    (sender as TextBox).Text = e.Key.ToString();
    Hotkey = e.Key;
    e.Handled = true;
}

这种方法的问题是我无法设置复杂的快捷方式,例如 [Ctrl]+[F4]。是否有一些第 3 方控件可以帮助解决这个问题?或者更适合在文本框上订阅的事件?

更新:我已经更改了我的代码,但似乎我仍然做错了。

Key Hotkey;
bool lControl = false;
bool lAlt = false;
bool lShift = false;
private void SetHotKey(object sender, KeyEventArgs e)
{
    var k = e.Key;
    if (e.IsDown)
    {
        var tb = sender as TextBox;
        tb.Text = "";
        lControl = Keyboard.IsKeyDown(Key.LeftCtrl);
        lAlt = Keyboard.IsKeyDown(Key.LeftAlt);
        lShift = Keyboard.IsKeyDown(Key.LeftShift);
        if (lControl) tb.Text += "Ctrl+";
        if (lAlt) tb.Text += "Alt+";
        if (lShift) tb.Text += "Shift+";
        tb.Text = e.Key.ToString();
        Hotkey = e.Key;
    }
    e.Handled = true;
}

我怎样才能使它工作并看起来更干净?

WinForms 您可以检查 KeyEventArgs.Modifiers 是否按下了 Ctrl。 https://msdn.microsoft.com/en-us/library/system.windows.forms.keyeventargs.modifiers(v=vs.110).aspx

WPF

您可以使用这个小片段来检查是否按下了 Control 键:

Keyboard.IsKeyDown(Key.LeftCtrl)

如果您想使用 MVVM 模式,您可以使用文本框(或任何其他控件)的 KeyBindings Create Key binding in WPF

让您朝着正确的方向开始。

首先,您需要 System/ImeProcessed/DeadCharProcessed 密钥后面的真实密钥。这可以通过扩展方法来完成,以便于访问。

public static Key RealKey(this KeyEventArgs e)
{
    switch (e.Key)
    {
        case Key.System:
            return e.SystemKey;

        case Key.ImeProcessed:
            return e.ImeProcessedKey;

        case Key.DeadCharProcessed:
            return e.DeadCharProcessedKey;

        default:
            return e.Key;
    }
}

然后你应该将修饰符格式化为快捷方式,而不仅仅是按下的键。您可以使用 Keyboard.ModifierKeys 来获取标志并为了更容易格式化,将它们收集在列表中。而且你应该只阻止修饰键(Ctrl、Alt 和 Shift)更新热键。

private void SetHotKey(object sender, KeyEventArgs e)
{
    var nonShortcuttableKeys = new[] { Key.LeftAlt, Key.RightAlt, Key.LeftCtrl, Key.RightCtrl, Key.LeftShift, Key.RightShift };
    var actualKey = e.RealKey();

    if (e.IsDown && !nonShortcuttableKeys.Contains(actualKey))
    {
        var tb = sender as TextBox;

        var modifiers = new List<ModifierKeys>();
        if (Keyboard.Modifiers.HasFlag(ModifierKeys.Control))
        {
            modifiers.Add(ModifierKeys.Control);
        }

        if (Keyboard.Modifiers.HasFlag(ModifierKeys.Alt))
        {
            modifiers.Add(ModifierKeys.Alt);
        }

        if (Keyboard.Modifiers.HasFlag(ModifierKeys.Shift))
        {
            modifiers.Add(ModifierKeys.Shift);
        }

        tb.Text = modifiers.Count == 0
            ? string.Format("{0}", actualKey)
            : string.Format("{0} + {1}", string.Join(" + ", modifiers), actualKey);

        Hotkey = actualKey;
    }

    e.Handled = true;
}