KeyDown事件没有触发,而KeyPreview被设置为true。

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;
        }
    }
}

1
表单不能获得焦点,控件必须获得焦点。当你尝试将焦点设置到表单时,Windows会将焦点设置到表单上接受用户输入的第一个控件上。 - Cody Gray
1
就我所知,KeyPreviewKeyUp 结合使用效果更佳,并且可以正确处理修饰键 SHIFT、ALT 和 CTRL。相比使用 ProcessCmdKey,我发现这种方法更可取,因为在那里,Shift 键是单独处理的,而键组合则需要进行状态跟踪。 - Memetican
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);
}

3

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

1. 使用{{link2:Application.AddMessageFilter方法}}:

定义一个消息过滤器类:

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 Forms 中的消息都通过它。您可以对事件进行任何操作。如果 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的生命周期内处于活动状态。

注意:这将捕获任何窗体中的事件!如果您只想对一个窗体起作用,请将该窗体传递给过滤器类,并在PreFilterMessage中将其Handle属性与m.HWnd进行比较。

2. 使用Windows Hooks

这是一种更高级和复杂(且低级)的方法。需要更多的代码。我已经编写了一个HookManager类,使该过程非常简单。我将发布该类到Github并撰写一篇文章。


这个很有效!现在需要弄清楚主机是否获得了焦点。当然,它没有将Focused属性设置为true -.- - Stig-Rune Skansgård
@Stig-RuneSkansgård,我刚在答案中添加了一条注释。如果你没弄明白,告诉我让我澄清一下。 - Mohammad Dehghan
谢谢,我不认为这会是个问题,虽然我不确定如何检测按键是否在主窗体控件中被按下,因为所有通常的控件告诉我们它们已经“损坏”了:P 简而言之,现在我可以获取表单上的所有按键,但无法区分我想要的控件中的按键:P - Stig-Rune Skansgård
在主机上发现了一个解决我的问题的事件,但正是这篇文章让我找到了绕过问题制造者的方法,所以你将获得赏金 :) 非常感谢! :) - Stig-Rune Skansgård

1
你观察到的行为原因是,像TAB、上/下/左/右箭头、页面上/下、HOME和END等特殊键通常被普通控件视为“输入键”。
例如,ARROW键被TabControl视为“输入键”,因为这些键允许您更改所选的TabPage。在多行文本框中,类似的行为存在,其中ARROWS键允许您移动文本光标。
我认为你所拥有的Rumba Mainframe控件出于同样的原因做了同样的事情。您可以尝试重写它并更改IsInputKey方法的实现或处理PreviewKeyDown事件并将IsInputKey属性设置为true。
请参阅Control.IsInputKey MethodControl.PreviewKeyDown Event的文档以获取更多详细信息。

顺便说一下,PreviewKeyDown从来没有被触发。 - Stig-Rune Skansgård

0

箭头键是一种特殊的按键,由控件自动处理。

如果您想让它们触发KeyDown事件,您可以:

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

或者

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

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

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


  1. 听起来非常破解。另外,你会如何处理这个问题?
  2. 当 Rumba Mainframe 获得焦点时,PreviewKeyDown 永远不会被触发。无论是主机框架还是表单的焦点都不会触发。
- Stig-Rune Skansgård
选项1真的很糟糕,但这是微软建议的,所以我指出了它。 关于第二个问题,它只在某些控件获得焦点时才会触发。我认为他在处理他试图添加到表单中的控件的KeyDown事件时遇到了问题。请注意,他在代码片段中省略了控件。 - gibertoni

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接