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;// 阻断事件
        }

结果:

相关推荐
獨枭43 分钟前
C# 本地项目引用失效与恢复全攻略
开发语言·c#·visual studio
人间打气筒(Ada)1 小时前
Centos7 搭建hadoop2.7.2、hbase伪分布式集群
数据库·分布式·hbase
清风与日月1 小时前
c# 上位机作为控制端与下位机通信方式
单片机·嵌入式硬件·c#
原来是好奇心3 小时前
消息队列终极选型:RabbitMQ、RocketMQ、Kafka与ActiveMQ深度对比
分布式·kafka·rabbitmq·rocketmq·activemq·mq
烛阴3 小时前
从零开始掌握C#核心:变量与数据类型
前端·c#
yue0084 小时前
C# 生成指定位数的编号
开发语言·c#
9ilk4 小时前
【仿RabbitMQ的发布订阅式消息队列】 ---- 功能测试联调
linux·服务器·c++·分布式·学习·rabbitmq
红黑色的圣西罗4 小时前
C# List.Sort方法总结
开发语言·c#
周杰伦_Jay4 小时前
【RPC:分布式跨节点透明通信协议】【Raft:简单易实现的分布式共识算法】
分布式·rpc·共识算法
嗝屁小孩纸5 小时前
免费测评RPC分布式博客平台(仅用云服务器支持高性能)
服务器·分布式·rpc