最近在学WPF的MVVM,有两种方式实现,一种是自己实现,一种是借助MVVM框架,接下来通过一个医院自助打印报告机键盘输入界面来演示自己实现、框架CommunityToolkit和Prism的区别。

项目源码:https://gitee.com/cplmlm/SelfServiceReportPrinter、https://github.com/cplmlm/SelfServiceReportPrinter

推荐学习博主:B站UP十月的寒流

一、自己实现

1、首先我们创建一个BaseNotifyPropertyChanged类,继承INotifyPropertyChanged,这个方法的作用是属性值变化时自动更新UI界面。

 public class BaseNotifyPropertyChanged : INotifyPropertyChanged{public event PropertyChangedEventHandler? PropertyChanged;public void RaisePropertyChanged(string propertyName){if (!string.IsNullOrEmpty(propertyName)){PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));}}}

2、创建一个KeyPressViewModel类,继承BaseNotifyPropertyChanged。

 public class KeyPressViewModel : BaseNotifyPropertyChanged{private string cardNumber;private int selectionStart=0;/// <summary>/// 输入文本框的值/// </summary>public string CardNumber{get { return cardNumber; }set{cardNumber = value;RaisePropertyChanged(nameof(CardNumber));}}/// <summary>/// 输入框光标位置/// </summary>public int SelectionStart{get { return selectionStart; }set { RaisePropertyChanged(nameof(SelectionStart)); }}/// <summary>/// 数字按钮绑定事件/// </summary>public ICommand NumberCommand{get { return new RelayCommand<string>(Number); }}/// <summary>/// 清空按钮绑定事件/// </summary>public ICommand ClearCommand{get { return new RelayCommand(Clear); }}/// <summary>/// 删除按钮绑定事件    /// </summary>public ICommand DeleteCommand{get { return new RelayCommand(Delete); }}/// <summary>/// 数字点击事件/// </summary>/// <param name="key"></param>private void Number(string? key){CardNumber += key;}/// <summary>/// 清空点击事件/// </summary>private void Clear(){CardNumber = string.Empty;}/// <summary>/// 删除点击事件/// </summary>private void Delete(){// 光标在输入框时,删除光标前一个字符if (!string.IsNullOrEmpty(CardNumber) && SelectionStart > 0){CardNumber = CardNumber.Remove(SelectionStart - 1, 1);}//光标没有在输入框时,删除最后一个字符if (SelectionStart == 0){CardNumber = CardNumber.Remove(CardNumber.Length - 1, 1);}}}

3、创建RelayCommand类,这个类的作用是绑定事件的操作,一个泛型版本,一个是非泛型。

public class RelayCommand : ICommand
{private Action _execute;private Func<bool> _canExecute;public RelayCommand(Action execute) : this(execute, null){}public RelayCommand(Action execute, Func<bool> canExecute){if (execute == null)throw new ArgumentNullException("execute");_execute = execute;_canExecute = canExecute;}public event EventHandler? CanExecuteChanged{add{if (_canExecute != null){CommandManager.RequerySuggested += value;}}remove{if (_canExecute != null){CommandManager.RequerySuggested -= value;}}}public bool CanExecute(object parameter){return _canExecute == null ? true : _canExecute();}public void Execute(object parameter){_execute();}
}
public class RelayCommand<T> : ICommand
{private readonly Predicate<T> _canExecute;private readonly Action<T> _execute;public RelayCommand(Action<T> execute) : this(execute, null){}public RelayCommand(Action<T> execute, Predicate<T> canExecute){if (execute == null)throw new ArgumentNullException("execute");_execute = execute;_canExecute = canExecute;}public event EventHandler? CanExecuteChanged{add { CommandManager.RequerySuggested += value; }remove { CommandManager.RequerySuggested -= value; }}public bool CanExecute(object parameter){return _canExecute == null ? true : _canExecute((T)parameter);}public void Execute(object parameter){_execute((T)parameter);}
}

4、通过binding绑定输入框的值和事件的操作,代替传统直接在后台cs文件写事件。

<TextBox Text="{Binding CardNumber}"     x:Name="CardNumberTextBox"   local:TextBoxSelectionHelper.SelectionStart="{Binding SelectionStart, Mode=TwoWay}" Height="40"     Width="460" />  
<Button  Content="1" Style="{StaticResource NumberButtonStyle}" Command="{Binding NumberCommand}"  CommandParameter="1"/>
<Button  Content="删除" Style="{StaticResource RedButtonStyle}" Command="{Binding DeleteCommand}"   />
<Button  Content="清空" Style="{StaticResource RedButtonStyle}" Command="{Binding ClearCommand}"   />

5、将MainWindow的DataContext赋值给ViewModel,我这里用了依赖注入的方法,所以直接是在app.cs里面赋值的,也可以在MainWindow.cs。

  public partial class App : Application{public App(){Services = ConfigureServices();this.InitializeComponent();}/// <summary>/// Gets the current <see cref="App"/> instance in use/// </summary>public new static App Current => (App)Application.Current;/// <summary>/// Gets the <see cref="IServiceProvider"/> instance to resolve application services./// </summary>public IServiceProvider Services { get; }/// <summary>/// Configures the services for the application./// </summary>private static IServiceProvider ConfigureServices(){          var services = new ServiceCollection();services.AddTransient<KeyPressViewModelCommunityToolkit>();services.AddTransient<KeyPressViewModelPrism>();services.AddTransient<KeyPressViewModel>();services.AddTransient(sp=>new MainWindow() { DataContext=sp.GetRequiredService<KeyPressViewModelCommunityToolkit>()});return services.BuildServiceProvider();}protected override void OnStartup(StartupEventArgs e){base.OnStartup(e);MainWindow= Services.GetRequiredService<MainWindow>();MainWindow.Show();}}
    public partial class MainWindow : Window{public MainWindow(){InitializeComponent();DataContext =App.Current.Services.GetService<KeyPressViewModel>();}}

6、如果不用依赖注入的方式,可以在MainWindow.xaml或者MainWindow.cs将MainWindow的DataContext赋值给ViewModel。

<Window x:Class="SelfServiceReportPrinter.MainWindow"xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d="http://schemas.microsoft.com/expression/blend/2008"xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"xmlns:selfservicereportprinter="clr-namespace:SelfServiceReportPrinter"      xmlns:local="clr-namespace:SelfServiceReportPrinter" mc:Ignorable="d"WindowStartupLocation="CenterScreen" AllowsTransparency="True"WindowStyle="None"Title="MainWindow" Height="1080" Width="1920"><Window.DataContext><local:KeyPressViewModel /></Window.DataContext>
 public partial class MainWindow : Window{public MainWindow(){InitializeComponent();DataContext =new KeyPressViewModel();}}

以上就是自己去实现mvvm的方式,会比较繁琐,如果不是复杂的项目大多数还是用社区的CommunityToolkit.MVVM,复杂的项目可以使用Prism

二、使用微软社区的CommunityToolkit.MVVM

1、首先安装CommunityToolkit.MVVM的包。

image

2、创建一个新的类KeyPressViewModelCommunityToolkit类,继承ObservableObject,CommunityToolkit代码就简洁很多了,直接在方法或者属性上面加特性就可以。

public partial class KeyPressViewModelCommunityToolkit : ObservableObject
{[ObservableProperty]private string cardNumber = string.Empty;[ObservableProperty]private int selectionStart;/// <summary>/// 数字点击事件/// </summary>/// <param name="key"></param>[RelayCommand]private void Number(string? key){CardNumber += key;}/// <summary>/// 清空输入框/// </summary>[RelayCommand]private void Clear(){CardNumber = string.Empty;}/// <summary>/// 删除点击事件/// </summary>[RelayCommand]private void Delete(){// 光标在输入框时,删除光标前一个字符if (!string.IsNullOrEmpty(CardNumber) && SelectionStart > 0){CardNumber = CardNumber.Remove(SelectionStart - 1, 1);}//光标没有在输入框时,删除最后一个字符if (SelectionStart == 0){CardNumber = CardNumber.Remove(CardNumber.Length - 1, 1);}}
}

三、Prism

1、安装Prism.Core和Prism.Wpf,其他的包根据后续实际使用在安装。

image

2、创建一个KeyPressViewModelPrism类继承BindableBase。

public partial class KeyPressViewModelPrism : BindableBase
{private string cardNumber;private int selectionStart;public string CardNumber{get { return cardNumber; }set { SetProperty(ref cardNumber, value); }}public int SelectionStart{get { return selectionStart; }set { SetProperty(ref selectionStart, value); }}public DelegateCommand<string> NumberCommand { get; }public DelegateCommand ClearCommand { get; }public DelegateCommand DeleteCommand { get; }public KeyPressViewModelPrism(){NumberCommand = new DelegateCommand<string>(Number);ClearCommand = new DelegateCommand(Clear);DeleteCommand = new DelegateCommand(Delete);}/// <summary>/// 数字点击事件/// </summary>/// <param name="key"></param>private void Number(string? key){CardNumber += key;}/// <summary>/// 清空点击事件/// </summary>private void Clear(){CardNumber = string.Empty;}/// <summary>/// 删除点击事件/// </summary>private void Delete(){// 光标在输入框时,删除光标前一个字符if (!string.IsNullOrEmpty(CardNumber) && SelectionStart > 0){CardNumber = CardNumber.Remove(SelectionStart - 1, 1);}//光标没有在输入框时,删除最后一个字符if (SelectionStart == 0){CardNumber = CardNumber.Remove(CardNumber.Length - 1, 1);}}
}

四、总结

以上就是三种不同ViewModel的实现方式,个人比较推荐使用社区的CommunityToolkit,但是还是要根据自己的项目情况来决定。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/pingmian/94321.shtml
繁体地址,请注明出处:http://hk.pswp.cn/pingmian/94321.shtml
英文地址,请注明出处:http://en.pswp.cn/pingmian/94321.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[e3nn] docs | 不可约表示(Irreps)

链接&#xff1a;https://docs.e3nn.org/en/latest/examples/examples.html docs&#xff1a;e3nn e3nn是一个用于构建欧几里得(E(3))等变神经网络的Python库&#xff0c;这意味着它们能自动保持三维旋转和反射的对称性。 该库使用不可约表示(Irreps)来描述数据变换方式&…

深入浅出 ArrayList:从基础用法到底层原理的全面解析(中)

四、ArrayList 常用方法实战 —— 从添加到遍历的全场景覆盖ArrayList 提供了数十个方法&#xff0c;但日常开发中常用的只有 10 个左右&#xff0c;我们按 “元素操作”“集合查询”“遍历方式” 三类来梳理&#xff0c;每个方法都附带示例和注意事项。4.1 元素添加&#xff1…

java后端如何实现下载功能

后端需要把要下载的若干文件 按 ZIP 格式编码成一段二进制字节流&#xff0c;然后以 Content-Type: application/zip Content-Disposition: attachment; filenamexxx.zip 的形式写进 HTTP 响应体。浏览器收到这段“ZIP 格式的字节流”后&#xff0c;就会弹出保存对话框&#xf…

AI生成技术报告:GaussDB与openGauss的HTAP功能全面对比

GaussDB 与 openGauss 的 HTAP 功能比较 前言 GaussDB集中式版本从505.2版本开始引入了HTAP混合负载功能&#xff0c;openGauss也从7.0.0 RC1版本开始引入了HTAP行列融合功能&#xff0c;加强了行存转列存的使用友好度&#xff0c;但两者的实现似乎存在不小的差异。 虽然文档…

小程序开发指南(四)(UI 框架整合)

✍讲解了微信小程序 UI 框架的使用方法和特点&#xff0c;根据项目需求选择合适的组件库。附有相应的组件库预览码&#xff0c;也是将所有的微信小程序原生组件库整合在一起方便后续开发的使用。如果有不好或者有错误的地方请告知&#xff01;希望可以与大家相互的交流学习&…

golang 1.25.0 安装

wget https://golang.google.cn/dl/go1.25.0.linux-amd64.tar.gz tar -C /usr/local/ -xzf go1.25.0.linux-amd64.tar.gz ln -s /usr/local/go/bin/* /usr/bin/ go env -w GO111MODULEon go env -w GOPROXYhttps://goproxy.cn,direct

基于深度学习的人脸表情识别系统:YOLOv5/v6/v7/v8/v10模型实现与UI界面集成

基于YOLOv5/v7/v8的智能人脸表情识别系统:从算法原理到应用实现 表情识别的技术价值与挑战 人脸表情识别(Facial Expression Recognition, FERYOLOv5/v7/v8等深度学习算法构建高效的表情识别系统,并设计直观的UI界面集成方案。无论你是深度学习初学者还是有经验的开发者,…

初步了解多线程

系列文章目录 目录 系列文章目录 前言 一、进程 二、线程 1. 线程解决资源开销的方式 2. 线程和进程的联系和区别 三、多线程编程 1. 直观了解多线程 2. 线程的创建方式 1. 继承 Thread 重写 run() 方法 2. 实现 Runable 接口&#xff0c;重写 run() 方法 3. 继承 …

安卓Android低功耗蓝牙BLE连接异常报错133

安卓Android低功耗蓝牙BLE连接异常报错133 之前连接一直好好的,不知道为什么今天突然就连接不了蓝牙了,报错133,按照 找网上的说明总是说清除GATT缓存,其实并不是我的问题,最后看到这里https://softs.im/android-ble-%e8%bf%9e%e6%8e%a5%e9%94%99%e8%af%af133/ 有如下说明: 情…

【分治】快排与归并专题

分治思想 分&#xff08;Divide&#xff09;&#xff1a;将待排序数组不断拆分为两个等长&#xff08;或近似等长&#xff09;的子数组&#xff0c;直到子数组长度为 1&#xff08;天然有序&#xff09;。 治&#xff08;Conquer&#xff09;&#xff1a;递归排序每个子数组。 …

[Linux]学习笔记系列 -- mm/page_alloc

文章目录mm/page_alloc.c 伙伴系统内存分配器(Buddy System Memory Allocator) 内核物理内存管理的核心历史与背景这项技术是为了解决什么特定问题而诞生的&#xff1f;它的发展经历了哪些重要的里程碑或版本迭代&#xff1f;目前该技术的社区活跃度和主流应用情况如何&#xf…

3秒传输大文件:cpolar+Localsend实现跨网络秒传

文章目录前言1. 在Windows上安装LocalSend2. 安装Cpolar内网穿透3. 公网访问LocalSend4. 固定LocalSend公网地址用 cpolar 让 Localsend 突破距离限制就是这么简单&#xff01;三步轻松搞定&#xff1a;在手机和电脑上都安装 Localsend&#xff0c;在其中一台设备上运行 cpolar…

基于STM32单片机智能RFID刷卡汽车位锁桩设计

1 系统功能介绍 本系统是一个 基于 STM32 单片机的智能 RFID 刷卡车位锁桩控制系统&#xff0c;其设计理念来源于现实中智能停车场的车位锁桩管理。通过 RFID 刷卡认证、LCD1602 显示、继电器控制以及按键辅助操作&#xff0c;实现对车位的安全管理。该系统不仅模拟了车辆驶入与…

SQL185 试卷完成数同比2020年的增长率及排名变化

描述现有试卷信息表examination_info&#xff08;exam_id试卷ID, tag试卷类别, difficulty试卷难度, duration考试时长, release_time发布时间&#xff09;&#xff1a;试卷作答记录表exam_record&#xff08;uid用户ID, exam_id试卷ID, start_time开始作答时间, submit_time交…

网络编程中的TCP——TCP的连接的建立、关闭、状态转移

网络编程中的TCP——TCP的连接的建立、关闭、状态转移 TCP连接的建立和关闭wireshark捕获数据&#xff1a;TCP三次握手四次挥手的时序图&#xff1a;三次握手&#xff1a; 报文段1包含SYN标志&#xff0c;这是一个同步报文段&#xff0c;表示发起连接请求&#xff0c;包含自己起…

SQL 语句拼接在 C 语言中的实现与安全性分析

代码解析 // 构建SQL插入语句 char *sql_insert (char *)malloc(sizeof(char) * 200); // 分配200字节内存 strcpy(sql_insert, "INSERT INTO user(username, passwd) VALUES("); // 复制基础SQL语句 strcat(sql_insert, ""); // 添加单引号 strcat(sq…

`lock()` 和 `unlock()` 线程同步函数

1) 函数的概念与用途 lock() 和 unlock() 不是特定的标准库函数&#xff0c;而是线程同步原语的一般概念&#xff0c;用于在多线程环境中保护共享资源。在不同的编程环境和库中&#xff0c;这些函数有不同的具体实现&#xff08;如 POSIX 线程的 pthread_mutex_lock() 或 C 的 …

升级openssh后ORACLE RAC EM 安装失败处理

升级过程中由于SCP传输时目标目录/tmp/tempRACTrans_2025_08_22--18-25-44-032/ractrans 不存在导致的OC4J配置失败&#xff1a;WARNING: /usr/bin/scp: dest open "/tmp/tempRACTrans_2025_08_22--18-25-44-032/ractrans": No such file or directory/usr/bin/scp…

ADB 调试工具的学习[特殊字符]

一、ADB 的工作原理 1.1 ADB 概念 ADB (Android Debug Bridge)&#xff1a;Android 调试桥&#xff0c;是开发/测试 Android 应用必备的调试工具。作用&#xff1a;通过 电脑终端命令 操作 安卓手机/模拟器。 1.2 ADB 构成与原理 ADB 由三部分组成&#xff1a; Client 端&#…

用一根“数据中枢神经”串起业务从事件流到 Apache Kafka

1. 为什么是“事件流”&#xff1f; 在一个软件定义、自动化、永远在线的世界里&#xff0c;系统之间最需要的是&#xff1a;把发生了什么这件事&#xff0c;第一时间、按正确顺序、可靠地传到该知道的人/系统那里。 事件流就像企业的中枢神经&#xff1a;它把数据库更新、设备…