title author date CreateTime categories
WPF 轻量级 MVVM 框架入门 2.1.2
lindexi
2019-11-29 10:16:10 +0800
2018-06-30 11:18:40 +0800
WPF mvvm

本文告诉大家如何使用本金鱼的 MVVM 轻量框架。
一个好的框架是不需要解释就可以让大家使用,但是本金鱼没有这个能力,所以就写了这个文章告诉大家如何使用。

本文的框架主要是简单,可以快速解耦 ViewModel 和 View 依赖,解耦 ViewModel 之间依赖,减少 ViewModel 的代码量。

具体的思想是使用消息发送的方法,多个 ViewModel 之间使用发送消息解除依赖。把原来很多需要写在 ViewModel 的代码通过发送指定的消息,在其他类处理,减少 ViewModel 代码。

下面来告诉大家如何使用这个框架。

首先是安装库

安装

首先需要从 Nuget 安装两个库

  • lindexi.wpf.Framework

  • lindexi.MVVM.Framework

第一个库是使用 wpf 的封装,因为我还有 UWP 的封装,实际上在使用,用 WPF 或 UWP 是差不多的。只要存在 UWP 和 WPF 不相同的库,我就把这写封装在不同的库。

因为 Nuget 可以找到依赖库,所以只需要安装 lindexi.wpf.Framework 就会自动安装 lindexi.MVVM.Framework 。如果现在使用的是 Xarmain ,那么安装 lindexi.MVVM.Framework 就可以,这个库使用 dotnet framework 4.5 和 dotnet standard 2.0 ,所以在很多项目都可以使用。

项目要求

安装这个库要求最低版本是 dotnet framework 4.5 以上,对于 dotnet framework 4.0 的项目无法使用。

创建主页面

和使用 win10 uwp 轻量级 MVVM 框架入门 2.1.5.3199 差不多,先创建一个项目,然后在这个项目的 MainWindow.xaml 放一个 Frame 作为跳转

创建 ViewModel 类,并且 MainWindow 在使用 ViewModel 属性,在 MainWindow 构造函数调用下面函数

  1. ViewModel.OnNavigatedTo(this, frame);

这里的 frame 就是 MainWindow 创建的 Frame,需要在 ViewModel 类继承 NavigateViewModel ,然后写 OnNavigatedTo 代码

  1. public override void OnNavigatedTo(object source, object e)
  2. // 忽略代码
  3.  
  4. Content = new NavigateFrame((Frame) e);

这里可以看到 NavigateFrame 在 UWP 的框架和在 WPF 框架都存在,但是两个类的实现不相同。

这里设置 Content 可以让之后的页面进行导航。

找到 ViewModel

现在在创建的 ViewModel 推荐在 App.xaml 作为资源,因为 WPF 的页面跳转没有和 UWP 一样可以获得参数,需要通过自己的方式拿到。打开 App.xaml 在里面添加 ViewModel 作为静态属性

  1. <Application x:Class="lindexi.Mvvm.App"
  2. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  3. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  4. xmlns:local="clr-namespace:lindexi.Mvvm"
  5. StartupUri="MainWindow.xaml">
  6. <Application.Resources>
  7. <local:ViewModel x:Key="ViewModel"></local:ViewModel>
  8. </Application.Resources>
  9. </Application>

现在打开 MainWindow.xaml 绑定 DataContext

  1. DataContext="{StaticResource ViewModel}"

在 MainWindow 的构造函数拿到 ViewModel ,需要强转

  1. public MainWindow()
  2. {
  3. InitializeComponent();
  4. ViewModel = (ViewModel) DataContext;
  5. ViewModel.OnNavigatedTo(this, frame);
  6. }
  7.  
  8. public ViewModel ViewModel { get; set; }

这样在被跳转的页面就可以通过 var viewModel = ViewModel["xx"]; 获得 ViewModel 。

通过附加属性找到 ViewModel

