WPF 原生绑定命令功能使用指南

魏刘宏 2020 年 2 月 21 日

如今,当谈到 WPF 时,我们言必称 MVVM、框架(如 Prism)等,似乎已经忘了不用这些的话该怎么使用 WPF 了。当然,这里说的不用框架和 MVVM,并不是说像使用 Winform 那样使用 WPF,而是追本溯源,重识 WPF 与生俱来的绑定命令的风采。

一、绑定的使用

目标:前台页面通过绑定获取后台属性的值。

这个目标实际上分为两部分,一是前台获取后台的属性值,二是属性值变动后能够及时体现出来。

要实现目标的第一部分,实际只需在窗体后台的构造函数中添加一行代码即可:

this.DataContext = this;

这行代码很关键,MVVM 模式中页面与 ViewModel 关联也是通过指定页面类的 DataContext 为相应的 ViewModel 对象来实现的。

下面再来说说如何实现目标的第二部分,也就是属性变化后能及时体现出来,包括后台属性变化后前台显示自动变化,以及前台修改了内容,后台属性的值跟着改变。众所周知,这就是绑定,而要实现这一功能,需要相关类实现一个属性变动通知接口 —— InotifyPropertyChanged 。具体演变过程可参考网上的文章《 .NET 4.5 (C#):INotifyPropertyChanged 执行的演变:从表达式树到调用方信息的 BindableBase 类型 | Mgen》,这里直接给出最后的结果。

首先,实现 InotifyPropertyChanged 当然是必要的,如果是要绑定其他类,则让该类实现之,如果是直接在窗口后台做相关功能,则最终窗口类看上去像这样:

public partial class MainWindow : Window, INotifyPropertyChanged

然后添加一个事件和两个方法:

public event PropertyChangedEventHandler PropertyChanged;

protected void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
var eventHandler = this.PropertyChanged;
eventHandler?.Invoke(this, new PropertyChangedEventArgs(propertyName));
} protected bool SetProperty<T>(ref T storage, T value, [CallerMemberName] String propertyName = null)
{
if (Equals(storage, value)) return false; storage = value;
this.OnPropertyChanged(propertyName);
return true;
}

最后就是要提供绑定的属性了,可以像下面这样写:

private string _UserName = "wlh";
public string UserName
{
get => _UserName;
set => SetProperty(ref _UserName, value);
}

前台绑定就很简单了:

<TextBox Text="{Binding UserName, Mode=TwoWay}"></TextBox>

二、命令 ICommand

WPF 和 Winform 的重大区别就是,用户的交互、数据的变化等,在 Winform 中,都需要程序员一点一点仔细地手动处理,而在 WPF 中,数据是绑定的,交互通过命令传递,所以很多事情其实 WPF 这个大框架就可以帮我们自动处理了。说了这么多,其实就是说 Winform 是事件驱动的,而 WPF 是数据驱动的,所以在 Winform 中常用的按钮点击事件等各种事件,在 WPF 中是不怎么用了,而是使用命令。

命令也是绑定的,先来看看前台的样子:

<TextBox Text="{Binding UserName, Mode=TwoWay}"></TextBox>

至于后台怎么写,先不急,通过《[WPF] ICommand 最佳使用方法》一文,我们知道首先需要一个辅助类:

public class RelayCommand : ICommand
{
private readonly Predicate<object> _CanExecute;
private readonly Action<object> _Execute; public RelayCommand(Predicate<object> canExecute, Action<object> execute)
{
this._CanExecute = canExecute;
this._Execute = execute;
} public event EventHandler CanExecuteChanged
{
add => CommandManager.RequerySuggested += value;
remove => CommandManager.RequerySuggested -= value;
} public bool CanExecute(object parameter)
{
return _CanExecute(parameter);
} public void Execute(object parameter)
{
_Execute(parameter);
}
}

可见 ICommand 中主要有两个方法,一个检查命令是否可用的 CanExecute (),以及实际干活的 Execute () 。

然后在后台添加一个 “DoSomething” 的命令,也就是上面新建的 RelayCommand 类型:

private ICommand _DoSomething;
public ICommand DoSomethingCommand
{
get
{
return _DoSomething ??= new RelayCommand(
o => _CanDoSomething(o),
o => { _DoSomethingMethod(o); });
}
} private readonly Predicate<object> _CanDoSomething = o => true; // 可在之后再赋值,避免方法体中访问属性等受阻;
private readonly Action<object> _DoSomethingMethod = o =>
{
// do something
};

这些还可以进一步简化为:

public ICommand DoSomethingCommand { get; set; }

/// <summary>
/// 命令方法赋值(在构造方法中调用)
/// </summary>
private void SetCommandMethod()
{
DoSomethingCommand ??= new RelayCommand(o => true, async o =>
{
// do something
});
}

最后来看看对应前台”GetTokenCommand” 命令的实际业务代码:

public ICommand GetTokenCommand { get; set; }

/// <summary>
/// 命令方法赋值 (在构造函数中调用)
/// </summary>
private void SetCommandMethod()
{
GetTokenCommand ??= new RelayCommand(o => !(string.IsNullOrEmpty(UserName) || string.IsNullOrEmpty(Password)), async o =>
{
var req = new ReqGetToken()
{
userName = UserName,
password = Password,
}; var res = await GetToken(req);
if (res.Code)
{
Token = res.Token;
}
});
}

可以看到,在检查命令是否可用的部分,没有像样板代码那样直接返回 true ,而是按照实际情况判断,这样的效果就是,当条件不满足时,前台相关控件自动禁用:

最后,经过我们这样写,其实和 MVVM 模式已经很接近了,只要把后台所有代码都移到另一个类,然后将页面的 DataContext 重新指定一下,就能实现页面显示和业务逻辑分离了。

