当前位置: 首页 > news >正文

理解WPF Stylet中Command=“{s:Action 方法名}“的设计与实现 - 实践

前言

Stylet是我最近很喜欢使用的一个WPF框架,它的很多设计都体现了约定优于配置的思想。因此你会发现使用它非常方便,几乎不需要任何配置,开箱即用,只需知道它的一些约定即可。

查看Samples中Hello这个例子,只要在xaml中这样写:

Say Hello

然后你点击这个按钮,就会触发对应ViewModel中的SayHello方法,使用起来非常简单方便。这背后Stylet框架做了什么呢?让我们揭开它的神秘面纱吧!!

ActionExtension

先来看看ActionExtension,它位于Stylet.Xaml命名空间:

s:Action的第一阶段是XAML解析,这个阶段的核心任务是将XAML标记中的{s:Action SayHello}语法翻译成CommandAction实例。这个过程由ActionExtension标记扩展类完成,它是整个命令绑定系统的入口点。

当XAML解析器遇到{s:Action SayHello}时,会调用ActionExtension.ProvideValue方法:

HandleDependencyObject方法根据目标属性的类型进行分支处理:

对于命令绑定场景,CreateCommandAction方法创建CommandAction实例:

rootObject就是具有s:Action的页面:

View.ActionTarget

现在我们已经找到了View,但是想要触发的方法是在ViewModel上的,那么就要想办法找到对应的ViewModelStylet中是通过View.ActionTarget这个附加属性实现的。通过View.ActionTarget附加属性将ViewModel注入到可视化树中,使得后续阶段能够找到正确的命令执行目标。

先来看看View.ActionTarget的定义,位置在命名空间Stylet.Xaml下的View类中:

默认值:InitialActionTarget(一个特殊的标记对象)

继承性:FrameworkPropertyMetadataOptions.Inherits确保属性值可以沿着可视化树向下传递

View.Model属性被设置时,会触发PropertyChangedCallback,通过ViewManager建立视图和ViewModel的关联。

在将ViewViewModel关联起来的时候,设置了当前ViewActionTarget为对应的ViewModel

CommandAction

CreateCommandAction方法中会返回一个CommandAction

这里是在把当前这个Subject对象里的ActionTargetProperty的值‘拴’到 某个尚未显式的目标属性上,而且只让它从源(Subject)流向目标,不会反向同步。

这里的SubjectButton对象,为什么也能找到ActionTargetProperty这个属性呢?

由于ActionTargetProperty设置了FrameworkPropertyMetadataOptions.Inherits标志,这个属性会自动沿着可视化树向下传播:

Window (View.ActionTargetProperty)
├── UserControl (继承Window的View.ActionTargetProperty)
│ ├── Button (继承UserControl的View.ActionTargetProperty)
│ └── TextBox (继承UserControl的View.ActionTargetProperty)
└── Grid (继承Window的View.ActionTargetProperty)

但是现在还没完成绑定,只是绑定的一端,还需要设置将这个属性绑定到哪里。

BindingOperations.SetBinding(this, targetProperty, multiBinding);ActionBasetargetProperty依赖属性绑定到View.ActionTargetProperty

你会发现现在并没有绑定到ViewModel,只是View.ActionTargetProperty的默认值:

由于targetProperty依赖属性从nullView.InitialActionTarget也会触发UpdateActionTarget,不过这一次不做任何处理,直接返回:

当设置View.ActionTargetProperty的值为对应的ViewModel时:

就又会触发UpdateActionTarget方法,这一次拿到了对应的ViewModel

拿到ViewModel上的方法:

拿到是否可以执行对应命令的属性:

现在重点来关注一下这里:

Expressions.ConstantExpression targetExpression = Expressions.Expression.Constant(newTarget);
Expressions.MemberExpression propertyAccess = Expressions.Expression.Property(targetExpression, guardPropertyInfo);
this.guardPropertyGetter = Expressions.Expression.Lambda<Func<
bool>
>(propertyAccess).Compile();

这段代码在运行时动态编译一段表达式树,生成一个无参、返回 bool的委托(Func<bool>),用来快速读取某个对象的布尔型属性值。

第一行相当于就是使用这个newTarget对象,第二行就是访问这个newTarget对象的CanSayHello属性,相当于newTarget.CanSayHello

Expression.Lambda<Func<bool>>将表达式树包装成一个无参Lambda表达式(形如 () => newTarget.CanSayHello)。

.Compile()把表达式树编译成可执行IL ,生成一个静态缓存的委托Func<bool>

然后调用this.UpdateCanExecute();触发CanExecuteChanged事件,就会执行CanExecute方法,在这个方法中就使用到了刚刚生成的获取newTarget.CanSayHello属性的委托:

现在是false,命令无法执行:

当我输入小明的时候就修改了这个属性值,就会触发CanExecuteChanged事件:

就又会调用刚刚生成的那个获取newTarget.CanSayHello属性的委托:

这一次返回true,命令可以执行。

当我点击按钮的时候,如果命令可以执行,就会执行Execute方法

这样就会执行ViewModel中对应的方法。

以上就是个人对于WPF Stylet中Command="{s:Action SayHello}"的设计与实现的理解。

http://www.hskmm.com/?act=detail&tid=28941

相关文章:

  • 2025环氧地坪漆厂家推荐:常州新禾,品质保证施工无忧!
  • 概率论习题
  • 2025上海经侦律师TOP5榜单:专业法律服务与高效解决方案
  • 概率论部分习题
  • 2025家居ERP推荐:赛思软件助力企业高效管理!
  • 2025彩钢瓦保养优质厂家推荐,江苏承优建筑工程专业服务!
  • 优维科技一面
  • 2025家纺摄影公司最新推荐榜:创意视觉与专业服务的完美结合
  • 2025磁力泵加工厂推荐中正化工,专业定制高效耐用产品!
  • 线段树分治
  • 2025双氧水供应厂家推荐:苏州市岚昱化工品质卓越选择!
  • 2025婚纱照拍摄推荐,南通造物摄影有限公司专业团队打造梦幻
  • 2025上海保洁公司最新推荐榜:高效清洁与贴心服务的优质选择
  • 10.11
  • 「解题报告」蓝桥杯2013省AB 错误票据
  • 2025精密弹簧优质厂家推荐:蓝侨盈科技,精准弹性解决方案!
  • 时时想起 寸步难行 叩问自己 无人回应 若我离去 若我死去 枯萎于这幽暗的井底 长眠不醒
  • 有限空间作业安全无死角!AI 视觉守护人员与操作合规
  • 2025抖音推广服务商最新推荐榜:精准引流与高效转化的营销利
  • 2025甘肃西服定制店推荐榜单:匠心工艺与贴心服务的完美结合
  • 完整教程:计算机毕业设计免费领源码-教师教学进度管理及建议系统的设计与实现
  • 2025表面瑕疵检测设备厂家最新推荐:精准高效,工业品质之选
  • 战略、运营、经营三循环:企业卓越绩效的密码 - 智慧园区
  • 2025书包柜定做厂家推荐:杰尚家具专业定制,品质卓越!
  • 2025环氧板定制厂家推荐:一博科技材料,专业定制品质卓越!
  • 2025农机带实力厂家推荐:浙江三星胶带,品质卓越供货无忧!
  • 打不动十个
  • CSP-S模拟29 2025.10.11
  • 2025风机盘管优质厂家推荐:洛卡尔环境科技,高效节能首选!
  • 最简单实用的SQL注入检测方法:Break Repair技巧详解