[Silverlight]奇技银巧系列-5 捕获路由事件

Silverlight奇技银巧系列:

所有该系列中的文章只为介绍一些Silverlight中或隐藏在Silverlight .NET framework鲜为人知的知识和技巧。这些文章并非教程,也是不是什么技术指导,您只需对文章介绍的内容保有印象。这些知识和技巧也许会在不经意间绽放银色光芒。

Silverlight中事件的处理方式有时会令人费解。其实Silverlight中有两种事件,它们分别是“直接事件”和“路由事件”。

所谓“直接事件”其实就是我们在winform,vb6,mfc等等中使用的事件模型,相信大家已经很熟悉了,MediaElement的MediaEnded事件就是这种类型的事件。

而“路由事件”是从WPF发展到Silverlight中的。路由事件沿着可视化树向上冒泡,它总是从嵌套中的最深层元素开始冒泡至嵌套最浅的元素。例如MouseLeftButtonUp事件先从鼠标直接点击的元素上触发然后才到其父元素(顺着可视化树继续上行)。

从路由事件发生后,任意阻止冒泡的操作都可被路由事件沿途的控件捕获并通知路由事件。在一些应用场景中,某些标准控件需要捕获所需的路由事件,比如一个Button捕获MouseLeftButtonUp路由事件。不过我们要如何才能得知事件是否已经被捕获呢?

答案是UIElement中的AddHandler方法

比如我们有如下XAML:

<StackPanel x:Name="LayoutRoot"
        Background="White">
    <Button Height="25"
      Width="100"
      Content="点我啊!"
      x:Name="clickButton"/>
    <Button Height="25"
      Width="100"
      Content="AddHandler"
      x:Name="addButton" />
</StackPanel>

当我们点击clickButton按钮时LayoutRoot对应的MouseLeftButtonUp事件并不会触发,而当我们点击clickButton按钮外的其他区域时,LayoutRoot的MouseLeftButtonUp又被触发了。

如果我们要让clickButton捕获LayoutRoot上的MouseLeftButtonUp事件,那就需要使用UIElement.AddHandler()方法。

该方法的签名如下:

public void AddHandler(RoutedEvent routedEvent,

 Delegate handler,

bool handledEventsToo);

  • 一个确切的路由事件 routedEvent
  • 一个具有正确对应类型的委托 handler
  • 以及一个用来确定捕获指定事件时是否发出通知的的布尔值handledEventsToo

于是在我们为addButton的Click事件增加如下代码

LayoutRoot.AddHandler(
  UIElement.MouseLeftButtonUpEvent,
  new MouseButtonEventHandler((o, args) => {
      MessageBox.Show("我是路由事件,我被触发了,oh yeah!");
  }), true);

如此,我们在点击clickButton的时候页一样可以捕获到MouseLeftButtonUp事件了。

全部的后台代码如下:

        public MainPage() {
            InitializeComponent();
            clickButton.Click += new RoutedEventHandler(clickButton_Click);
            addButton.Click += new RoutedEventHandler(addButton_Click);
        }

        void addButton_Click(object sender, RoutedEventArgs e) {

            LayoutRoot.AddHandler(
              UIElement.MouseLeftButtonUpEvent,
              new MouseButtonEventHandler((o, args) => {
                  MessageBox.Show("我是路由事件,我被触发了,oh yeah!");
              }), true);
        }

        void clickButton_Click(object sender, RoutedEventArgs e) {
            MessageBox.Show("我是按钮,我被点了。");
        }

image

posted @ 2010-01-29 14:24  紫色永恒  阅读(1282)  评论(8编辑  收藏  举报