3

我目前正在研究一个处理设置热键以执行操作的项目。我有热键功能,但如果键仅由修饰符组成,我不想允许设置热键。例如,Ctrl+F是有效的热键,但Ctrl+Alt不是。

我尝试了几种方法将修饰键从实际的 Keys 枚举对象中剥离,然后对照 Keys.None 检查它以确定它是否仅由修饰键组成。然而,在实践中,这并不是那么好。

我首先尝试了这种方法:

private Keys StripModifiers(Keys Key)
{
    return Key & ~Keys.Modifiers;
}

那没有用,因为我的 Ctrl 键似乎正在使用 LButton | 的 KeyData 属性调用 PreviewKeypress 方法。Shift键 | 控制,Keys.Modifiers 位掩码显然没有完全捕捉到。

我尝试编写自己的,更复杂的:

private Keys StripModifiers(Keys Key)
{
    return Key &
        ~Keys.Alt &
        ~Keys.CapsLock &
        ~Keys.Control &
        ~Keys.ControlKey &
        ~Keys.LControlKey &
        ~Keys.LMenu &
        ~Keys.LShiftKey &
        ~Keys.LWin &
        ~Keys.MButton &
        ~Keys.Menu &
        ~Keys.NumLock &
        ~Keys.RButton &
        ~Keys.RControlKey &
        ~Keys.RMenu &
        ~Keys.RShiftKey &
        ~Keys.RWin &
        ~Keys.Scroll &
        ~Keys.Shift &
        ~Keys.ShiftKey;
}

但是,这也不起作用,因为按下 A 键,它只是调用了 KeyData 属性为 65 的事件,将其减少为 Keys.None,因此限制性太强。

在这一点上我几乎不知所措,以前有没有人遇到过或解决过这个问题?

4

1 回答 1

2

您的原始代码是正确的。你可以让它更清晰一点,Key & Keys.KeyCode但这会产生相同的结果。所以你需要在一个选项窗口中使用这个方法:

private static IsValidShortcutKey(Keys key) {
    return (key & Keys.KeyCode) != Keys.None;
}

然后,您在谈论 ProcessKeyPreview() 时会有些迷茫。您应该使用 ProcessCmdKey() 而不是顺便说一句。那是后来,在用户选择了一个被 IsValidShortcutKey() 认为有效的密钥之后很久。是的,任何键都会使该方法运行,无论它是否与您选择的快捷方式匹配。由于 Keys.ToString() 方法在显示真正的击键方面做得很糟糕,这是由具有 [Flags] 属性的 Keys 枚举引起的。转换为 int 以查看实际值。

所以你应该有两个非常不同的代码块。选项菜单中的一个块,用于检查请求的击键是否有效。它应该使用 IsValidShortcutKey()。

还有另一个检查键是否被按下的块。这应该类似于这样:

    protected override bool ProcessCmdKey(ref Message msg, Keys keyData) {
        if (keyData == keySelectedInConfig) {
            RunOperationSelectedInConfig();
            return true;
        }
        return base.ProcessCmdKey(ref msg, keyData);
    }
于 2012-05-18T01:26:45.840 回答