3

我正在构建一个小型表单应用程序,我刚刚启动它。但是我有这个问题:如果我在表单中放置一个控件,则 KeyDown 事件不会触发。我知道 KeyPreview 属性,并将其设置为 true。但这并没有帮助...... :(我也尝试将焦点设置为主要形式,也没有成功。

有什么想法吗?

编辑:

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
        KeyDown += new KeyEventHandler(Form1_KeyDown);
        this.KeyPreview = true;
    }

    void Form1_KeyDown(object sender, KeyEventArgs e)
    {
        switch (e.KeyCode)
        {
            case Keys.Left: MessageBox.Show("Left");
                break;
            case Keys.Right: MessageBox.Show("Right");
                break;
        }
    }
}
4

4 回答 4

9

我已经评论了我的解决方案,但我也将其作为答案发布,因此很容易找到。

protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
{
    switch (keyData)
    {
        case Keys.Left:
            // left arrow key pressed
            return true;
        case Keys.Right:
            // right arrow key pressed
            return true;
        case Keys.Up:
            // up arrow key pressed
            return true;
        case Keys.Down:
            // down arrow key pressed
            return true;
    }

    return base.ProcessCmdKey(ref msg, keyData);
}
于 2013-05-17T10:41:32.273 回答
3

如果您在 WPF 上,您可以轻松捕获所需的事件,因为 WPF 使用路由事件系统来调度事件。在 winforms 中,我推荐以下两种方式之一:

1.使用Application.AddMessageFilter Method

定义一个消息过滤器类:

public class KeyMessageFilter : IMessageFilter
{
    private enum KeyMessages
    {
        WM_KEYFIRST = 0x100,
        WM_KEYDOWN = 0x100,
        WM_KEYUP = 0x101,
        WM_CHAR = 0x102,
        WM_SYSKEYDOWN = 0x0104,
        WM_SYSKEYUP = 0x0105,
        WM_SYSCHAR = 0x0106,
    }

    [DllImport("user32.dll")]
    private static extern IntPtr GetParent(IntPtr hwnd);

    // We check the events agains this control to only handle
    // key event that happend inside this control.
    Control _control;

    public KeyMessageFilter()
    { }

    public KeyMessageFilter(Control c)
    {
        _control = c;
    }

    public bool PreFilterMessage(ref Message m)
    {
        if (m.Msg == (int)KeyMessages.WM_KEYDOWN)
        {
            if (_control != null)
            {
                IntPtr hwnd = m.HWnd;
                IntPtr handle = _control.Handle;
                while (hwnd != IntPtr.Zero && handle != hwnd)
                {
                    hwnd = GetParent(hwnd);
                }
                if (hwnd == IntPtr.Zero) // Didn't found the window. We are not interested in the event.
                    return false;
            }
            Keys key = (Keys)m.WParam;
            switch (key)
            {
                case Keys.Left:
                    MessageBox.Show("Left");
                    return true;
                case Keys.Right:
                    MessageBox.Show("Right");
                    return true;
            }
        }
        return false;
    }
}

所以你有一个类,Windows 窗体中的每条消息都通过它。你可以对事件做任何你想做的事情。如果PreFilterMessage方法返回 true,则意味着不应将事件分派给它的相应控件。

(请注意,Keys枚举中的值几乎与虚拟键码相同)

在此之前,您必须将其添加到应用程序的消息过滤器中:

public partial class Form1 : Form
{
    // We need an instance of the filter class
    KeyMessageFilter filter;

    public Form1()
    {
        InitializeComponent();

        filter = new KeyMessageFilter(panel1);
        // add the filter
        Application.AddMessageFilter(filter);
    }

    protected override void OnFormClosed(FormClosedEventArgs e)
    {
        base.OnFormClosed(e);

        // remove the filter
        Application.RemoveMessageFilter(filter);
    }
}

该过滤器仅在Form1.

注意:这将捕获任何形式的事件!如果您希望它仅适用于一种表单,请将表单传递给过滤器类,并将其Handle属性与m.HWndin进行比较PreFilterMessage

2. 使用Windows 挂钩

这是一种更高级和更复杂(和低级)的方法。它需要更多的代码。我写了一个HookManager类,使这个过程非常简单。我将把课程发布到 github 并写一篇关于它的文章。

于 2013-02-27T09:22:59.433 回答
1

您观察到的行为的原因是特殊键(如 TAB、UP/DOWN/LEFT/RIGHT ARROW、PAGE UP/DOWN、HOME、END 等)通常被常用控件视为“输入键”。

例如,箭头键被 TabControl 视为“输入键”,因为这些键允许您更改选定的 TabPage。多行文本框也存在类似的行为,其中箭头键允许您移动文本光标。

我假设您拥有的 Rumba Mainframe 控件出于相同的原因做同样的事情。您可以尝试覆盖它并更改 IsInputKey 方法的实现或处理 PreviewKeyDown 事件并将 IsInputKey 属性设置为 true。

有关详细信息,请参阅Control.IsInputKey 方法Control.PreviewKeyDown 事件的文档

于 2013-02-26T15:14:35.483 回答
0

方向键是一种特殊键,由 Controls 自动处理。因此,如果您想让他们引发 KeyDown 事件,您可以:

1) 在表单的每个控件中覆盖 isInputKey 方法

或者

2) 处理 PreviewKeyDown 事件并将 IsInputKey 属性设置为 true

更多信息可以在这里找到。

我知道 WonderCsabo 已经解决了他的问题,但其他人悬赏它,因为遇到了同样的问题并且没有选择答案。WonderCsabo 也请发布您的解决方案作为答案。

于 2013-02-26T15:12:08.170 回答