在WPF自學入門(十)WPF MVVM簡單介紹中的示例似乎運行起來沒有什麼問題,也可以進行更新。但是這並不是我們使用MVVM的正確方式。正如上一篇文章中在開始說的,MVVM的目的是為了最大限度地降低了Xaml文件和CS文件的耦合度,分離界面和業務邏輯,所以我們要儘可能的在View後臺不寫代碼。但是這 ...
在WPF自學入門(十)WPF MVVM簡單介紹中的示例似乎運行起來沒有什麼問題,也可以進行更新。但是這並不是我們使用MVVM的正確方式。正如上一篇文章中在開始說的,MVVM的目的是為了最大限度地降低了Xaml文件和CS文件的耦合度,分離界面和業務邏輯,所以我們要儘可能的在View後臺不寫代碼。但是這個例子中,我們將更新ViewModel的代碼寫在了View里,下一個例子中,我們要通過命令(Command)的來將Button的事件分離出來。
因為本文中需要使用Command命令,我們先來簡單瞭解Command命令。在WPF中使用命令的步驟很簡單
WPF中命令的核心是System.Windows.Input.ICommand介面,所有命令對象都實現了此介面。當創建自己的命令時,不能直接實現ICommand介面,而是要使用System.Windows.Input.RouteCommand類,該類已經實現了ICommand介面,所有WPF命令都是RouteCommand類的實例。在程式中處理的大部分命令不是RoutedCommand對象,而是RoutedUICommand類的實例,它繼承自RouteCommand類。
WPF提供了一個很好的方式來解決事件綁定的問題--ICommand。很多控制項都有Command屬性,如果沒有,我們可以將命令綁定到觸發器上。接下來我們來先實現一個ICommand介面。ICommand需要用戶定義兩個方法bool CanExecute和void Execute。第一個方法可以讓我們來判斷是否可以執行這個命令,第二個方法就是我們具體的命令。
1 using System; 2 3 using System.Collections.Generic; 4 5 using System.Linq; 6 7 using System.Text; 8 9 using System.Windows.Input; 10 11 12 13 /***********************作者:黃昏前黎明後********************************** 14 15 * 作者:黃昏前黎明後 16 17 * CLR版本:4.0.30319.42000 18 19 * 創建時間:2018-04-05 22:57:56 20 21 * 命名空間:Example3 22 23 * 唯一標識:b9043d4c-fdd7-4e0f-a324-00f0f09286d0 24 25 * 機器名稱:HLPC 26 27 * 聯繫人郵箱:[email protected] 28 29 * 30 31 * 描述說明: 32 33 * 34 35 * 修改歷史: 36 37 * 38 39 * 40 41 *****************************************************************/ 42 43 namespace Example3 44 45 { 46 47 public class RelayCommand : ICommand 48 49 { 50 51 #region 欄位 52 53 readonly Func<Boolean> _canExecute; 54 55 readonly Action _execute; 56 57 #endregion 58 59 60 61 #region 構造函數 62 63 public RelayCommand(Action execute) 64 65 : this(execute, null) 66 67 { 68 69 } 70 71 public RelayCommand(Action execute, Func<Boolean> canExecute) 72 73 { 74 75 if (execute == null) 76 77 throw new ArgumentNullException("execute"); 78 79 _execute = execute; 80 81 _canExecute = canExecute; 82 83 } 84 85 #endregion 86 87 88 89 #region ICommand的成員 90 91 public event EventHandler CanExecuteChanged 92 93 { 94 95 add 96 97 { 98 99 100 101 if (_canExecute != null) 102 103 CommandManager.RequerySuggested += value; 104 105 } 106 107 remove 108 109 { 110 111 112 113 if (_canExecute != null) 114 115 CommandManager.RequerySuggested -= value; 116 117 } 118 119 } 120 121 122 123 [DebuggerStepThrough] 124 125 public Boolean CanExecute(Object parameter) 126 127 { 128 129 return _canExecute == null ? true : _canExecute(); 130 131 } 132 133 134 135 public void Execute(Object parameter) 136 137 { 138 139 _execute(); 140 141 } 142 143 #endregion 144 145 } 146 147 } 148 149
我們再在我們的NameViewModel中聲明一個ICommand欄位:
1 #region 命令 2 3 void UpdateNameExecute() 4 5 { 6 7 this.UserName = "黃昏前黎明後"; 8 9 this.CompanyName = "中軟易通科技"; 10 11 } 12 13 14 15 bool CanUpdateNameExecute() 16 17 { 18 19 return true; 20 21 } 22 23 24 25 public ICommand UpdateName { get { return new RelayCommand(UpdateNameExecute, CanUpdateNameExecute); } } 26 27 28 29 #endregion
<Button Content="更新" Command="{Binding UpdateName}" Margin="20"/>
看到上面的結果,似乎目前為止我們已經很好的解決了所有的問題。我們看到運行的數據,事件都是綁定的,實現了界面的完美分離。實際在處理問題是好像需要考慮通用性,這時我們能否把MVVM提取出來作為一個框架,來去更好的解決問題。下一次我們一起來看看怎麼進行提取成為通用框架。
https://pan.baidu.com/s/1xZvsrMbDOXlpvDbCh2Af3Q 密碼:6666