为什么按钮点击事件"冒泡视觉树"到StackPanel,如MSDN文章所述?

Edw*_*uay 21 c# wpf event-routing

在MSDN文章" 了解WPF中的路由事件和命令"中,它说明了这一点

事件将从源元素向上冒泡(传播)可视树,直到它被处理或到达根元素.

但是,在此示例中,当您单击按钮时,它不会"冒泡可视树"以由父StackPanel事件处理,即单击按钮不会触发任何事件.

为什么不?如果不是这样的话,他们是什么意思"冒泡"

XAML:

<Window x:Class="TestClickEvents456.Window1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="Window1" Height="300" Width="300">
    <StackPanel x:Name="TheStackPanel"
                Background="Yellow"
                MouseDown="TheStackPanel_MouseDown">
        <Button x:Name="TheButton"
                Margin="10"
                Content="Click This"/>
        <TextBlock x:Name="TheMessage"
                   Text="Click the button or the yellow area"/>
    </StackPanel>
</Window>
Run Code Online (Sandbox Code Playgroud)

后台代码:

using System.Windows;
using System.Windows.Input;

namespace TestClickEvents456
{
    public partial class Window1 : Window
    {
        public Window1()
        {
            InitializeComponent();
        }

        private void TheStackPanel_MouseDown(object sender, MouseButtonEventArgs e)
        {
            TheMessage.Text = "StackPanel was clicked.";
        }

    }
}
Run Code Online (Sandbox Code Playgroud)

Arc*_*rus 25

事件冒出来,直到它被处理完毕......

由于Button通过鼠标点击执行某些操作,因此它会吸收鼠标事件并将其转换为ClickEvent.

如果您使用PreviewMouseDown,您会看到StackPanel首先在按钮之前接收到该事件.预览事件使用隧道向下方法.

  • 是的,但是按钮的默认实现是吸收鼠标事件以将其转换为点击事件......即使您没有对点击进行任何操作...... (2认同)
  • 找到有关事件顺序的相关信息:http://msdn.microsoft.com/en-us/library/ms742806.aspx#how_event_processing_works (2认同)
  • 网格似乎也吸收了鼠标事件. (2认同)

Rob*_*nee 8

正如其他人所说,这是因为MouseDown事件在Button被进一步冒泡之前就被处理了.您可以在Reflector中看到ButtonBase.OnMouseLeftButtonDown:

protected override void OnMouseLeftButtonDown(MouseButtonEventArgs e)
{
    if (this.ClickMode != ClickMode.Hover)
    {
        e.Handled = true;
        // SNIP...
    }
    base.OnMouseLeftButtonDown(e);
}
Run Code Online (Sandbox Code Playgroud)

一种解决方案是监听MouseDown事件,并指出您不关心事件是否得到处理.您可以使用该AddHandler方法执行此操作.它有一个布尔重载,可以让你监听已经处理过的事件.

如果您在某处执行此操作而不是在XAML中设置MouseDown处理程序:

TheStackPanel.AddHandler(MouseDownEvent, new MouseButtonEventHandler(TheStackPanel_MouseDown), true);
Run Code Online (Sandbox Code Playgroud)

无论是否已经处理,您都将收到所有MouseDown活动TheStackPanel.


Raz*_*zie 7

此外,如果您希望stackpanel接收事件,请将stackpanel xaml更改为:

<StackPanel x:Name="TheStackPanel" 
            Background="Yellow"
            Button.Click="TheStackPanel_MouseDown" />
Run Code Online (Sandbox Code Playgroud)

和事件签名:

private void TheStackPanel_MouseDown(object sender, RoutedEventArgs e)
Run Code Online (Sandbox Code Playgroud)

在这种情况下,stackpanel将接收按钮的单击事件.但是,单击stackpanel本身不会触发任何事件,因为它专门监听按钮单击.