我已经看到了几个答案,但不知怎的,我不能让我的工作。我想动态使用各种控件的任何事件(文本框,复选框,按钮等),最好将它们分配给一个事件处理程序。应在运行时分配处理程序。此外,我想在处理程序中知道哪个事件触发了处理程序。
我得到了部分工作。使用lambda表达式我调用我的处理程序(EventAssistant)并传递一个包含事件名称的额外参数(命令)。它适用于使用EventHandler类型的事件。但是,它不适用于期望不同处理程序的事件,例如类型MouseEventHandler。它无法在AddEventHandler订阅。
private void RegisterEventHandlers(Control ctl)
{
foreach (Command command in CommandList)
{
EventInfo eventInfo = ctl.GetType().GetEvent(command.Name);
EventHandler handler = (sender, args) =>
{
EventAssistant(sender, args, command);
};
eventInfo.AddEventHandler(ctl, handler);
}
}
public void EventAssistant(object sender, EventArgs e, Command c)
{
//do lots of other fun stuff
}
基于C# passing extra parameters to an event handler?
作为替代方案,我尝试使用表达式树来解决问题,如下所示:Why am I getting an Argument exception when creating event handler dynamically? 显然,可以从EventInfo中检索EventHandlerType并在lambda表达式中使用。
但是,无论我做什么,我总是会得到一个InvalidOperationException“Lambda参数不在范围内”。
private void RegisterEventHandlers(Control ctl)
{
foreach (Command command in CommandList)
{
EventInfo eventInfo = ctl.GetType().GetEvent(command.Name);
var sender = Expression.Parameter(typeof(object), "sender");
var e = Expression.Parameter(typeof(EventArgs), "e");
var c = Expression.Parameter(typeof(Command), "command");
Expression[] arg = new Expression[] { sender, e, c };
MethodInfo mi = this.GetType().GetMethod("EventAssistant");
var body = Expression.Call(Expression.Constant(this), mi, arg);
var lambda = Expression.Lambda(eventInfo.EventHandlerType, body, sender, e);
eventInfo.AddEventHandler(ctl, lambda.Compile());
}
}
我在表达树上做错了什么?
此外,第一段代码看起来更干净。是否有可能使用第一个代码示例得到我想要的东西?
答案 0 :(得分:1)
在您的第二次尝试中,变量c
不应该是ParameterExpression
,而是ConstantExpression
,而值设置为当前command
。使用当前代码,您将创建一个处理程序,基本上如下所示:
(_sender, _e) => this.EventAssistant(_sender, _e, _c)
// The expression expects "_c" to be a parameter of the lambda, which is why
// you're getting that exception
但是,如果你改变了
var c = Expression.Parameter(typeof(Command), "command");
到
var c = Expression.Constant(command);
生成的代码将按预期显示(并且当然可以工作):
(_sender, _e) => this.EventAssistant(_sender, _e, command)