5

我发现了很多解释冒泡的例子,但没有关于隧道的,这是关于隧道的,例如父母对孩子。我认为我的主要问题是我不明白如何在孩子中注册路由事件(WindowControl 到 UserControl)。我有:

public partial class MyParent : UserControl
{
  public static readonly RoutedEvent RoutedMouseUpEvent = EventManager.RegisterRoutedEvent(
        "PreviewMouseLeftButtonUp", RoutingStrategy.Tunnel, typeof(RoutedEventHandler),   typeof(WindowControl)); 

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

public addView(UserControl view)
{
WindowControl win = new WindowControl();
win.Content = view;
}

private void Grid_MouseLeftButtonUp(object sender, MouseButtonEventArgs e)
{
  RoutedEventArgs newEventArgs = new RoutedEventArgs(MyParent.RoutedMouseUpEvent);
            RaiseEvent(newEventArgs);
}
}

addView的封装是必须的,应该没问题吧?孩子是通过 addView 添加的。调用 Grid_MouseLeftButtonUp。
接收器看起来像这样(它是 mvvm 所以没有太多):

public partial class ChildView : UserControl
{
 void UserControl_PreviewMouseLeftButtonUp(object sender, RoutedEventArgs args)
 {
    int i = 0; // The breakpoint is never called
 }
}

在xml中

<Grid>
   <Border BorderBrush="black" BorderThickness="1" HorizontalAlignment="Center"   VerticalAlignment="Center" PreviewMouseLeftButtonUp="UserControl_PreviewMouseLeftButtonUp">
</Border>
</Grid>

如果我忘记了什么,请告诉我。问题是,路由事件没有到达 UserControl_PreviewMouseLeftButtonUp

4

1 回答 1

11

这不是隧道路由策略的工作方式。隧道意味着事件将从根开始并沿着树路径向下到达调用控件。例如,如果我们有以下可视化树

Window
|
|--> SomeUserControl
|--> MyParent
     |
     |--> ChildView

那么如果MyParent会引发隧道事件,隧道事件将访问:

  1. 窗户
  2. 我的父母

不是

  1. 我的父母
  2. 子视图

总而言之,冒泡事件将始终从引发事件的控件开始并在可视化树的根部停止,而隧道事件将从可视化树的根部开始并在引发事件的控件结束(完全相同的路径,只能倒序)。

编辑:您可以在MSDN 的 Routed Events Overview中阅读有关路由事件的更多信息。它还有一个很好的图像来证明这一点:

在此处输入图像描述

于 2012-11-20T14:37:44.427 回答