3

小さなFormsアプリケーションを作成していますが、開始したばかりです。しかし、私にはこの問題があります。フォームにコントロールを配置すると、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では、次の2つの方法のいずれかをお勧めします。

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フォームのすべてのメッセージが通過するクラスがあります。あなたはイベントであなたがやりたいことを何でもすることができます。メソッドがtrueを返す場合PreFilterMessage、それはイベントがその応答制御にディスパッチされるべきではないことを意味します。

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ます。

注意:これにより、あらゆる形式のイベントがキャッチされます。1つのフォームでのみ機能させる場合は、フォームをフィルタークラスに渡し、その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などの特別なキーが一般的なコントロールによって「入力キー」と見なされることが多いためです。

たとえば、矢印キーを使用すると、選択したTabPageを変更できるため、TabControlでは「入力キー」と見なされます。ARROWSキーを使用してテキストカーソルを移動できる複数行のTextBoxでも、同様の動作が見られます。

あなたが持っているRumbaメインフレームコントロールは同じ理由で同じことをしていると思います。これをオーバーライドしてIsInputKeyメソッドの実装を変更するか、PreviewKeyDownイベントを処理してIsInputKeyプロパティをtrueに設定してみてください。

詳細については、 Control.IsInputKeyメソッドControl.PreviewKeyDownイベントのドキュメントを参照してください。

于 2013-02-26T15:14:35.483 に答える
0

矢印キーは、コントロールによって自動的に処理される特殊なキーの一種です。したがって、KeyDownイベントを発生させる場合は、次のようにできます。

1)フォームのすべてのコントロールでisInputKeyメソッドをオーバーライドします

また

2)PreviewKeyDownイベントを処理し、IsInputKeyプロパティをtrueに設定します

詳細については、こちらをご覧ください

WonderCsaboはすでに彼の問題を解決したことは知っていますが、同じ問題があり、回答が選択されなかったため、他の誰かがそれに報奨金を出しました。WonderCsaboも答えとしてあなたの解決策を投稿してください。

于 2013-02-26T15:12:08.170 に答える