第二个方法是通过附加属性的方法找到 ViewModel ,因为在 WPF 是可以定义可继承的附加属性,而在 UWP 是不可以自己定义可以继承的附加属性。在 WPF 可以通过定义附加属性的方式让页面拿到上一级的 ViewModel ,那么两个方法的不同在哪?如果定义为资源,那么使用比较简单。如果定义为附加属性,可以在相同的 ViewModel 被不同的地方使用,支持一个程序有多个可跳转的 ViewModel 可以做出比较复杂的程序。

打开 MainWindow.xaml.cs 定义一个附加属性

  1. public MainWindow()
  2. {
  3. InitializeComponent();
  4.  
  5. ViewModel = (ViewModel) GetValue(ViewModelProperty);
  6. ViewModel.OnNavigatedTo(this, frame);
  7. DataContext = ViewModel;
  8. }
  9.  
  10. public ViewModel ViewModel { get; set; }
  11.  
  12. public static readonly DependencyProperty ViewModelProperty = DependencyProperty.RegisterAttached(
  13. "ViewModel", typeof(NavigateViewModel), typeof(MainWindow),
  14. new FrameworkPropertyMetadata(new ViewModel(), FrameworkPropertyMetadataOptions.Inherits));

这样被跳转的页面就可以通过附加属性拿到值,可以通过 var viewModel = ViewModel["xx"]; 获得

跳转页面

现在来多创建两个页面 A 页面和 B 页面,并且创建两个页面的 ViewModel 添加到 ViewModel 的列表。

  1. public class ViewModel : NavigateViewModel
  2. {
  3. /// <inheritdoc />
  4. public override void OnNavigatedTo(object sender, object obj)
  5. {
  6. ViewModelPage = new List<ViewModelPage>()
  7. {
  8. new ViewModelPage(new NavigatableViewModel<AModel>(), new NavigatablePage<A>()),
  9. new ViewModelPage(new NavigatableViewModel<BModel>(), new NavigatablePage<B>())
  10. };
  11.  
  12. Content = new NavigateFrame((Frame) obj);
  13.  
  14. Navigate("AModel", null);
  15. }
  16.  
  17. }
  18.  
  19. public class AModel : ViewModelMessage
  20. {
  21. /// <inheritdoc />
  22. public override void OnNavigatedFrom(object sender, object obj)
  23. {
  24. }
  25.  
  26. /// <inheritdoc />
  27. public override void OnNavigatedTo(object sender, object obj)
  28. {
  29. }
  30. }
  31.  
  32. public class BModel : ViewModelMessage
  33. {
  34. /// <inheritdoc />
  35. public override void OnNavigatedFrom(object sender, object obj)
  36. {
  37. }
  38.  
  39. /// <inheritdoc />
  40. public override void OnNavigatedTo(object sender, object obj)
  41. {
  42. }
  43. }

添加 ViewModel 到 ViewModelPage 的方法就是通过创建 ViewModelPage 合并,创建的时候提供两个方法,一个是输入 Type 的方法,另一个是泛型。输入 Type 的方法是用来反射,很少推荐使用。

修改 A 页面的背景,可以用来看到被跳转到 A 页面

本文使用的是上面的第一个方式拿到 ViewModel 所以需要在 A 页面添加一些代码获得 ViewModel 。

在 A 页面的 xaml 添加下面代码

  1. DataContext="{Binding Source={StaticResource ViewModel},Path=[AModel]}"

这样就可以通过 NavigateViewModel["xx"] 的方法获得 ViewModel

跳转命令

可以看到在 A 页面有跳转按钮,点击这个按钮可以用来跳转到 B 页面

因为界面很简单,我就不告诉大家了。

打开 AModel 添加一个函数,在这个函数就是按钮点击下去调用的函数

  1. Send(new NavigateMessage(this, nameof(BModel)));

从代码可以看到跳转到 B 页面只需要发送一个 NavigateMessage ,具体怎么跳转是不需要写的

自定义命令

现在可以尝试使用框架的用法,自己定义消息和处理

在 B 页面发送文字到 MainWindow 显示,不让 BModel 和 ViewModel 有耦合,减少在 BModel 和在 ViewModel 的代码。

