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

时间:2011-12-10 19:05:25

标签: winforms keyboard

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

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

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

 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;
    }
在我按下CTRL键后,似乎立即调用了ProcessCmdKey。

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

Capture Key Sequence via ProcessCmdKey

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

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

Capture multiple key downs in C#

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

[DllImport ("user32.dll")]

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.");
  }
}

感谢您查看我的问题。

更新

我已经为我的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);

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

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

更新

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

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

谢谢。

2 个答案:

答案 0 :(得分:3)

将表单的KeyPreview属性设置为true。此属性的摘要说:“获取或设置一个值,该值指示在将事件传递给具有焦点的控件之前表单是否将接收键事件。”然后使用KeyUp事件。除非KeyPressed它还提供有关特殊键的信息,如控制键。

答案 1 :(得分:1)

我通过这样做解决了这个问题,

<mat-card>