C#使用反射从表单控件访问事件名称

时间:2014-10-14 19:30:30

标签: c# winforms events reflection

我目前正在开发一个持续的解决方案,它采用包含表单的程序集,创建这些表单的实例,然后从中提取控件。到目前为止都很好。但是,我收到了另一个请求,我发现这个请求更难实现。

我现在需要做的是在运行时提取与表单上每个控件相关联的事件(特别是实际的事件名称和事件类型)。为了促进这一点,我创建了一个非常简单的winforms项目来模仿我将在完成的应用程序中做的事情。

我已经尝试了多种方式来获取这些事件,而且到目前为止我已经接近了,具体如下:

EventHandlerList events = this.button1.GetType()
                                      .GetProperty("Events", BindingFlags.Instance | 
                                                             BindingFlags.Public |
                                                             BindingFlags.Static | 
                                                             BindingFlags.NonPublic)
                                      .GetValue(this.button1) as EventHandlerList;

在我的按钮内,我创建了2个点击和鼠标离开事件。当我调试上面的代码并检查事件时,我可以看到我创建的两个,但它们存在于“None-public members \ head”下。

我在调试模式中可以看到的事件类似于:

  

{Method = {Void button1_MouseLeave(System.Object,System.EventArgs)}}

根据另一个表单的建议,我尝试了以下内容,看看我是否可以创建事件的委托(并且它不起作用):

var myEvent = events["button1_MouseLeave"];

我不知道从哪里开始。我好像几乎在那里,因为我实际上可以在调试模式下看到事件,但我不确定如何得到它们的名称。

1 个答案:

答案 0 :(得分:1)

你几乎就在那里,只是EventHandlerList是一个不容易迭代的链表。您可以在此处查看源代码:http://referencesource.microsoft.com/#System/compmod/system/componentmodel/EventHandlerList.cs

这将为您提供Control(WinForms)的所有订阅事件

        EventHandlerList events = (EventHandlerList)typeof(Component)
               .GetField("events", BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.GetField)
               .GetValue(this);

        object current = events.GetType()
               .GetFields(BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.GetField)[0]
               .GetValue(events);

        List<Delegate> delegates = new List<Delegate>();
        while(current != null)
        {
            delegates.Add((Delegate)GetField(current,"handler"));
            current = GetField(current,"next");
        }

静态助手

        public static object GetField(object listItem, string fieldName)
        {
            return listItem.GetType()
               .GetField(fieldName, BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.GetField)
               .GetValue(listItem);
        }