WPF路由事件
这节讲一下WPF中的路由事件(Routed Event)。
【什么是事件】
在了解路由事件前,我们应先来了解一下什么是事件(Event)。
在Windows系统中,像鼠标单击,双击,移动这样的,都是在触发着一个个事件,事件代表着用户在Windows上的一个动作,相当于用户给系统交代了一个任务让它去执行。本质上事件就是条信息数据,这条数据有对事件的描述,以及携带着事件的参数,这些参数可以看做是事件的“Metadata”,比如你点击鼠标左键,会触发MouseLeftDown和MouseLeftUp这两个事件,它们的参数中就携带了鼠标在屏幕的点击位置(X,Y值)等等信息。
我们回到编程概念中,在事件这个模型中,我们要理解以下三个跟事件有关的抽象:
-
事件的拥有者:事件的拥有者就是事件的触发者,比如按钮被点击,那么按钮就是事件的拥有者;
-
事件的响应者:事件的响应者就是事件的处理者,比如我们在winform后置代码中声明的一个一个事件处理方法,拥有事件处理方法的from体就是事件的响应者;
-
事件订阅关系:要想一个事件被处理,需要让事件的响应者去订阅事件拥有者的事件,在winfrom中这一操作被具象化为在“小闪电”操作栏中对对应的事件关联上后置代码中的事件处理器。
如果说事件的拥有者和响应者是河的两岸,那事件的订阅关系就是连接两岸的桥,让事件的 拥有者通过桥把事件数据交代给事件的响应者,而桥并不是唯一的,事件的订阅可以是多个,它是一对n(n>=0)的关系。
当然,上述事件模型也有其弊端:
-
事件的响应者必须要显式订阅事件才能生效
-
事件的拥有者必须能访问到事件的响应者,这样才能建立订阅关系
所以我们能了解到,原始的事件模型,对于订阅关系的建立有严格的要求,因此,微软在WPF中推出了路由事件,它使得事件可以不再以订阅关系建立,下面来了解一下。
【路由事件】
提到路由事件,首先一点,什么是路由呢?这里引入《深入浅出WPF》一书中对路由的解释:“起点与终点间有若干个中转站,从起点出发后经过每个中转站时要做出选择,最终以正确(比如最短或者最快)的路径到达终点。” 路由描述的就是这样的一个过程。
路由事件,是指事件的拥有者和响应者不必建立订阅关系,拥有者只管激发事件,响应者通过在自身设置事件监听器去监听对应的事件,并可以决定事件是否继续传播,如果说原始事件是两个人窃窃私语的话,那路由事件就是一队人挨个传话。当事件响应者通过事件监听器监听到某个事件的发生,通过事件携带的参数可以获取到事件的来源,从而做出判断该事件是否是自己关心的某个控件激发的,如果是,可以处理并停止事件的传播,如果不是,则放行不予理睬。
请设想如下图所示的一个XAML控件层级关系:
蓝色代表Window控件,其内部有两个按钮和一个Grid布局,按钮2在Grid布局中,当按钮1激发单击事件后,该事件的传播路径为:
按钮1–>Window
当按钮2激发单击事件后,该事件的传播路径为:
按钮2–>Grid–>Window
【如何使用路由事件】
下面来学习一下如何使用事件监听器监听路由事件,请看如下代码:
XAML页面结构是名为grid的Grid布局中有个点击按钮。我们在后置代码中使用AddHandler方法设置事件监听器,该方法的第一个参数是指定监听的路由事件类型对象,第二个参数是指定事件处理器,处理器方法由RoutedEventHandler对象包装,当按钮点击时,在输出窗口中输出了“监听到了btn_click的事件”字样。此处要注意,跟原始事件处理器不同的是,路由事件处理器的第一个参数sender,是监听事件的控件对象在此处就是grid对象,而我们要获取是谁激发的事件则是根据第二个参数e的OriginalSource属性。
当我们捕获到关心的事件时,控制事件不再继续传播该怎么做呢,事件处理器的第二个参数e有个Handled属性,该属性是个bool值,设置其为true即可。
当然,事件监听器也可以从XAML代码中指定:
通过为ButtonBase(Button的父类)的Click路由事件处理器绑定方法,来实现单击事件的监听。从ButtonBase源代码中可以找到如下图所示的路由事件处理器,该类型跟AddHandler方法的第二个参数类型一致。