如果 ViewModel 要让 MainWindows 显示文字,就需要在 ViewModel 添加属性

  1. public string SawbelChaceredis
  2. {
  3. get => _sawbelChaceredis;
  4. set
  5. {
  6. _sawbelChaceredis = value;
  7. OnPropertyChanged();
  8. }
  9. }
  10.  
  11. private string _sawbelChaceredis;

这个属性是我随意定义,因为也不知道这个属性叫什么

在 ViewModel 有特殊的属性,可以定义一个接口,表示这个 ViewModel 有 属性 SawbelChaceredis ,定义的代码

  1. public interface ISawbelChaceredisModel : IViewModel
  2. {
  3. string SawbelChaceredis { get; set; }
  4. }

在 MainWindow 绑定这个属性

  1. <TextBlock Margin="10,200,10,10" Text="{Binding SawbelChaceredis,Mode=OneWay}" />

已经让 ViewModel 绑定了页面,这时的 ViewModel 和界面没有耦合

定义消息用来发送字符串到 ISawbelChaceredisModel ,定义的方法请看代码

  1. public class TextMessage : Message
  2. {
  3. /// <inheritdoc />
  4. public TextMessage(ViewModelBase source, string str) : base(source)
  5. {
  6. SawbelChaceredis = str;
  7. Goal = new PredicateInheritViewModel(typeof(ISawbelChaceredisModel));
  8. }
  9.  
  10. public string SawbelChaceredis { get; }
  11. }

定义的消息使用了 PredicateInheritViewModel 表示这个消息会发送到 ISawbelChaceredisModel ,如果没有找到 ISawbelChaceredisModel 就没有处理消息

定义了消息还需要定一个类告诉如何处理这个消息

  1. public class SawbelChaceredisComposite : Composite
  2. {
  3. /// <inheritdoc />
  4. public SawbelChaceredisComposite()
  5. {
  6. Message = typeof(TextMessage);
  7. }
  8.  
  9. /// <inheritdoc />
  10. public override void Run(IViewModel source, IMessage message)
  11. {
  12. var viewModel = (ISawbelChaceredisModel) source;
  13. viewModel.SawbelChaceredis = ((TextMessage) message).SawbelChaceredis;
  14. }
  15. }

定义了这个类就告诉哪个消息可以处理,在构造函数的代码。然后使用 Run 处理这个消息。因为 SawbelChaceredisComposite 和消息是耦合的,只有处理知道消息,所以这里可以转换 ViewModel 因为这时知道是哪个收到

这里的处理还可以继承Composite<T> 这里的泛型就是消息的类型,可以把上面的代码减小,不需要写构造函数

  1. public class SawbelChaceredisComposite : Composite<TextMessage>
  2. {
  3. /// <inheritdoc />
  4. protected override void Run(IViewModel source, TextMessage message)
  5. {
  6. var viewModel = (ISawbelChaceredisModel) source;
  7. viewModel.SawbelChaceredis = message.SawbelChaceredis;
  8. }
  9. }

在 B 页面点击时,发送消息

  1. public class BModel : ViewModelMessage
  2. {
  3. /// <inheritdoc />
  4. public override void OnNavigatedFrom(object sender, object obj)
  5. {
  6. }
  7.  
  8. /// <inheritdoc />
  9. public override void OnNavigatedTo(object sender, object obj)
  10. {
  11. }
  12.  
  13. public void SendText()
  14. {
  15. Send(new TextMessage(this, "欢迎来我博客 https://lindexi.github.io/lindexi/ 有很多无聊博客"));
  16. }
  17. }

通过发送消息就可以把消息发送到界面

代码:WPF 轻量级 MVVM 框架入门 2.1.2-CSDN下载

参见:

win10 uwp MVVM入门

win10 uwp MVVM 轻量框架

win10 uwp MVVM 语义耦合

