WPF 学习笔记 路由事件
1. 可传递的消息: WPF的UI是由布局组建和控件构成的树形结构,当这棵树上的某个节点激发出某个事件时,程序员可以选择以传统的直接事件模式让响应者来响应之,也可以让这个事件在UI组件树沿着一定的方向传递且路过多个中转结点,并让这个路由过程被恰当的处理。
2,WPF有两种树,Logical Tree和Visual tree。 LogicTree上,充当叶子的一般都是控件,如果我们把WPF的控件也放在“放大镜下观察”,你会发现每个WPF空间本身也是一棵更细微级别的组件组成的树。用来观察WPF控件的放大镜是我们提到的Blend。如果把Logical Tree衍生至Template组件级别,我们的到的就是Visual Tree。
3,路由事件是沿着Visual Tree传递的。
4.一个事件包括5方面:
- 事件的拥有者
- 事件
- 事件的响应者
- 事件处理器
- 订阅关系
5. 路由事件的原理: 舍弃直接事件响应者,让不同的控件变成监听者。 而事件拥有者,只负责出发事件。
6,使用WPF内置的路由事件:
Wpf系统的大多数事件都是可路由事件。我们以Button的Click事件来说明路由事件的使用。
1,如何添加监听者:
this.gridRoot.AddHandler(Button.ClickEvent, new RoutedEventHandler(DefineMethod));
上面这个AddHandler方法来自URElement类。 第一个参数是Button.ClickEvent, 这个叫做路由事件,这里也使用了类似依赖属性包装器的方式。
这里要注意的是,路由事件方法中RoutedEventHandler.Source 是gridRoot. 如果想要得到Button的话,使用RoutedEventHandler.OrigenalSource.
可以在Xaml中使用简化的方法: <Grid X:Name= “gridRoot” BackGround=”Lime” Button.Click=”ButtonClicked”>
7,自定义路由事件
三个步骤:
- 声明并注册路由事件
- 为路由事件添加CLR事件包装
- 创建可以激发路由事件的方法
看Code:
public abstract class ButtonBase: ContentControl, ICommandSource { public static readonly RoutedEvent ClickEvent = /*注册路由事件*/ EventManager.RegisterRoutedEvent("Click",RoutingStrategy.Bubble,typeof(RoutedEventHandler),typeof(ButtonBase)); public event RoutedEventHandler Click { add {this.AddHandler(ClickEvent,value)} Remove {this.RemoveHandler(ClickEvent, value)} } protected virtual void OnClick() { RoutedEventArgs newEvent = new RoutedEventArgs(ButtonBase.ClickEvent,this); this.RaiseEVent(newEvent); } }
路由策略:
- Bubble 向上
- Tunnel 向下
- Direct 模仿CLR直接事件
Demo 2: 实现一个继承自Button的TimeButton类,并添加路由事件
//创建一个RoutedEventArgs类: class ReportTimeEventArgs:RoutedEventArgs { Public ReportTimeEventArgs(RoutedEvent routedEvent, object source) :Base(routedEvent,source) {} Public DateTime ClickTime {get;set;} }
class TimeButton : Button { //生命和注册路由事件 public static readonly RoutedEvent ReportTimeEvent = EventManager.RegisterRoutedEvent ("ReportTime",RoutingStrategy.Bubble,typeof(EventHandler<ReportTimeEventArgs>),typeof(TimeButton)); //CLR 事件包装器 public event RoutedEventhandler ReportTime { add {this.AddHandler(ReportTimeEvent , value) ;} remove {this.RemoveHandler(ReportTimeEvent,value); } //激发路由事件,借用Click事件的激发方法 protected override void OnClick() { base.OnClick(); ReportTimeEventArgs args = new ReportTimeEventArgs(ReportTimeEvent, this); args.ClickTime = DateTime.Now; this.RaiseEvent(args); } //看看怎么使用吧
local:TimeButton.ReportTime = “ReportTimeHandler”;
//ReportTimeEvent 路由事件处理器
private void ReportTimeHandler(objecet sender, ReportTimeEventArgs e)
{
FrameworkElement element = sender as FrameworkElement;
string timeStr = e.ClickTime.ToLongTimeString();
string content = string.Format(“{0} 到达 {1}”,timeStr,element.Name);
this.listBox.Items.Add(content);
}
8, 如何让路由事件停止传播:
使用e.handled 如果e.handled==true,则停止传递。
9. RoutedEventArgs的Source和OriginalSource
Source是logicTree的消息源头,而OriginalSource是visualTree上的源头。
如果一个Button是在UserControl中,那么Source应该是UserControl , OriginalSource 应该是 Button/
10. 事件也附加----深入浅出附加事件
那些类拥有附加事件
Binding类: SourceUpdated事件、 TargetUpdated事件 Mouse类: MouseEnter事件、MouseLeave事件、MouseDown事件、MouseUp事件等 Keyboard类: KeyDown事件、KeyUp事件对比路由事件会发现: 附加事件不具备显示在用户界面上的能力。看一个Demo:设计一个Student类,如果Student实例的Name属性值发生了变化就激发一个路由事件,然后使用界面元素来捕捉这个事件。public class Student { public static readonly RoutedEvent NameChangedEvent = EventManager.ResisterRoutedEvent ( "NameChanged",routingStrategy.Bubble,typeof(RoutedEventHandler),typeof(Student)); public int Id {get;set;} public string Name {get;set;} }
然后我们设计一个button.<Button x:Name = "button1" Content="OK" Width="80" Height="80" Click="Button_Click">
看看后台代码//添加事件监听器this.gridMain.AddHandler(Student.nameChangedEvent, new RoutedEventHandler(this.DtudentnameChangedHandler))
//Click 事件处理器private void Button_Click(objcet sender, RoutedEventArgs e) { Student stu = new Student() {Id=10,Name="Tim"}; stu.Name="Tom"; //准备事件消息并发送路由事件
//附加事件宿主是没有办法发送路由事件的,要借助一个FrameworkElement来RaiseEvent(arg)
//其中RoutedEventArgs 有两个参数,一个附加事件,一个是实例。 RoutedEventArgs arg = new RoutedEventArgs(Student.NameChangedEvent,stu); this.button1.RaiseEvent(arg); } //Grid 捕捉到nameChangedEvent后的处理器 private void StudentNameChangedHandler(object sender, RoutedEventArgs e) { MessageBox.Show((e.OriginalSource as Student).Id.ToString()); }
11. 事件也附加2
其实上面那个例子已经是一个附加文件了,但是微软的官方文档约定要为这个附加事件添加一个CLR包装以便XAML编辑器识别并进行只能提示。 但是,因为Student类不是UIElement的派生类,因为不具备Addhandler和 RemoveHandler这两个方法,所以不能使用CLR属性作为包装器:
为目标UI元素附加事件侦听器的包装器是一个名为Add*Handler的public static 方法,星号代表事件名称public class Student { //声明并定义路由事件 public static readonly RoutedEvent NameChangedEvent = EventManager.RegisterRouredEvent ("NameChanged",RoutingStrategy.Bubble,typeof(RoutedEventHandler),typdof(Student)); //为界面元素添加路由事件侦听 public static void AddNameChangedHandler(DependencyObject d,RoutedEventHandler h) { UIElement e = d as UIElement; if(e!=null) { e.AddHandler(Student.NameChangedEvent, h); } } public static voidRemoveNameChangedHandler(DependencyObject d,RoutedEventHandler h) { UIElement e = d as UIElement; if(e!=null) { e.AddHandler(Student.NameChangedEvent, h); } } public int Id {get;set;} public string Name {get;set;} }
public Window1()
{
Student.AddNameChangedHandler(this.gridMain, new RoutedEventHandler(this.StudnetnameChagnedHandler));
}
再次理解一下附加事件:
UIElement类是路由事件宿主与附加事件宿主的分水岭,不但是因为从UIElemtn类开始才具备了界面上显示的能力,还因为RaiseEvent、AddHandler和RemoveHandler 这些方法也定义在UIElement类中。 如果在一个非UIElement派生类中注册了路由事件,则这个类的实例既不能自己激发,也无法自己侦听此路由事件。
转载:http://www.cnblogs.com/zhaoyun2007/archive/2012/12/06/2804581.html