WinForms - 使用ProcessCmdKey VS KeyDown捕获按键组合

noc*_*ier 5 keyboard winforms

我的目标是实现一个自定义Control + S键按下处理程序,以连接到winforms应用程序中的自定义保存方法.

基于我的研发,有几种方法可以实现这一目标.首先,我尝试了明显的KeyPress事件处理程序.这不足以捕获我需要的按键(它没有在编辑器级别调用,这是我需要的).

看起来更好的第二个选项是受保护的覆盖bool ProcessCmdKey(ref Message msg,Keys keyData)覆盖.这有效 - 它拦截了CTRL键点击,但显然我需要编写额外的代码来坚持按下CTRL键的事实并拦截下一次按键(在我的情况下将是S),然后执行自定义操作.

 protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
    {
        if (keyData == (Keys.S | Keys.Control))
        {
            // This is never called
        }
        else if (keyData == (Keys.Menu | Keys.Alt))
        {
           // this is called each time I hit CTRL
        }
        return true;
    }
Run Code Online (Sandbox Code Playgroud)

在我按下CTRL键后,似乎立即调用了ProcessCmdKey.

这篇文章建议创建一个KeyTracker类,它将持久按下按键并执行它需要做的事情:

通过ProcessCmdKey捕获密钥序列

这似乎是一个不错的选择,但在我开始实现一个纪念品跟踪模式之前,是否有人对如何完成这个看似常见的功能有任何意见?

另一种模式使用GetKeyboardState API函数:

在C#中捕获多个关键点

这看起来很有趣,但我不确定它是否符合我的需求.

[DllImport ("user32.dll")]
Run Code Online (Sandbox Code Playgroud)

public static extern int GetKeyboardState(byte [] keystate);

private void Form1_KeyDown( object sender, KeyEventArgs e )
{
  byte[] keys = new byte[255];

  GetKeyboardState (keys);

  if( keys[(int)Keys.Up] == 129 && keys[(int)Keys.Right] == 129 )
  {
      Console.WriteLine ("Up Arrow key and Right Arrow key down.");
  }
}
Run Code Online (Sandbox Code Playgroud)

谢谢你看看我的问题.

UPDATE

我已经为我的DataPanel添加了三个用于键处理的事件.当我在事件中设置断点时,VS都没有收到这些事件,所以这让我相信ProcessCmdKey是我的最佳选择.

如果我能让这些事件发挥作用,那也会很好:

        // Ctrl + S: Save Support
        this.ParentForm.KeyPreview = true;
        this.KeyPress             += new KeyPressEventHandler(DataPanel_KeyPress);
        this.KeyDown              += new KeyEventHandler(DataPanel_KeyDown);
        this.PreviewKeyDown       += new PreviewKeyDownEventHandler(DataPanel_PreviewKeyDown);
Run Code Online (Sandbox Code Playgroud)

按任意键时似乎没有抓住这些事件:

void DataPanel_PreviewKeyDown(object sender, PreviewKeyDownEventArgs e)
    {
        if (e.KeyCode == (Keys.S | Keys.Control))
        {
            SessionManager.Trace.AddTrace("You Hit Save!!");
        }
    }

    void DataPanel_KeyDown(object sender, KeyEventArgs e)
    {
        if (e.KeyCode == (Keys.S | Keys.Control))
        {
            SessionManager.Trace.AddTrace("You Hit Save!!");
        }
    }

    void DataPanel_KeyPress(object sender, KeyPressEventArgs e)
    {
        var key = e.KeyChar;
    }
Run Code Online (Sandbox Code Playgroud)

UPDATE

我通过使用简单的KeyUp事件和KeyPreview标志解决了这个问题:

 void ShipmentDataPanel_KeyUp(object sender, KeyEventArgs e)
    {
        if (e.Control && e.KeyCode == Keys.S)
        {
            MessageBox.Show("Control + S Key Hit!");
        }
    }
Run Code Online (Sandbox Code Playgroud)

谢谢.

Oli*_*bes 4

KeyPreview表单的属性设置为true。该属性的摘要如下:

获取或设置一个值,该值指示在事件传递到具有焦点的控件之前窗体是否将接收按键事件。

然后使用该KeyUp事件。除非KeyPressed它还提供有关特殊键(如控制键)的信息。