2019-11-29-WPF-轻量级-MVVM-框架入门-2.1.2的更多相关文章

  1. Farseer.net轻量级开源框架 入门篇:使用前说明

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 框架性能测试 下一篇:Farseer.net轻量级开源框架 入门篇: 增.删.改. ...

  2. Farseer.net轻量级开源框架 入门篇:逻辑层的选择

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 入门篇:增.删.改.查操作演示 下一篇:Farseer.net轻量级开源框架 入门 ...

  3. Farseer.net轻量级开源框架 入门篇:分类逻辑层

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 缓存逻辑层 下一篇:Farseer.net轻量级开源框架 入门篇: 添加数据详解 ...

  4. Farseer.net轻量级开源框架 入门篇:添加数据详解

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 分类逻辑层 下一篇:Farseer.net轻量级开源框架 入门篇: 修改数据详解 ...

  5. Farseer.net轻量级开源框架 入门篇:修改数据详解

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 添加数据详解 下一篇:Farseer.net轻量级开源框架 入门篇: 删除数据详解 ...

  6. Farseer.net轻量级开源框架 入门篇:删除数据详解

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 修改数据详解 下一篇:Farseer.net轻量级开源框架 入门篇: 查询数据详解 ...

  7. Farseer.net轻量级开源框架 入门篇:查询数据详解

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 删除数据详解 下一篇:Farseer.net轻量级开源框架 中级篇: Where条 ...

  8. Farseer.net轻量级开源框架 入门篇:Where条件的终极使用

    导航 目   录:Farseer.net轻量级开源框架 目录 上一篇:Farseer.net轻量级开源框架 入门篇: 查询数据详解 下一篇:Farseer.net轻量级开源框架 中级篇: 事务的使用 ...

  9. 在WPF的MVVM框架中获取下拉选择列表中的选中项

    文章概述: 本演示介绍怎样在WPF的MVVM框架中.通过数据绑定的方式获取下拉列表中的选中项.程序执行后的效果例如以下图所看到的: 相关下载(代码.屏幕录像):http://pan.baidu.com ...

  10. 轻量级MVVM框架 Stylet

    这两天试了下Stylet框架,这个框架虽然很小,但是功能齐全,简化了很多MVVM的代码,比如Command,对Dialog,MessageBox都有很好的支持. 开源地址 https://github ...

随机推荐

  1. python3笔记七:break和continue语句

    一:学习内容 break语句 continue语句 二:break语句 1. 说明 作用:跳出for和while的循环注意:只能跳出距离它最近的那一层循环 2.举例1 for i in range(1 ...

  2. Java-线程等待、唤醒与中断

    一.sleep() 与 wait() 两者都会让当前线程进入等待状态.唤醒后都需要等待 CPU 资源,不一定会立即执行.若在等待期间被调用此线程的的 interrupt() 方法,将会产生 Inter ...

  3. C# 查看计算机端口使用状态

    using System.Net.NetworkInformation; /// <summary> /// 获取第一个可用的端口号 /// </summary> /// &l ...

  4. 全面解读PHP-数据库缓存

    一.什么是数据库缓存? 1.定义 mysql等一些常见的关系型数据库的数据都存储在磁盘当中,在高并发场景下,业务应用对mysql产生的增删改查的操作会造成巨大的IO开销和查询压力,这无疑对数据库和服务 ...

  5. golang tcp keepalive实践

    前文中已经介绍了TCP keep alive的做了详尽说明,本文结合golang,介绍如何使用TCP keep alive. 目前golang net包不提供TCP keep alive 空闲多长时间 ...

  6. leetcode 658找到k个最接近的元素

    class Solution { public: vector<int> findClosestElements(vector<int>& arr, int k, in ...

  7. 使用ajax获取servelt数据乱码

    修改tomcat编码 <Connector port="8080" protocol="HTTP/1.1" connectionTimeout=" ...

  8. nohup及pip命令总结

    最近在搭建Python的Web开发环境的时候,用到nohup和pip等一些工具,先简单总结一下,以备后续查用. 1.nohup nohup(no hang up)就是不挂断的意思,如果你正在运行一个进 ...

  9. HTML基础之DOM

    DOM(Document Object Model 文档对象模型) 一个web页面的展示,是由html标签组合成的一个页面,js是一门语言,dom对象实际就是将html标签转换成了一个文档对象.可以通 ...

  10. web开发(六) EL表达式

    在网上看见一篇不错的文章,写的详细. 以下内容引用那篇博文.转载于<http://www.cnblogs.com/whgk/p/6432044.html>,在此仅供学习参考之用. 一.EL ...