在应用程序级别使用C#替换enter键

时间:2013-09-05 10:32:52

标签: c# winforms events replace key

我正在开发一个大型的C#winforms项目。在向我的最终用户解释了数千次他们必须按Tab而不是输入文本框,数据网格以及任何地方之后,我决定在某处添加一个复选框,因此用户可以选择设置是否要用tab替换enter。我自己并不喜欢它,因为我认为会发生奇怪的事情,但我想尝试一下。

问题是我有很多表格,而且很多地方我必须设置一个keydown事件或类似事件。我想在应用程序级别将所有这些放在一个地方。有办法吗?

4 个答案:

答案 0 :(得分:0)

我想这是不可能的,因为一些控件会直接暴露keydown事件(例如在gridview的单元格中)。您可以递归地遍历表单中的所有控件,并为基本控件分配事件。 然后可以在中心位置处理事件本身

答案 1 :(得分:0)

我建议你创建一个单独的类,构造函数接受你需要的参数(比如textbox),你创建global variables并将参数分配给构造函数中的变量。 / p>

然后在类中创建事件处理程序,然后可以使用变量将代码放在事件处理程序中。

然后,您可以在需要keydown事件

的地方调用该课程

答案 2 :(得分:0)

表单级别(您可以在基本表单中实现行为并从中继承):

this.KeyPreview = true;
 protected override void OnKeyDown(KeyEventArgs e)
        {
            if (e.KeyCode == Keys.Enter)
                PressedEnter();
            base.OnKeyDown(e);
        }
        private bool PressedEnter()
        {
           bool res = false; // true if handled
           Control ctr = GetFocusedControl();
           if (ctr != null && ctr is TextBox)
           {
               res = this.SelectNextControl(ctr, true, true, true, true);
           }
           return res;
        }

        [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.Winapi)]
        internal static extern IntPtr GetFocus();
        private Control GetFocusedControl()
        {
            Control focusedControl = null;
            IntPtr focusedHandle = GetFocus();
            if (focusedHandle != IntPtr.Zero)
                // if control is not a .Net control will return null
                focusedControl = Control.FromHandle(focusedHandle);
            return focusedControl;
        }

它也可以在应用程序级别完成:在主窗体中,您必须预先过滤来自消息循环的消息(使用消息过滤器:Application.AddMessageFilter(您的过滤器)),检查消息WM_KEYDOWN = 0x100,检查如果按下的键是ENTER,那么就像上面一样处理它。你只做一次,在你的主表单中,它将适用于你所有的孩子表格。

在您的主要表单类中:

protected override void OnLoad(EventArgs e)
        {
            base.OnLoad(e);

            this.mouseMessageFilter = new MouseMoveMessageFilter();
            this.mouseMessageFilter.TargetForm = this;
            Application.AddMessageFilter(this.mouseMessageFilter);
        }

protected override void OnClosed(EventArgs e)
        {
            Application.RemoveMessageFilter(this.mouseMessageFilter);

            base.OnClosed(e);
        }

private class MouseMoveMessageFilter : IMessageFilter
        {
            public FormMain TargetForm { get; set; }

            public bool PreFilterMessage(ref Message m)
            {
                if (TargetForm.IsDisposed) return false;

                int numMsg = m.Msg;

                int VK_RETURN=0x0D;
                if (m.Msg == 0x100 &&(int)m.WParam == VK_RETURN) // WM_KEYDOWN and enter pressed
                {
                      if (TargetForm.PressedEnter()) return true;
                }

                return false;
            }
        }

来源:

https://stackoverflow.com/a/435510/891715

http://msdn.microsoft.com/en-us/library/windows/desktop/dd375731(v=vs.85).aspx

http://www.autohotkey.com/docs/misc/SendMessageList.htm

答案 3 :(得分:0)

MessageFilterSendKeys结合使用要简单得多:

public partial class Form1 : Form, IMessageFilter
{
    public Form1()
    {
        InitializeComponent();
        Application.AddMessageFilter(this);
    }        
    public bool PreFilterMessage(ref Message m)
    {
        if (m.Msg == 0x100)//WM_KEYDOWN
        {
            if (m.WParam.ToInt32() == 0xd)//VK_RETURN = 0xd
            {         
                SendKeys.Send("{TAB}");                                        
                return true; //Discard the Enter key
            }
        }
        return false;
    }
}