WinForms:如何通过编程方式触发事件处理程序?

20

我想通过编程的方式调用控件的事件处理程序。例如:

DateTimePicker dtpLastConsummated;

我希望触发dtpLastConsummated的TextChanged事件处理程序,我该怎么做?

在其他语言中,我会调用类似于以下内容的东西:

dtpLastConsummated.TextChanged(this, new EventArgs());

但在.NET中,您可以拥有多个事件处理程序:

dtpLastConsummated.Click +=new EventHandler(dtpLastConsummated_TextChanged);
dtpLastConsummated.Click +=new EventHandler(dtpLastConsummated_AnotherHandler);
dtpLastConsummated.Click +=new EventHandler(dtpLastConsummated_MoreHandlers);
...
dtpLastConsummated.Click +=new EventHandler(dtpLastConsummated_Nminus1);

所以你需要一种方法来触发所有附加的事件处理程序。


答案

以下代码将触发该事件:

Toolkit.FireEvent(dtpLastConsummated, "TextChanged", new EventArgs());

这是静态工具包函数的代码:

/// <summary>
/// Programatically fire an event handler of an object
/// </summary>
/// <param name="targetObject"></param>
/// <param name="eventName"></param>
/// <param name="e"></param>
public static void FireEvent(Object targetObject, string eventName, EventArgs e)
{
   /*
    * By convention event handlers are internally called by a protected
    * method called OnEventName
    * e.g.
    *     public event TextChanged
    * is triggered by
    *     protected void OnTextChanged
    * 
    * If the object didn't create an OnXxxx protected method,
    * then you're screwed. But your alternative was over override
    * the method and call it - so you'd be screwed the other way too.
    */

   //Event thrower method name //e.g. OnTextChanged
   String methodName = "On" + eventName;

   MethodInfo mi = targetObject.GetType().GetMethod(
         methodName, 
         BindingFlags.Instance | BindingFlags.NonPublic);

   if (mi == null)
      throw new ArgumentException("Cannot find event thrower named "+methodName);

   mi.Invoke(targetObject, new object[] { e });
}

注意:我并不会为.NET框架中的每一个控件和第三方控件创建子类,并说服企业级开发人员将每个窗体改为使用我的自定义控件。


1
Ian的回答很好,但如果你把答案放在“答案”部分,它就可以得到适当的投票了。 - Ocean Airdrop
4个回答

14

触发TextChanged事件的三个建议:

手动更改文本:

        string s = dateTimePicker1.Text;
        dateTimePicker1.Text = String.Empty;
        dateTimePicker1.Text = s;

继承自DateTimePicker并创建一个新方法,将DateTimePicker受保护的OnTextChanged暴露/调用

public class MyDateTimePicker : DateTimePicker
{
    public void OnTextChanged(EventArgs e)
    {
        base.OnTextChanged(e);
    }
}

如果您不喜欢面向对象编程并想要打破封装性,您可以通过反射访问受保护的OnTextChanged方法:

        MethodInfo onTextChanged = dateTimePicker1.GetType().GetMethod("OnTextChanged", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);
        onTextChanged.Invoke(dateTimePicker1, new object[] { new EventArgs() });

就像其他对象库一样,如果不遵守封装原则,你无法走得太远。 - Ian Boyd

8

在Windows Forms中,按钮是一个特殊的情况,因为它有一个PerformClick方法可以精确地执行你所说的操作。但对于其他控件中的其他事件,实际上没有类似的方法。


1
实际上,如果触发事件的代码位于Control子类的方法中,您可以使用InvokeOnClick来执行操作:InvokeOnClick(control, new EventArgs())。请注意,它是一个受保护的实例方法(不使用this),而不是像人们所期望的公共静态方法。 - Andres Riofrio

4
您可以继承该类并创建一个新方法,调用受保护的OnSomeEvent()方法。
class MyButton : Button
{
    public void CauseClick()
    {
        this.OnClick();
    }
}

如Matt所指出的那样,您可以在这个具体的例子中使用PerformClick()。然而,大多数事件没有相应的公共函数来触发它们。

0

以下代码帮助我触发事件

ddlTopic_SelectedIndexChanged(this, new EventArgs());

实际事件

private void ddlTopic_SelectedIndexChanged(object sender, EventArgs e)
{
   // do something
}

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