原文:WPF 路由事件 Event Routing

1.路由事件介绍

之前介绍了WPF的新的依赖属性系统,本篇将介绍更高级的路由事件,替换了之前的.net普通事件。相比.net的事件,路由事件具有更强的传播能力,支持向上冒泡和向下隧道传播。路由事件允许源自某个元素的事件由另一个元素引发。

2.路由事件定义

WPF事件模型和WPF属性模型非常类似。都是只读的静态字段。

    [DefaultEvent("Click")]
[Localizability(LocalizationCategory.Button)]
public abstract class ButtonBase : ContentControl, ICommandSource
{
public static readonly RoutedEvent ClickEvent;
}

3.路由事件注册

和WPF事件模型的注册与属性几乎一样,使用EventManager.RegisterRoutedEvent()来进行注册。可以查看该函数的说明:

        //
// 摘要:
// 向 Windows Presentation Foundation (WPF) 事件系统注册新的路由事件。
//
// 参数:
// name:
// 路由事件的名称。该名称在所有者类型中必须是唯一的,并且不能为 null 或空字符串。
//
// routingStrategy:
// 作为枚举值的事件的路由策略。
//
// handlerType:
// 事件处理程序的类型。该类型必须为委托类型,并且不能为 null。
//
// ownerType:
// 路由事件的所有者类类型。该类型不能为 null。
//
// 返回结果:
// 新注册的路由事件的标识符。现在可将该标识符对象存储为类中的静态字段,然后将其用作将处理程序附加到事件的方法的参数。路由事件标识符也用于其他事件系统
// APIs。
public static RoutedEvent RegisterRoutedEvent(string name, RoutingStrategy routingStrategy, Type handlerType, Type ownerType);

与依赖属性同样,在一个静态构造函数中注册:

        static ButtonBase()
{
ButtonBase.ClickEvent=EventManager.RegisterRoutedEvent
("Click",RoutingStrategy.Bubble,typeof(RoutedEventHandler),typeof(ButtonBase));
}

4.路由事件包装

路由事件通过普通的.net事件进行包装。从而使所有.net语言都能访问它们。事件包装器可以使用AddHandler()和RemoveHandler()方法添加和删除已注册的调用程序。

AddHandler与RemoveHandler在基类UIElement中定义,每个WPF元素都继承它们。

        // 摘要:
// 在单击 System.Windows.Controls.Button 时发生。
[Category("Behavior")]
public event RoutedEventHandler Click
{
add
{
base.AddHandler(ButtonBase.ClickEvent, value);
}
remove
{
base.RemoveHandler(ButtonBase.ClickEvent, value);
}
}

5.路由事件共享

public RoutedEvent AddOwner(Type ownerType);

将路由事件关联另一个所有者类型,并启用事件及其处理的路由

WPF中所有控件的基类UIElement类型就共享了MouseUp事件。MouseUp事件是在System.Windows.Input.Mouse类定义的。UIElement只是通过了AddOwner()方法重用了MouseUp事件。

UIElement.MouseUpEvent=Mouse.MouseUpEvent.AddOwner(typeof(UIElement));

6.引发路由事件

            RoutedEventArgs args = new RoutedEventArgs(ButtonBase.ClickEvent,this);
base.RaiseEvent(args);

RaiseEvent方法负责为每个已经通过AddHandler()方法注册的调用程序引发路由事件。 

这里先贴出来RoutedEventArgs的构造函数。

        // 参数:
// routedEvent:
// System.Windows.RoutedEventArgs 类的此实例的路由事件标识符。
//
// source:
// 将在处理事件时报告的备用源。这将预先填充 System.Windows.RoutedEventArgs.Source 属性。
public RoutedEventArgs(RoutedEvent routedEvent, object source);

RoutedEventArgs有下面四个属性:

1.Source  指定了引发事件的对象 

2.OriginalSource 指出了最初是什么对象引发了事件。比Source更深一层。

