WPF—路由事件详解

WPF---路由事件

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

路由事件与一般事件的区别在于:路由事件是一种用于元素树的事件,当路由事件触发后,它可以向上或向下遍历可视树和逻辑树,他用一种简单而持久的方式在每个元素上触发,而不需要任何定制的代码(如果用传统的方式实现一个操作,执行整个事件的调用则需要执行代码将事件串联起来)。

路由事件的路由策略:

所谓的路由策略就是指:路由事件实现遍历元素的方式。

路由事件一般使用以下三种路由策略:

(1) 冒泡:由事件源向上传递一直到根元素。

(2) 直接::直接路由事件与普通的.NET事件是非常相似的,他们都起源于一个元素,并且不能够传递给其它的元素。只有事件源才有机会响应事件。

(3) 隧道:从元素树的根部调用事件处理程序并依次向下深入直到事件源。一般情况下,WPF提供的输入事件都是以隧道/冒泡对实现的。隧道事件常常被称为Preview事件。

管道

从元素树的根部调用事件处理程序并依次向下深入直到事件源。

管道事件按照惯例,他们的名字中都有一个preview前缀,一般来说管道事件都有他的配对的冒泡事件,例如:PreviewMouseDown和MouseDown就是配对事件,如果同时存在的话,那么就会先执行管道事件然后才执行配对的冒泡事件。当然e.Handled=true,依然能够阻断事件。

当TextBlock的事件触发时,会从根元素进行传递直到TextBlock自身

cs 复制代码
    // XAML标签布局
        <Grid>
            <StackPanel Width="200"
                        Background="Gray"
                        Height="40"
                        PreviewMouseUp="StackPanel_MouseUp">
                <TextBlock Height="40" Background="blue" Width="40" Margin="-150,0,0,0"
                           PreviewMouseUp="TextBlock_MouseUp">
                </TextBlock>
            </StackPanel>
        </Grid>
    
    // 在CS后台文件写对应的数据
        private void TextBlock_MouseUp(object sender, MouseButtonEventArgs e)
        {
            Console.WriteLine("子控件textblock 鼠标松开");
        }
    
        private void StackPanel_MouseUp(object sender, MouseButtonEventArgs e)
        {
            Console.WriteLine("父控件StackPane 鼠标松开");
        }

结果:

冒泡

由事件源向上传递一直到根元素。

当触发TextBlock的事件时,会从自身一直向上传递直到根元素。

cs 复制代码
    // XAML标签布局
    <Grid>
        <StackPanel Width="200"
                    Background="Gray"
                    Height="40"
                    MouseUp="StackPanel_MouseUp">
            <TextBlock Height="40"
                       Background="blue"
                       Width="40"
                       Margin="-150,0,0,0"
                       MouseUp="TextBlock_MouseUp">
            </TextBlock>
        </StackPanel>
    </Grid>
    
    // 在CS后台文件写对应的数据
    private void TextBlock_MouseUp(object sender, MouseButtonEventArgs e)
    {
        Console.WriteLine("子控件textblock 鼠标松开");
    }
    
    private void StackPanel_MouseUp(object sender, MouseButtonEventArgs e)
    {
        Console.WriteLine("父控件StackPane 鼠标松开");
    }

结果:

直接

直接路由事件只传递一层。它源自一个元素,并且不传递给其他元素,MouseEnter事件(当鼠标指针移动到元素上时发生)是直接路由事件。

当鼠标移到TextBlock的时候会响应BlockMouseEnter事件,但是不会响应StackMouseEnter事件

cs 复制代码
    <StackPanel Orientation="Vertical" MouseEnter="StackMouseEnter" Background="Yellow">
      <TextBlock Text="Block鼠标移进来" Background="AliceBlue" Height="50" Width="100" MouseEnter="BlockMouseEnter" />
    </StackPanel>

如何阻断事件

e.Handled = true; 阻断事件

当Handled属性为true时阻断该事件的传递,后续事件不会再进行执行

cs 复制代码
    // XAML标签布局
        <Grid>
            <StackPanel Width="200"
                        Background="Gray"
                        Height="40"
                        PreviewMouseUp="StackPanel_MouseUp">
                <TextBlock Height="40" Background="blue" Width="40" Margin="-150,0,0,0"
                           PreviewMouseUp="TextBlock_MouseUp">
                </TextBlock>
            </StackPanel>
        </Grid>
    
    // 在CS后台文件写对应的数据
        private void TextBlock_MouseUp(object sender, MouseButtonEventArgs e)
        {
            Console.WriteLine("子控件textblock 鼠标松开");
        }
    
        private void StackPanel_MouseUp(object sender, MouseButtonEventArgs e)
        {
            Console.WriteLine("父控件StackPane 鼠标松开");
            e.Handled = true;// 阻断事件
        }

结果:

相关推荐
wearegogog1235 小时前
基于C# WinForm实现的带条码打印的固定资产管理
开发语言·c#
Light606 小时前
数据要素与数据知识产权交易中心建设专项方案——以领码 SPARK 融合平台为技术底座,构建可评估、可验证、可交易、可监管的数据要素工程体系
大数据·分布式·spark
SmoothSailingT7 小时前
C#——LINQ方法
开发语言·c#·linq
superman超哥7 小时前
仓颉条件变量深度解析与实践:解锁高效并发同步
开发语言·python·c#·仓颉
世洋Blog8 小时前
装饰器模式实践:告别臃肿的继承链,优雅解耦初始化状态管理
unity·设计模式·c#·装饰器模式
10 小时前
TIDB——TIKV——raft
数据库·分布式·tidb
我是唐青枫10 小时前
C#.NET ref struct 深度解析:语义、限制与最佳实践
c#·.net
火一线10 小时前
【C#知识点详解】基类、抽象类、接口类型变量与子类实例的归纳总结
开发语言·c#
Lv117700811 小时前
Visual Studio 中的密封类和静态类
ide·笔记·c#·visual studio
永亮同学11 小时前
【探索实战】从零开始搭建Kurator分布式云原生平台:详细入门体验与功能实战分享!
分布式·云原生·交互