C#中动态订阅控件中任意事件的方法
2010-03-19 14:45
288 查看
这个题目想了半天,不太好用一句话描述。这样,举个简单的应用场景:在用Windows Forms制作向导程序的时候,通常会有“上一步”、“下一步”这样的按钮。假设现在需要做一个通用的“向导制作框架”,那么我们就需要在这个“向导制作框架”中,对“上一步”、“下一步”这些按钮是否可用(是否Enabled)进行控制。而控制条件是由开发人员在实际使用“向导制作框架”进行开发时确定的。比如:只有在当前向导页面的某个文本框里被输入了字符串以后,“下一步”才可用;或者只有在某个按钮被按下的时候,“下一步”才可用。于是,我们的“向导制作框架”要能够允许开发人员来确定,当触发什么事情的时候,“下一步”才可用。
我们先从特例入手,假设向导页面上只有一个按钮叫btn,只有点击btn以后,“下一步”才能够被点击。编程上很容易实现这个效果:直接在窗体上订阅btn的Click事件,在Click事件里,写下“btnNext.Enabled = true;"这样一句话。
现在问题来了:我们需要为开发人员提供一个“向导制作框架”,也就是说,这个框架根本无法预测开发人员需要订阅哪些控件的哪些事件,只能留出一个接口,让开发人员自己调用这个接口实现事件的动态注册。Windows Forms提供的控件类型多种多样,而且不同的事件有着不同的函数签名(也就是委托,比如Click事件和MouseDown事件就是用的两个不同的委托),如何让我们的框架能够支持任意的控件,并在任意控件的任意事件发生时,调用“btnNext.Enabled = true;”这条语句,使得“下一步”按钮可用呢?
要实现这样的功能,我们需要用到反射。首先,定义一个泛型方法,在这个方法里,我们直接对btnNext进行设置,如下:
注意:如果是标准的事件委托,一般情况下都是第一个参数为object类型,第二个参数为EventArgs绑定类型,无返回值的签名格式。换句话说,一般情况下,上面的这段代码返回的数组包含两个对象:object和一个继承于EventArgs(或者是EventArgs本身)的类型。在这里,我们取数组里的第二个成员。
好了,现在通过反射,获得DoTrigger方法的MethodInfo,并通过MethodInfo.MakeGenericMethod方法,将上一步获得的EventArgs类型绑定到DoTrigger方法上,并使用Delegate.CreateDelegate生成Event Handler:
现在,不管是btn被单击,还是textBox1里的文字被更改,还是在textBox2鼠标按钮被按下,都会直接触发DoTrigger函数,进而使得“下一步”按钮变得可用(Enabled为true)。
我们先从特例入手,假设向导页面上只有一个按钮叫btn,只有点击btn以后,“下一步”才能够被点击。编程上很容易实现这个效果:直接在窗体上订阅btn的Click事件,在Click事件里,写下“btnNext.Enabled = true;"这样一句话。
现在问题来了:我们需要为开发人员提供一个“向导制作框架”,也就是说,这个框架根本无法预测开发人员需要订阅哪些控件的哪些事件,只能留出一个接口,让开发人员自己调用这个接口实现事件的动态注册。Windows Forms提供的控件类型多种多样,而且不同的事件有着不同的函数签名(也就是委托,比如Click事件和MouseDown事件就是用的两个不同的委托),如何让我们的框架能够支持任意的控件,并在任意控件的任意事件发生时,调用“btnNext.Enabled = true;”这条语句,使得“下一步”按钮可用呢?
要实现这样的功能,我们需要用到反射。首先,定义一个泛型方法,在这个方法里,我们直接对btnNext进行设置,如下:
private Type[] GetDelegateParameterTypes(Type d) { if (d.BaseType != typeof(MulticastDelegate)) { throw new InvalidOperationException("Not a delegate."); } MethodInfo invoke = d.GetMethod("Invoke"); if (invoke == null) { throw new InvalidOperationException("Not a delegate."); } ParameterInfo[] parameters = invoke.GetParameters(); Type[] typeParameters = new Type[parameters.Length]; for (int i = 0; i < parameters.Length; i++) { typeParameters[i] = parameters[i].ParameterType; } return typeParameters; }
注意:如果是标准的事件委托,一般情况下都是第一个参数为object类型,第二个参数为EventArgs绑定类型,无返回值的签名格式。换句话说,一般情况下,上面的这段代码返回的数组包含两个对象:object和一个继承于EventArgs(或者是EventArgs本身)的类型。在这里,我们取数组里的第二个成员。
好了,现在通过反射,获得DoTrigger方法的MethodInfo,并通过MethodInfo.MakeGenericMethod方法,将上一步获得的EventArgs类型绑定到DoTrigger方法上,并使用Delegate.CreateDelegate生成Event Handler:
通过向上面的方法传入一个任意控件和该控件中任意事件的Method Info,即可获得处理该事件的Event Handler,也就是由DoTrigger泛型方法来处理该指定的事件:public void RegisterTrigger(Control control, string eventName) { try { EventInfo eventInfo = control.GetType().GetEvent(eventName, BindingFlags.Public | BindingFlags.Instance); Delegate d = this.GetEventHandler(control, eventInfo); eventInfo.AddEventHandler(control, d); } catch { throw; } }
最后,在使用的时候,代码就简单啦:
private void Form_Load (object sender, System.EventArgs e)
{
this.RegisterTrigger (btn, "Click");
this.RegisterTrigger (textBox1, "TextChanged");
this.RegisterTrigger (textBox2, "MouseDown");
}
现在,不管是btn被单击,还是textBox1里的文字被更改,还是在textBox2鼠标按钮被按下,都会直接触发DoTrigger函数,进而使得“下一步”按钮变得可用(Enabled为true)。
相关文章推荐
- 转 C#中动态订阅控件中任意事件的方法
- C#动态代码生成控件后其他事件不能获取该控件值的解决方法
- C# 动态生成窗口中的控件不显示的解决方法
- C#动态生成控件以及添加事件处理
- c# ListView控件的常用屬性、方法及事件
- c# 给窗体添加MouseEnter事件,可鼠标移到任意控件均会产生MouseEnter事件,如何识别是哪个产生的呢?
- c# TableLayoutPanel控件应用 动态添加控件及事件
- C# 中的动态创建组件(属性及事件)的实现思路及方法
- C#中删除任意控件任意事件的事件列表的实现!
- c# 动态改变控件大小的方法
- [习题].FindControl()方法 与 PlaceHolder控件 #2(动态加入「子控件」的事件)
- C#动态添加控件,并生成事件
- C# 控件如何获取事件已经注册的方法
- 【转】C# 中动态添加控件及其事件
- c# TableLayoutPanel控件应用 动态添加控件及事件
- Winform(C#)Flash控件 属性 方法 事件
- JS脚本实现动态给标签控件添加事件的方法
- C#中动态创建控件及事件处理程序
- c# winform动态生成窗体及控件,并使用控件事件
- C#给控件动态添加事件