1. 点击事件的执行顺序

  1. PreviewMouseLeftButtonDown
  2. PreviewMouseDown
  3. MouseLeftButtonDown
  4. MouseDown
  5. PreviewMouseLeftButtonUp
  6. PreviewMouseUp
  7. Click
  8. MouseLeftButtonUp
  9. MouseUp

1)PreviewMouseLeftButtonDown 是隧道事件,从根节点往下传递

2)MouseLeftButtonDown 是冒泡事件,从下往根节点传递

3)知道这两点,再来看前面从 PreviewMouseLeftButtonDown 到 MouseUp 的整个过程及 Click 在当中所处的顺序,就好理解很多

4)具体:

鼠标即将点下,从根节点往下传递

->控件接受到鼠标点下,从下往根节点冒泡

->鼠标即将弹起,从上往下传递

->控件接受到鼠标要松开,若是 ButtonBase 等有 Click 事件的控件则触发 Click 事件

->控件从下往上冒泡鼠标弹起事件。

2. 例子

<StackPanel x:Name="s0" Height="100" Width="200" Background="Red"
PreviewMouseLeftButtonDown="PreviewMouseLeftButtonDownEventHandler"
PreviewMouseDown="PreviewMouseDownEventHandler"
MouseLeftButtonDown="MouseLeftButtonDownEventHandler"
MouseDown="MouseDownEventHandler"
PreviewMouseLeftButtonUp="PreviewMouseLeftButtonUpEventHandler"
PreviewMouseUp="PreviewMouseUpEventHandler"
MouseLeftButtonUp="MouseLeftButtonUpEventHandler"
MouseUp="MouseUpEventHandler">
<StackPanel x:Name="s1" Height="90" Width="180" Background="Yellow"
PreviewMouseLeftButtonDown="PreviewMouseLeftButtonDownEventHandler"
PreviewMouseDown="PreviewMouseDownEventHandler"
MouseLeftButtonDown="MouseLeftButtonDownEventHandler"
MouseDown="MouseDownEventHandler"
PreviewMouseLeftButtonUp="PreviewMouseLeftButtonUpEventHandler"
PreviewMouseUp="PreviewMouseUpEventHandler"
MouseLeftButtonUp="MouseLeftButtonUpEventHandler"
MouseUp="MouseUpEventHandler">
<Button x:Name="btn" Height="40" Width="78" Content="click"
Click="Button_ClickEventHandler"
MouseLeftButtonDown="MouseLeftButtonDownEventHandler"
PreviewMouseLeftButtonDown="PreviewMouseLeftButtonDownEventHandler"
PreviewMouseDown="PreviewMouseDownEventHandler"
MouseDown="MouseDownEventHandler"
PreviewMouseLeftButtonUp="PreviewMouseLeftButtonUpEventHandler"
PreviewMouseUp="PreviewMouseUpEventHandler"
MouseLeftButtonUp="MouseLeftButtonUpEventHandler"
MouseUp="MouseUpEventHandler">
</Button>
</StackPanel>
</StackPanel>
private void PrintControlAndEvent(string controlName, string routedEventName)
{
Console.WriteLine(string.Format("{0} : {1}", controlName, routedEventName));
} private void PreviewMouseLeftButtonDownEventHandler(object sender, MouseButtonEventArgs e)
{
PrintControlAndEvent((sender as FrameworkElement).Name, e.RoutedEvent.ToString());
}
...
private void Button_ClickEventHandler(object sender, RoutedEventArgs e)
{
PrintControlAndEvent((sender as FrameworkElement).Name, e.RoutedEvent.ToString());
}

2.1 点击 Button

输出:

s0 : UIElement.PreviewMouseLeftButtonDown
s0 : Mouse.PreviewMouseDown
s1 : UIElement.PreviewMouseLeftButtonDown
s1 : Mouse.PreviewMouseDown
btn : UIElement.PreviewMouseLeftButtonDown
btn : Mouse.PreviewMouseDown
s0 : UIElement.PreviewMouseLeftButtonUp
s0 : Mouse.PreviewMouseUp
s1 : UIElement.PreviewMouseLeftButtonUp
s1 : Mouse.PreviewMouseUp
btn : UIElement.PreviewMouseLeftButtonUp
btn : Mouse.PreviewMouseUp
btn : ButtonBase.Click

2.2 点击 s1

输出:

s0 : UIElement.PreviewMouseLeftButtonDown
s0 : Mouse.PreviewMouseDown
s1 : UIElement.PreviewMouseLeftButtonDown
s1 : Mouse.PreviewMouseDown
s1 : UIElement.MouseLeftButtonDown
s1 : Mouse.MouseDown
s0 : UIElement.MouseLeftButtonDown
s0 : Mouse.MouseDown
s0 : UIElement.PreviewMouseLeftButtonUp
s0 : Mouse.PreviewMouseUp
s1 : UIElement.PreviewMouseLeftButtonUp
s1 : Mouse.PreviewMouseUp
s1 : UIElement.MouseLeftButtonUp
s1 : Mouse.MouseUp
s0 : UIElement.MouseLeftButtonUp
s0 : Mouse.MouseUp

2.3 点击 Button 后未能触发 MouseLeftButtonDown、MouseLeftButtonUp、MouseDown、MouseUp 的原因

button 的 mousedown 等已经被内部处理了,Handled 已被标记为 True。如果想强制执行,可以

