WinForms: Как программно запускать обработчик событий?

Я хочу программно вызывать обработчик события для элемента управления. Например:

DateTimePicker dtpLastConsummated;

Я хочу вызвать обработчик событий TextChanged для dtpLastConsummated, как я могу это сделать?

В других языках я бы назвал нечто похожее на:

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

3 предложения для запуска события TextChanged:

Вручную изменить текст:

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

или

Наследовать от DateTimePicker и создать новый метод, который предоставляет/вызывает DateTimePicker protected OnTextChanged

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

или

Если вам не нравится OOP и вы хотите разбить инкапсуляцию, вы можете получить доступ к защищенному методу OnTextChanged с помощью отражения:

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

Ответ 2

Кнопка в Windows Forms является особым случаем, потому что она имеет метод PerformClick, чтобы делать именно то, о чем вы говорите. Однако для других событий в других элементах управления действительно ничего подобного.

Ответ 3

Вы можете наследовать этот класс и создать новый метод, который вызывает защищенный метод OnSomeEvent().

class MyButton : Button
{
    public void CauseClick()
    {
        this.OnClick();
    }
}

Как отметил Мэтт, вы можете использовать PerformClick() для этого конкретного примера. Однако большинство событий не имеют соответствующих публичных функций для их запуска.

Ответ 4

Следующий код помог мне запустить событие

ddlTopic_SelectedIndexChanged(this, new EventArgs());

Фактическое событие

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