3.Handled 用户来终止事件的冒泡或者隧道过程。如果一个控件将Handled设置为true,刚这个事件就不会继续传递下去。

4.RoutedEvent 获取或设置与此RoutedEventArgs 实例关联的路由事件。

通过使用RoutedEventArgs我们可以为事件提供相应的源。在WPF中,如果一个事件确实需要传递额外的信息,我们可以自定义一个对象,继承自RoutedEventArgs。例如WPF中常见的MouseEventArgs。

7.关联路由事件

将后台事件处理程序与前台元素相关联有很多种方法。最常见的就是为Xaml添加事件特性。如:

<Button x:Name="button1" Content="ok" Width="80" Height="40" Click="button1_Click"/>

或者不需要元素名,如:

<Button Content="ok" Width="80" Height="40" Click="OK_Click"/>

也可以在后台代码中进行连接事件,如:

img.MouseUp+=new MouseButtonEventHandler(img_MouseUp);

C#还支持隐式地创建委托对象:

img.MouseUp+=img_MouseUp;

上述方法实质上都是调用了事件包装器(小节4)。我们可以直接调用UIElement.AddHandler()方法来直接连接事件。如:

img.AddHandler(Image.MouseUpEvent,new MouseButtonEventHandler(img_MouseUp));

8.断开关联

断开与路由事件的关联主要有两种方法:

1.-=运算符:

img.MouseUp-=img_MouseUp;

2.使用UIElement.RemoveHandler()方法:

img.RemoveHandler(Image.MouseUpEvent,new MouseButtonEventHandler(img_MouseUp));

9.最后的例子

与之前一样,以一个小例子结尾,来帮助理解。代码不多,直接贴上来

 public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
this.AddHandler(BigDog.TestEvent, new RoutedEventHandler(this.TestHandler));
}
private void TestHandler(object sender, RoutedEventArgs e)
{
MessageBox.Show((e.OriginalSource as BigDog).Name.ToString());
e.Handled = true;
} private void button1_Click(object sender, RoutedEventArgs e)
{
BigDog dear = new BigDog { Name="Dear"};
RoutedEventArgs args = new RoutedEventArgs(BigDog.TestEvent, dear);
//引发路由事件
this.button1.RaiseEvent(args);
}
}
public class BigDog
{
public string Name { get; set; }
//定义路由事件
public static readonly RoutedEvent TestEvent = EventManager.RegisterRoutedEvent
("Test", RoutingStrategy.Bubble, typeof(RoutedEventHandler), typeof(BigDog));
}