WPF 原生绑定和命令功能使用指南的更多相关文章

  1. WPF DataGrid 绑定行双击行命令

    WPF DataGrid 绑定行双击行命令 <DataGrid ...> <DataGrid.InputBindings> <MouseBinding MouseActi ...

  2. 整理:WPF用于绑定命令和触发路由事件的自定义控件写法

    原文:整理:WPF用于绑定命令和触发路由事件的自定义控件写法 目的:自定义一个控件,当点击按钮是触发到ViewModel(业务逻辑部分)和Xaml路由事件(页面逻辑部分) 自定义控件增加IComman ...

  3. WPF自学入门(十一)WPF MVVM模式Command命令 WPF自学入门(十)WPF MVVM简单介绍

    WPF自学入门(十一)WPF MVVM模式Command命令   在WPF自学入门(十)WPF MVVM简单介绍中的示例似乎运行起来没有什么问题,也可以进行更新.但是这并不是我们使用MVVM的正确方式 ...

  4. Prism 源码解读5-数据绑定和命令

    介绍 WPF本身就支持通知.绑定和命令,实现ViewModel和VIew之间的通讯,但相对来说功能比较少,Prism扩充了这些功能并提供更加强有力,简洁的数据绑定和命令. 0 绑定通知 WPF的绑定通 ...

  5. 【WPF】最近在学习wpf 的绑定,,

    最近在学习wpf 的绑定,,1.简单的说就是版前端和后端用自己的方法给分开了2.baseVm 模型 baseCmd 命令3.命令传参修改的只是界面里的属性,而不修改其它的值4.前端改变后端, 后端改变 ...

  6. 转: windows 10使用原生linux bash命令行

    转: https://www.zybuluo.com/pandait/note/337430 windows 10使用原生linux bash命令行 linux bash windows-10 第一时 ...

  7. C#使用Xamarin开发可移植移动应用(3.进阶篇MVVM双向绑定和命令绑定)附源码

    前言 系列目录 C#使用Xamarin开发可移植移动应用目录 源码地址:https://github.com/l2999019/DemoApp 可以Star一下,随意 - - 说点什么.. 嗯..前面 ...

  8. C#使用Xamarin开发可移植移动应用(4.进阶篇MVVM双向绑定和命令绑定)附源码

    前言 系列目录 C#使用Xamarin开发可移植移动应用目录 源码地址:https://github.com/l2999019/DemoApp 可以Star一下,随意 - - 说点什么.. 嗯..前面 ...

  9. WPF DataGrid 绑定数据及时更新的处理

    原文:WPF DataGrid 绑定数据及时更新的处理 默认情况下datagrid 绑定数据源后,在界面编辑某一列后,数据不会及时更新到内存对象中.如在同一行上有一个命令对来获取 当前选中行(内存对象 ...

随机推荐

  1. [bzoj4815] [洛谷P3700] [Cqoi2017] 小Q的表格

    Description 小Q是个程序员. 作为一个年轻的程序员,小Q总是被老C欺负,老C经常把一些麻烦的任务交给小Q来处理. 每当小Q不知道如何解决时,就只好向你求助.为了完成任务,小Q需要列一个表格 ...

  2. 玩转Django2.0---Django笔记建站基础九(二)(Auth认证系统)

    9.4 设置用户权限 用户权限主要是对不同的用户设置不同的功能使用权限,而每个功能主要以模型来划分.以9.3节的MyDjango项目为例,在Admin后台管理系统可以查看并设置用户权限,如下图: 用户 ...

  3. Frameworks.Entity.Core 1

    CommonEnums 1系统模块BlockType 2证件类型IDType 3在线支付类型OnLineType 4操作权限,支持位移运算OperatorAuthority 5订单状态: 1000-待 ...

  4. 深入Node模块Buffer-学会操作二进制

    Buffer 作为 nodejs 中重要的概念和功能,为开发者提供了操作二进制的能力.本文记录了几个问题,来加深对 Buffer 的理解和使用: 认识缓冲器 如何申请堆外内存 如何计算字节长度 如何计 ...

  5. jmeter使用—计数器的使用

    说一下jmeter中,配置元件-计数器的使用. 如果需要引用的数据量较大,且要求不能重复或者需要自增,那么可以使用计数器来实现. 1.启动jmeter,添加线程组,右键添加配置元件——计数器,如下图: ...

  6. CSS-09-背景属性

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  7. 笔记常用Linux命令(二) 进程和端口

    查看系统进程 ps:用于报告当前系统的进程状态 a:显示所有终端机下执行的程序 ps -ef/ps aux: 这两个命令都是查看当前系统正在运行进程,两者的区别是展示格式不同. 如果想要查看特定的进程 ...

  8. 实验3: DHCP 基本配置

    实验3-1: DHCP 基本配置 实验目的通过本实验可以掌握:(1)DHCP 的工作原理和工作过程(2)DHCP 服务器的基本配置和调试(3)客户端配置 拓扑结构 实验步骤n    步骤1:配置路由器 ...

  9. LR中解决接口请求中包含中文字符,服务器不识别的问题

    在LR中,直接写的接口请求,如果请求字段包含中文字段,服务器会不识别,这个时候就要用到lr_convert_string_encoding这个函数: 具体用法: lr_convert_string_e ...

  10. 时序数据库 Apache-IoTDB 源码解析之文件格式简介(三)

    上一章聊到在车联网或物联网中对数据库的需求,以及 IoTDB 的整体架构,详情请见: 时序数据库 Apache-IoTDB 源码解析之系统架构(二) 打一波广告,欢迎大家访问IoTDB 仓库,求一波 ...