在自定义Richtextbox中,给Richtextbox订阅MouseLeftButtonUp事件用于处理自定义任务,程序运行后发现自定义MouseLeftButtonUp未触发。
概念
WPF中的隧道事件和冒泡事件
- 冒泡事件(Bubbling):
- 这类事件从UI元素触发,并沿着元素树向上传播至根元素。
- 在冒泡阶段,事件会从子元素传递给父元素,直到被处理或者到达元素树的顶部。
例如,MouseClick
或KeyDown
等事件属于冒泡事件。 - 事件处理程序可以在冒泡过程中的任何级别被调用,允许父元素响应子元素上发生的事件。
- 隧道事件(Tunneling):
- 这类事件从UI元素树的根开始向下传播至触发事件的源元素。
- 在隧道阶段,事件从根元素传递给子元素,直到到达目标元素。
- 隧道事件在WPF中通常以“Preview”作为前缀,例如
PreviewMouseDown
或PreviewKeyDown
。 - 合成控件的过程中可能会用到隧道事件,以便在事件到达目标元素之前捕获和处理事件。
解决方法
RichTextBox中的MouseLeftButtonUp事件应该是在控件内部将 e.Handled = true;导致我们订阅的事件未触发。为了实现监测鼠标点击释放事件,我们有如下两种解决方法:
1. richTextBox.AddHandler(MouseLeftButtonUpEvent, new RoutedEventHandler(richText_MouseUp), true);
注:
A. 使用RoutedEventArgs而不是MouseButtonEventArgs来声明你的事件处理程序,但你可以在内部将其强制转换回MouseButtonEventArgs。
B. AddHandler 中的最后一个参数 —— 将其设置为 true 会导致事件即使在之前的处理程序将 e.Handled 设置为 true 时也会触发。
2. richTextBox.PreviewMouseLeftButtonUp += richText_MouseUp;
隧道事件在子元素之前为父元素触发。冒泡则相反。如果你涉及许多事件处理程序,你真的应该坚持全部使用冒泡或全部使用隧道,否则你添加的事件处理程序越多,就会变得越混乱 —— 添加一个新的事件处理程序可能会导致你重新审视应用程序中的所有其他事件处理程序。大多数人认为冒泡模型更加自然。