WPF 路由事件 Event Routing的更多相关文章

  1. WPF自学入门(三)WPF路由事件之内置路由事件

    有没有想过在.NET中已经有了事件机制,为什么在WPF中不直接使用.NET事件要加入路由事件来取代事件呢?最直观的原因就是典型的WPF应用程序使用很多元素关联和组合起来,是否还记得在WPF自学入门(一 ...

  2. WPF路由事件二:路由事件的三种策略

    一.什么是路由事件 路由事件是一种可以针对元素树中的多个侦听器而不是仅仅针对引发该事件的对象调用处理程序的事件.路由事件是一个CLR事件. 路由事件与一般事件的区别在于:路由事件是一种用于元素树的事件 ...

  3. .NET: WPF 路由事件

    (一)使用WPF内置路由事件 xaml: <Window x:Class="WpfApplication1.MainWindow" xmlns="http://sc ...

  4. WPF自学入门(四)WPF路由事件之自定义路由事件

    在上一遍博文中写到了内置路由事件,其实除了内置的路由事件,我们也可以进行自定义路由事件.接下来我们一起来看一下WPF中的自定义路由事件怎么进行创建吧. 创建自定义路由事件分为3个步骤: 1.声明并注册 ...

  5. WPF路由事件三:自定义路由事件

    与依赖项属性类似,WPF也为路由事件提供了WPF事件系统这一组成.为一个类型添加一个路由事件的方式与为类型添加依赖项属性的方法类似,添加一个自定义路由事件的步骤: 一.声明路由事件变量并注册:定义只读 ...

  6. WPF路由事件

    ​    这节讲一下WPF中的路由事件(Routed Event). [什么是事件] 在了解路由事件前,我们应先来了解一下什么是事件(Event). 在Windows系统中,像鼠标单击,双击,移动这样 ...

  7. WPF 路由事件总结

    1.什么是路由事件 已下为MSDN中的定义 功能定义:路由事件是一种可以针对元素树中的多个侦听器(而不是仅针对引发该事件的对象)调用处理程序的事件. 实现定义:路由事件是一个 CLR 事件,可以由 R ...

  8. WPF路由事件学习(一)

    路由事件与一般事件的区别在于:路由事件是一种用于元素树的事件,当路由事件触发后,它可以向上或向下遍历可视树和逻辑树,他用一种简单而持久的方式在每个元素上触发,而不需要任何定制的代码(如果用传统的方式实 ...

  9. WPF 路由事件

    最近想封装一个关于手势的控件,但是由其他的控件覆盖之后发现不能触发,据说是有一些事件在定义的时候就处理过e.Handle了. 定义的时候就处理了,就是为了控件能够正常的工作,别如Button.Mous ...

随机推荐

  1. 无意中发现Markdown,最终解放了我

    文件夹 概述 换行 删除线 链接自己主动识别 表格 代码块高亮 定义列表 脚注 自己主动生成文件夹 參考资料 正文 概述 大部分情况下,Markdown的基本的语法已够我们使用,比方随性记录点东西.非 ...

  2. VMware Workstation 12 安装mac os x 10.11

    本人近期在学习iOS开发,由于初学,购买设备有点太昂贵了点.和我有意向想法的朋友能够看看在这篇文章.在虚拟机里装MAC os系统. 第一步:准备 VMware Workstation 12版本号 ma ...

  3. Android 控件EditText的setOnEditorActionListener方法的理解

    需要注意的是 setOnEditorActionListener这个方法,并不是在我们点击EditText的时候触发,也不是在我们对EditText进行编辑时触发,而是在我们编辑完之后点击软键盘上的回 ...

  4. 基于bootstrap的富文本框——wangEditor【欢迎增加开发】

    先来一张效果图: 01. 引言 老早就開始研究富文本框的东西,在写完<深入理解javascript原型与闭包>之后,就想着要去做一个富文本框的插件的样例. 如今网络上开源的富文本框插件许多 ...

  5. php实现 字符个数统计

    php实现 字符个数统计 一.总结 一句话总结: 1.php字符转ascii码函数? ord() 6 if(ord($input[$i]) < 128 and 0 < ord($input ...

  6. 使用AJAX实现页面跳转

    $.ajax({ type:"POST", url: //你的请求程序页面随便啦 async:false,//同步:意思是当有返回值以后才会进行后面的js程序. data://请求 ...

  7. gcc for Windows 开发环境介绍

    导读: 在Windows操作系统下的gcc 第一节 GCC家族概览 GCC是一个原本用于Unix-like系统下编程的编译器. 只是,如今GCC也有了很多Win32下的移植版本号.所以,或许对于很多W ...

  8. 一个封装比较完整的FTP类——clsFTP

    前几天,看见园子里面的博友写了一个支持断点续传的FTP类,一时技痒,干脆写了个更完整的clsFtp类.只是我写这个clsFtp不是支持断点续传的目的,而是为了封装FTP几个基本常用的操作接口. 功能 ...

  9. ppt转flash kindeditor上传视频全屏问题

    最近要增加页面的ppt显示功能,于是考虑把ppt转成flash,在网上搜到了ispingfree,链接: https://pan.baidu.com/s/1QZzx6qmdsnwzWCuULXzUOw ...

  10. Role-based access control modeling and auditing system

    A role-based access control (RBAC) modeling and auditing system is described that enables a user to  ...