聊聊WPF继续响应被标记为已处理事件的方法

开发 前端
WPF中在冒泡事件或者隧道事件会随其层间关系在visual tree上层层传递,但是,某些事件传递到某些控件是即会”终止“(不再响应相应的注册事件),给人一种事件终结者的印象。

[[384081]]

本文转载自微信公众号「CSharp编程大全」,作者zls365 。转载本文请联系CSharp编程大全公众号。 

WPF中在冒泡事件或者隧道事件会随其层间关系在visual tree上层层传递,但是,某些事件传递到某些控件是即会”终止“(不再响应相应的注册事件),给人一种事件终结者的印象。例如:textbox对mousdown事件。

产生原因:事件处理到达该控件后,其事件对象属性Handled被标记为True。WPF事件引擎在处理控件对应事件时,若检测到该属性为True,就不会调用相应的处理程序。即 WPF路由事件被标记为handled以后, 并不是不在visual tree上传递了;而是,事件引擎不再去调用这个事件的handler了。

若仍想再其上层元素中(上层是相对事件的传递方向而言)仍然处理响应的事件,解决方式:

1、若上层控件可以注册相应事件。即没有重写对应控件的Template属性。直接上代码:

<Grid MouseDown="Grid_MouseDown" >     
     <TextBox Name="txt1" Text="{Binding Path=txt1 ,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}" MouseDown="Txt_MouseDown" />                    
</Grid> 
  • 1.
  • 2.
  • 3.
private void txt_MouseDown(object sender, MouseEventArgs e) 
  { 
     MessageBox.Show("TextMouseDown事件"); 
     e.Handled = false;//使冒泡继续上传 
  } 
 
private void Grid_MouseDown(object sender, MouseEventArgs e) 
  { 
     MessageBox.Show("GridMouseDown事件"); 
  } 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.

2、当自定义控件模板时,绑定模版事件不起效,此时用上面的方法不再生效。例如:自定义列表控件模版样式

<UserControl x:Class="Test"  
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"  
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"  
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"   
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008"   
             mc:Ignorable="d"   
             d:DesignHeight="290" d:DesignWidth="180">  
    <Grid  name="gridMain">  
               <ItemsControl Focusable="False" Background="Transparent" ItemsSource="{Binding InfoModel, Mode=OneWay}">  
                        <ItemsControl.Template>  
                            <ControlTemplate TargetType="ItemsControl">  
                                <Border x:Name="scrBorder">  
                                    <ScrollViewer x:Name="scrollViewer" VerticalScrollBarVisibility="Auto" HorizontalScrollBarVisibility="Auto"  
                                                 Padding="{TemplateBinding Padding}" MouseLeftButtonDown="MouseLeftButtonDown">  
                                        <ItemsPresenter />  
                                    </ScrollViewer>  
                                </Border>  
                            </ControlTemplate>  
                        </ItemsControl.Template>  
                        
                        <ItemsControl.ItemTemplate>  
                            <DataTemplate>  
                                <Grid>  
                                    <Grid.ColumnDefinitions>  
                                        <ColumnDefinition Width="*"/>  
                                        <ColumnDefinition Width="Auto"/>  
                                        <ColumnDefinition Width="Auto"/>  
                                    </Grid.ColumnDefinitions>  
                                    <TextBlock Text="{Binding Desc}" Grid.Column="0"/>  
                                    <TextBlock Text="{Binding Value}" Margin="0" Grid.Column="1"/>  
                                    <TextBlock Text="{Binding Unit}" Grid.Column="2" Margin="3,0,15,0"/>  
                                </Grid>  
                            </DataTemplate>  
                        </ItemsControl.ItemTemplate>  
                        <ItemsControl.ItemsPanel>  
                            <ItemsPanelTemplate>  
                                <StackPanel></StackPanel>  
                            </ItemsPanelTemplate>  
                        </ItemsControl.ItemsPanel>  
                        <ItemsControl.ItemContainerStyle>  
                            <Style>  
                                <Setter Property="Control.Margin" Value="1"/>  
</Style>  
                        </ItemsControl.ItemContainerStyle>  
                    </ItemsControl>  
                </Grid>  
</UserControl>  
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.

ScrollViewer在控件模版中,ScrollViewer的MouseButtonDown事件处理事件如下:断点设置会发现鼠标点击时并不会触发。

private void MouseLeftButtonDown(object sender, MouseButtonEventArgs e) 
   { 
      //e.Handled = false ; 
   } 
  • 1.
  • 2.
  • 3.
  • 4.

解决办法:UIElement.AddHandler 方法:为指定的路由事件添加路由事件处理程序,并将该处理程序添加到当前元素的处理程序集合中。具体解释详见:https://docs.microsoft.com/zh-cn/dotnet/api/system.windows.uielement.addhandler?f1url=https%3A%2F%2Fmsdn.microsoft.com%2Fquery%2Fdev15.query%3FappId%3DDev15IDEF1%26l%3DZH-CN%26k%3Dk(System.Windows.UIElement.AddHandler);k(TargetFrameworkMoniker-.NETFramework,Version%3Dv4.5);k(DevLang-csharp)%26rd%3Dtrue&view=netframework-4.8

大体意思:由于WPF事件visual tree上传递过程中,某个元素将该事件标记为已处理,导致事件在传递时不再继续有响应,(原因:Handled被标记为True)如果希望后续元素也能响应此方法,可以使用此方法。

因此我们可以在上面UserControl的构造函数中添加下面代码:表示gridMain处理相应鼠标点击事件

public UserControl() 
        { 
             InitializeComponent(); 
 
            gridMain.AddHandler(MouseLeftButtonDownEvent, new MouseButtonEventHandler(MouseLeftButtonDown), true); 
       } 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.

再次断点调试MouseLeftButtonDown,会发现断点命中。

AddHandler这个代码的关键点是最后那个true,它告诉WPF引擎相应元素call这个handle,即使它被标记为Handled=true。但是元素处理后其上层元素也照样不会相应,因为handle仍被标记已处理。由此可见,WPF路由事件被标记为handled以后,并不是不在visual tree上传递了;而是,不去call这个handler了。

上例中如果想要UserControl继续响应,处境就与1一样了,只需将handle标记为false即可。

责任编辑:武晓燕 来源: CSharp编程大全
相关推荐

2021-09-07 09:37:04

AI 数据Facebook

2011-08-23 16:36:05

SQL Server 数据库被标记为可疑

2022-02-09 00:08:26

黑客攻击事件响应

2022-06-06 11:29:16

软件包安装开发包

2010-08-03 15:54:43

FlexBuilder

2020-11-13 07:31:10

自动化事件安全

2009-12-25 10:28:10

WPF路由事件

2023-10-07 11:04:58

WPF数据UI

2021-02-11 08:27:28

数据

2009-12-24 16:11:07

WPF图像处理

2021-08-06 11:46:46

Go三色标记法

2012-02-16 09:26:26

微软Google

2023-08-09 11:59:29

微软Windows

2009-11-13 09:15:34

WPF

2021-10-20 09:24:47

Windows性能微软

2009-12-30 10:32:31

Silverlight

2020-10-12 12:45:23

苹果漏洞黑客

2009-07-10 11:45:44

Java Swing

2012-02-16 10:34:27

2025-02-04 08:27:13

点赞
收藏

51CTO技术栈公众号