原文: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. 【hdu5527】【2015ACM/ICPC亚洲区长春站 】Too Rich

    题目链接: pid=5527">http://acm.hdu.edu.cn/showproblem.php?pid=5527 题意&题解: 感觉自己真是弱啊,自己想的贪心是错的 ...

  2. php替换空格(php函数的设计思路)

    php替换空格(php函数的设计思路) 一.总结 1.替换和也是先查找了再替换,截取的话就是先查找到再截取 2.设计函数的时候按照的是缺省参数在后,核心东西在前的思路来设计函数的:查找的话是$sear ...

  3. Win10安装后必做的优化,解决磁盘100%占用

    Win10安装后必做的优化,解决磁盘100%占用 01关闭家庭组 控制面板–管理工具–服务– HomeGroup Listener和HomeGroup Provider禁用. 02关闭磁盘碎片整理.自 ...

  4. [AngularFire 2] Protect Write Access Using Security Rules

    We cannot allow un-auth user to change the database data as they want, for Firebase, it is easy just ...

  5. html5 背景音乐 js控制播放 暂停

    <html> <head> <title> 测试页面 </title> <script src="jquery.min.js" ...

  6. svn: E200033: database is locked解决办法

    svn执行update,却被告知database is locked! 执行 svn update,却抛出个错误警报: svn: E200033: database is locked, execut ...

  7. CF 559B(Equivalent Strings-构造法)

    B. Equivalent Strings time limit per test 2 seconds memory limit per test 256 megabytes input standa ...

  8. thinkphp中如何实现无限级分类?

    thinkphp中如何实现无限级分类? 一.总结 1.数据表设计+递归算法 二.php实现无限级分类实例总结 1.数据库数据如下: 2.任务需求:给一个id,求自己和所有父亲. 3.实现代码如下:th ...

  9. Android自定义控件View(三)组合控件

    不少人应该见过小米手机系统音量控制UI,一个圆形带动画效果的音量加减UI,效果很好看.它是怎么实现的呢?这篇博客来揭开它的神秘面纱.先上效果图 相信很多人都知道Android自定义控件的三种方式,An ...

  10. oracle的number的浅析

    author:skate time:2011-02-14 oracle的number的浅析 从例如以下几个方面来认识number 1.表示的数值范围   2.占用的存储空间   3.number的性能 ...