`

WPF快速指导10:WPF中的事件及冒泡事件和隧道事件(预览事件)的区别(转载)

 
阅读更多
WPF快速指导10:WPF中的事件及冒泡事件和隧道事件(预览事件)的区别(转载)
本文摘要:

1:什么是路由事件;

2:中断事件路由;

3:自定义路由事件;

4:为什么需要自定义路由事件;

5:什么是冒泡事件和预览事件(隧道事件);

1:什么是路由事件

    WPF中的事件为路由事件,所谓路由事件,MSDN定义如下:

    功能定义:路由事件是一种可以针对元素树中的多个侦听器(而不是仅针对引发该事件的对象)调用处理程序的事件。

    实现定义:路由事件是一个 CLR 事件,可以由 RoutedEvent 类的实例提供支持并由 Windows Presentation Foundation (WPF) 事件系统来处理。

    但这两类定义都比较抽象,我们来看更具体的定义:

       <Border Height="50" Width="250" BorderBrush="Gray" BorderThickness="1" >
            <StackPanel Background="LightGray" Orientation="Horizontal" MouseUp="StackPanel_MouseUp">
                <TextBlock Name="YesTB" Width="50"  MouseUp="YesTB_MouseUp" Background="Blue" >Yes</TextBlock>
            </StackPanel>
        </Border>
    在这个例子中,事件的事件路由为:

TextBlock -->StackPanel-->Border —>...


2:中断事件路由
     所有的路由事件都共享一个公共的事件数据基类 RoutedEventArgs。 RoutedEventArgs 定义了一个采用布尔值的 Handled 属性。 Handled 属性的目的在于,允许路由中的任何事件处理程序通过将 Handled 的值设置为 true 来将路由事件标记为“已处理”。

        private void StackPanel_MouseUp(object sender, MouseButtonEventArgs e)
        {
            MessageBox.Show("Panel");
        }

        private void YesTB_MouseUp(object sender, MouseButtonEventArgs e)
        {
            MessageBox.Show("button");
            e.Handled = true;
        }
    在上面的例子中,将不再触发StackPanel_MouseUp事件。


3:自定义路由事件

      如下面的示例所示,首先使用 RegisterRoutedEvent 方法注册一个 RoutedEvent。按照约定,RoutedEvent 静态字段名称应当以后缀 Event 结束。在本示例中,事件的名称是 Tap,事件的路由策略是 Bubble。在注册调用之后,可以为该事件提供添加和移除公共语言运行时 (CLR) 事件访问器。

      请注意,尽管该事件在本特定示例中是通过 OnTap 虚方法引发的,但您引发事件的方式或者事件响应更改的方式取决于您的需要。

      还要注意,本示例主要实现 Button 的一整个子类;该子类是作为单独的程序集构建的,之后将在单独的可扩展应用程序标记语言 (XAML) 页上实例化为一个自定义类。这是为了说明这样一个概念:创建子类的控件可以插入到由其他控件组成的树中,在这种情况下,这些控件上的自定义事件具有与任何固有的 Windows Presentation Foundation (WPF) 元素完全相同的事件路由功能。

public class MyButtonSimple: Button
{
    // Create a custom routed event by first registering a RoutedEventID
    // This event uses the bubbling routing strategy
    public static readonly RoutedEvent TapEvent = EventManager.RegisterRoutedEvent(
        "Tap", RoutingStrategy.Bubble, typeof(RoutedEventHandler), typeof(MyButtonSimple));

    // Provide CLR accessors for the event
    public event RoutedEventHandler Tap
    {
            add { AddHandler(TapEvent, value); }
            remove { RemoveHandler(TapEvent, value); }
    }

    // This method raises the Tap event
    void RaiseTapEvent()
    {
            RoutedEventArgs newEventArgs = new RoutedEventArgs(MyButtonSimple.TapEvent);
            RaiseEvent(newEventArgs);
    }
    // For demonstration purposes we raise the event when the MyButtonSimple is clicked
    protected override void OnClick()
    {
        RaiseTapEvent();
    }

}
<Window 
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:custom="clr-namespace:SDKSample;assembly=SDKSampleLibrary"
    x:Class="SDKSample.RoutedEventCustomApp"

    >
    <Window.Resources>
      <Style TargetType="{x:Type custom:MyButtonSimple}">
        <Setter Property="Height" Value="20"/>
        <Setter Property="Width" Value="250"/>
        <Setter Property="HorizontalAlignment" Value="Left"/>
        <Setter Property="Background" Value="#808080"/>
      </Style>
    </Window.Resources>
    <StackPanel Background="LightGray">
        <custom:MyButtonSimple Name="mybtnsimple" Tap="TapHandler">Click to see Tap custom event work</custom:MyButtonSimple>
    </StackPanel>
</Window>

4:为什么需要自定义路由事件

      一直到目前看来,我们都不太需要自定义的路由事件。但是,在我们创建自定义控制的时候,创建一些和业务相关的路由事件,就显得很有必要。

      如,创建一个在线考试中的题型展示控件,可以为该控件设计一个自定义事件,为“提交”。这样一来,这个题型控件不仅仅只有一些通用事件,还可以看上去更“业务”。


5:什么是冒泡事件和预览事件(隧道事件)

路由事件实际上分两类:冒泡事件和预览事件(隧道事件)。上文中的例子就是冒泡事件。

     冒泡事件是WPF路由事件中最为常见,它表示事件从源元素扩散(传播)到可视树,直到它被处理或到达根元素。这样您就可以针对源元素的上方层级对象处理事件。例如,您可向嵌入的 Grid 元素附加一个 Button.Click 处理程序,而不是直接将其附加到按钮本身。气泡事件有指示其操作的名称(例如,MouseDown)。

     隧道事件采用另一种方式,从根元素开始,向下遍历元素树,直到被处理或到达事件的源元素。这样上游元素就可以在事件到达源元素之前先行截取并进行处理。根据命名惯例,隧道事件带有前缀 Preview(例如 PreviewMouseDown)。

     在本文一开始的例子中,如果我们将MouseUP,改为PreviewMouseUP,效果会如何呢。

区别:

     冒泡事件:在YesTB上点击,首先弹出“button”,再弹出“panel”。

     预览事件(隧道事件)事件:在YesTB上点击,首先弹出“panel”,再弹出“button”。

     看到了这点区别,那么我们加入e.Handled=true的时机也要不同。首先,

     冒泡事件例子中:e.Handled=true加在YesTB_PreviewMouseUp中,加入后,点击YesTB,将只弹出“button”。

     预览事件(隧道事件)例子中:e.Handled=true家在StackPanel_PreviewMouseUp中,加入后,点击YesTB,将只弹出“panel”。


本文基于Creative Commons Attribution 2.5 China Mainland License发布,欢迎转载,演绎或用于商业目的,但是必须保留本文的署名http://www.cnblogs.com/luminji(包含链接)。如您有任何疑问或者授权方面的协商,请给我留言。
分享到:
评论

相关推荐

    WPF纯MVVM事件绑定+事件参数 完美示例

    在WPF中,我们通常会遇到需要处理UI控件的事件,而在MVVM模式下,我们需要找到不直接操作View的方法来实现这一目标。 标题中的“WPF纯MVVM事件绑定+事件参数 完美示例”意味着我们将学习如何在不违反MVVM原则的情况...

    WPF Image 点击事件

    在WPF中,事件处理主要基于事件路由机制,分为三种类型: bubbling(冒泡)、tunneling(隧道)和direct(直接)。通常,我们可以为控件添加事件处理程序来响应用户操作,如鼠标点击。 ### 二、定义点击事件 由于...

    5个关于wpf事件路由的例子

    **WPF(Windows Presentation Foundation)事件路由是UI编程中的一个重要概念,它决定了事件在控件树中的传播方式。本文将通过五个具体的例子深入探讨WPF事件路由机制,帮助你更好地理解和应用这一技术。** ### 1. ...

    WPF资源字典中的控件事件触发

    3. **行为(Behaviors)**:WPF的 Blend SDK 提供了一种称为行为(Behaviors)的机制,它允许你为任何对象添加事件处理,即使该对象不在当前XAML文件中。你可以在资源字典中定义一个行为,然后在需要的地方引用它。 ...

    WPF 常见文件预览DEMO

    **WPF 常见文件预览DEMO** 在Windows Presentation Foundation(WPF)中,文件预览功能是一项常见的需求,特别是在...通过学习和理解这个DEMO,开发者可以更好地掌握WPF中的文件预览技术,并将其应用到实际项目中。

    WPF打印及预览

    **WPF打印及预览详解** 在Windows Presentation Foundation(WPF)中,打印和预览功能是通过丰富的API集来实现的,使得开发者能够轻松地处理应用程序中的打印需求。本篇将详细介绍如何在WPF中打印Canvas控件内的...

    WPF实现MVVM的事件绑定

    WPF实现MVVM的事件绑定的两种非常规方式。 参考博客:https://blog.csdn.net/IQQQQU/article/details/86530285 1、重写InvokeCommandAction来扩充返回的参数 2、运用Behavior来实现事件,可以通过两种方式来获取...

    wpf中鼠标点击时触发事件效果

    总结来说,WPF中的Button控件提供了丰富的事件和样式机制,允许开发者创建具有高度交互性和定制化视觉效果的用户界面。通过理解和运用`MouseLeftButtonDown`、`MouseLeftButtonUp`和`Click`事件,以及自定义`...

    WPF编程宝典:使用C# 2012和.NET 4.5 第4版 PDF与源码

    2. **控件和布局**:详细讲解WPF中的各种内置控件,如Button、TextBox等,以及如何通过布局容器(如Grid、StackPanel、DockPanel)进行控件定位和布局。 3. **数据绑定**:阐述WPF强大的数据绑定机制,包括单向、...

    WPF_MouseWheel事件实例

    在WPF中,可以有两种方式处理`MouseWheel`事件:直接在控件上处理,或者在更高层次的容器上处理以实现全局效果。例如,你可以在`TextBox`、`ListView`等控件上直接添加事件处理程序,或者在`Window`或`Grid`等容器上...

    wpf路由事件简单示例

    在Windows Presentation Foundation (WPF) 中,路由事件是一种特殊类型的事件,它允许事件不仅在触发事件的元素上处理,还能沿着元素树传播。这种机制使得事件处理更加灵活和高效。标题"wpf路由事件简单示例"指向了...

    wpf快速入门教程

    4. **依赖属性**: 这是WPF中实现数据绑定和属性系统的关键技术。依赖属性允许属性之间的相互影响,并提供了事件触发和动画支持。 5. **样式和模板**: 通过样式和模板,开发者可以为控件设置统一的外观,同时模板还...

    WPF中的事件与命令

    以下是关于WPF中事件与命令的详细解释: 一、键盘类和键盘事件 WPF 提供的 `System.Input.Keyboard` 类是处理键盘输入的主要接口。它提供了如下的键盘事件: 1. **KeyDown** 事件和 **PreviewKeyDown** 事件:这...

    C# WPF 路由事件介绍

    - **事件拦截**:父控件可以通过隧道事件在子控件处理事件前进行预处理,或者在冒泡事件中处理子控件未处理的事件。 ### 6. 示例 ```xml &lt;!-- XAML 示例 --&gt; Click Me! ``` ```csharp // 代码-behind 示例 ...

    C#+wpf窗体事件

    在2010版本的C#和WPF中,这些基本概念仍然适用,并且是开发高效、用户体验良好的应用程序的基础。 通过实践项目,例如"Ex02"这样的练习,你可以更深入地了解这些事件的用法,提升你的C# WPF编程技能。这些基础知识...

    wpf冒泡提示自定义控件库

    2. **自定义控件** - 在WPF中,开发者可以创建自己的控件来满足特定需求,这些控件可以扩展内置控件的功能,或者提供全新的界面元素。 3. **冒泡提示** - 这是UI交互设计中的一种技术,通过从特定元素出发向父级元素...

    WPF 路由事件(包含路由事件使用和自定义路由事件)

    WPF中的事件分为两种类型:**常规事件(Tunneling Events)**和**冒泡事件(Bubbling Events)**,它们都是路由事件的子类。 - **隧道事件(Preview Events)**:也称为预览事件,它们首先在元素树的根部触发,然后...

    WPF中的DateTimePicker控件

    总结,WPF中的DateTimePicker控件是一个强大的工具,它提供了日期和时间选择的便捷方式。通过理解和利用其属性、事件和样式模板,你可以创建符合业务需求的用户界面。在DateTimePickerTestSolution这个项目中,你...

    WPF Carousel 样式Demo 源码

    - **故事板(Storyboard)和动画(Animation)**: WPF中的动画系统是实现平滑滚动的关键,通过改变项目的位置和透明度,模拟出滑动效果。 4. **数据绑定和模板** - Carousel通常会与数据源绑定,每个项目代表数据...

    WPF非常精美界面 WPF漂亮首页界面 WPF精美首页可以直接使用在项目当中 WPF入门布局的完美Demo 经典WPF快速入门漂

    "经典WPF快速入门漂亮布局教程"表明这个压缩包可能包含了一系列的教程或指导,旨在帮助初学者迅速掌握WPF的基本概念和布局技巧。这些教程可能覆盖了事件处理、数据绑定、样式和模板、资源管理等多个主题,帮助开发者...

Global site tag (gtag.js) - Google Analytics