btn.AddHandler(Button.MouseDownEvent, new MouseButtonEventHandler(MouseDownEventHandler), true);
btn.AddHandler(Button.MouseLeftButtonDownEvent, new MouseButtonEventHandler(MouseLeftButtonDownEventHandler), true);
btn.AddHandler(Button.MouseUpEvent, new MouseButtonEventHandler(MouseUpEventHandler), true);
btn.AddHandler(Button.MouseLeftButtonUpEvent, new MouseButtonEventHandler(MouseLeftButtonUpEventHandler), true);
第三个入参:
handledEventsToo:
//如果为 true,则将按以下方式注册处理程序:即使路由事件在其事件数据中标记为已处理,也会调用处理程序;
//如果为 false,则使用默认条件注册处理程序,即当路由事件被标记为已处理时,将不调用处理程序。

xaml 里写的 MouseLeftButtonDown="MouseLeftButtonDown" 的 handledEventsToo 值就是 false。

所以,在已经被内部处理的情况下,无法触发。

而即使在后台写 true,也只能触发 Button 的 MouseDownEvent,不会继续往上传递。

因为 handled 标记为 true,除非再把上层的控件也添加路由回调为 true。

WPF 基础 - 点击事件的执行顺序及 Button 点击事件的特殊性的更多相关文章

  1. 关于Unity物理事件的执行顺序的最新理解

    物体A: public class A:{ B b; void FixedUpdate(){ if(input.GetKeyDow(Keycode.I)) { collider.enable=fals ...

  2. jquery ajax中事件的执行顺序

    jquery中各个事件执行顺序如下: 1.ajaxStart(全局事件) 2.beforeSend 3.ajaxSend(全局事件) 4.success 5.ajaxSuccess(全局事件) 6.e ...

  3. 原生js事件和jquery事件的执行顺序问题

    场景:近日,写前端页面时候,在针对输入框input操作时,用到了jquery的插件,插件中使用了jquery的focus()和blur()方法.但是同时,又需要在插件之外再针对输入框的获取焦点和失去焦 ...

  4. JAVA基础2——类初始化相关执行顺序

    类初始化相关执行顺序 几个概念说明 代码块的含义与作用 static静态代码块: 一般用于初始化类中的静态变量.比如:给静态的数组或者list变量赋初值.使用static静态代码块进行初始化与直接在定 ...

  5. go语言基础之多个defer执行顺序

    1. 多个defer执行顺序 如果一个函数中有多个defer语句,它们会以LIFO(后进先出)的顺序执行.哪怕函数或某个延迟调用发生错误,这些调用依旧会被执.示例: package main //必须 ...

  6. testng基础知识:注解的执行顺序

    1. 单类,无继承父子关系 code: public class basicTest { @BeforeSuite(alwaysRun = true) public void beforeSuite_ ...

  7. - > 贪心基础入门讲解五——任务执行顺序

    分析: 本题可以抽象成,从一个整数开始,每次减去a,再加上b (a,b都是正数),要求每次操作都不产生负数. 针对本题a[i] = R[i], b[i] = R[i] – O[i],注意O[i] &l ...

  8. Java 基础:继承中的执行顺序

    1.单独的父类测试 Java中,new一个类的对象,类里面的静态代码块.非静态代码.无参构造方法.有参构造方法.类的一般方法等部分, 它们的执行顺序相对来说比较简单,用程序也很容易验证. 比如新建一个 ...

  9. iframe子页面点击按钮,执行父页面的点击事件

    iframe 子页面点击.parent 父页面  的id(auth-link-btn)的事件 <a href="javascript:void(0);" onclick=&q ...

随机推荐

  1. 如何创建一个GETH节点(单节点,windows环境)

    所有命令都是在powershell上执行的 1.创建"创世块" 初始化配置 创建一个 hdgenesis.json文件,拷贝到geth根目录 {    "config&q ...

  2. Bootstrap导航组件

    Bootstrap 中的导航组件都依赖同一个 .nav 类,状态类也是共用的.改变修饰类可以改变样式. 标签页 注意 .nav-tabs 类依赖 .nav 基类 <ul class=" ...

  3. 智能广告投放平台 All in One

    智能广告投放平台 All in One app demos 知之数据 一站式广告营销平台 https://hio.cn/ refs https://www.jonmellman.com/posts/p ...

  4. vuepress & ReferenceError: window is not defined

    vuepress & ReferenceError: window is not defined bug Client Compiled successfully in 15.35s Serv ...

  5. js console.log color all in one

    js console.log color all in one console.log color Chrome console.log 语法 / grammar %c, %s, css style ...

  6. IndexedDB All In One

    IndexedDB All In One https://developer.mozilla.org/en-US/docs/Web/API/IndexedDB_API https://develope ...

  7. svg-path-to-polygons & svg-path-parser

    svg-path-to-polygons & svg-path-parser svg-path-to-polygons https://www.npmjs.com/package/svg-pa ...

  8. SVG to GeoJSON

    SVG to GeoJSON GEOJSON https://geojson.org/ http://geojson.io/ https://github.com/mapbox/geojson.io/ ...

  9. ES进行date_histogram时间聚合,聚合结果时间不正确问题

    在做项目中,有一个需求是统计本周内每天的漏洞数量,我选用的是ES中的date_histogram函数来进行聚合统计: 但是出现了一个问题,聚合出来的结果和想要统计的结果时间不一致,如下图所示 时间区间 ...

  10. NGK生态之星空计划启动在即,稀有VAST高兑换比带来高价值!

    NGK生态之星空计划启动在即,为了感谢NGK布道者的支持,NGK官方将全力辅助算力市场,开展全新的星空计划,并发行星空币SPC,空投给算力持有者进行额外奖励. 至此,SPC已经顺利完成2轮空投,也初步 ...