首页 > 代码库 > 利刃 MVVMLight 6:命令基础

利刃 MVVMLight 6:命令基础

    在MVVM Light框架中,事件是WPF应用程序中UI与后台代码进行交互的最主要方式,与传统方式不同,mvvm中主要通过绑定到命令来进行事件的处理,

因此要了解mvvm中处理事件的方式,就必须先熟悉命令的工作原理。

RelayCommand命令:
    WPF命令是通过实现 ICommand 接口创建的。 ICommand 公开了两个方法(Execute 及 CanExecute)和一个事件(CanExecuteChanged)。

Execute方法执行与命令关联的操作
CanExecute方法 确定是否可以在当前命令目标上执行命令,返回值为true则按钮可用,为false的时候按钮disable。在MvvmLight中实现ICommand接口的类是RelayCommand

 

 

RelayCommand通过构造函数初始化Execute 和 CanExecute方法,因此,构造函数传入的是委托类型的参数,

Execute 和 CanExecute则执行的是委托的方法。如图:

技术分享

技术分享

 

相对于CodeBehind 的方式,使用命令会好很多:

最大的特点就是解耦View和ViewModel的行为交互,将视图的显示和业务逻辑分开。对View上的某个元素进行命令的绑定,触发点击操作的时候,这个按钮实际完成
的是对应ViewModel中的所绑定的方法的执行。这里我们用到Mvvm框架中的RelayCommand。

现在我们来看一个例子,将我们上篇的那个例子改装一下,加进CanExcute()方法和列表数据的呈现。

Model代码:

 1  [MetadataType(typeof(BindDataAnnotationsViewModel))] 2     public class ValidateUserInfo:ValidateModelBase 3     { 4         #region 属性  5         private String userName; 6         /// <summary> 7         /// 用户名 8         /// </summary> 9         [Required]10         public String UserName11         {12             get { return userName; }13             set { userName = value; RaisePropertyChanged(() => UserName); }14         }15 16 17 18         private String userPhone;19         /// <summary>20         /// 用户电话21         /// </summary>22         [Required]23         [RegularExpression(@"^[-]?[1-9]{8,11}\d*$|^[0]{1}$", ErrorMessage = "用户电话必须为8-11位的数值.")]24         public String UserPhone25         {26             get { return userPhone; }27             set { userPhone = value; RaisePropertyChanged(() => UserPhone); }28         }29 30 31 32         private String userEmail;33         /// <summary>34         /// 用户邮件35         /// </summary>36         [Required]37         [StringLength(100, MinimumLength = 2)]38         [RegularExpression("^\\s*([A-Za-z0-9_-]+(\\.\\w+)*@(\\w+\\.)+\\w{2,5})\\s*$", ErrorMessage = "请填写正确的邮箱地址.")]39         public String UserEmail40         {41             get { return userEmail; }42             set { userEmail = value; RaisePropertyChanged(() => UserEmail);  }43         }44         #endregion

 View代码:

提交按钮绑定了一个Command,这个Command指向对用的ViewModel中的SubmitCmd 方法。这样确实很赞,SubmitCmd 独立性、复用性很高。

 1 <Window x:Class="MVVMLightDemo.View.CommandView" 2         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 3         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" 4         DataContext="{Binding Source={StaticResource Locator},Path=Command}" 5         Title="CommandView" Height="500" Width="800"> 6     <Grid> 7         <StackPanel Orientation="Vertical" > 8             <GroupBox Header="命令" Margin="10 20 10 10" >                 9                 <StackPanel Orientation="Vertical" Margin="0,10,0,0">10                     <StackPanel.Resources>11                         <Style TargetType="StackPanel">12                             <Setter Property="Orientation" Value=http://www.mamicode.com/"Horizontal" />13                             <Setter Property="Margin" Value=http://www.mamicode.com/"0,0,0,4" />14                         </Style>15                         <Style TargetType="Label" BasedOn="{StaticResource {x:Type Label}}">16                             <Setter Property="Width" Value=http://www.mamicode.com/"100" />17                             <Setter Property="VerticalAlignment" Value=http://www.mamicode.com/"Center" />18                         </Style>19                         <Style TargetType="CheckBox" BasedOn="{StaticResource {x:Type CheckBox}}">20                             <Setter Property="Padding" Value=http://www.mamicode.com/"0,3" />21                         </Style>22                         <Style TargetType="RadioButton" BasedOn="{StaticResource {x:Type RadioButton}}">23                             <Setter Property="Padding" Value=http://www.mamicode.com/"0,3" />24                         </Style>25                     </StackPanel.Resources>26 27                     <StackPanel>28                         <Label Content="用户名" Target="{Binding ElementName=UserName}"/>29                         <TextBox Width="150" 30                                  Text="{Binding ValidateUI.UserName,UpdateSourceTrigger=PropertyChanged,ValidatesOnDataErrors=True}" >31                         </TextBox>32                     </StackPanel>33                     <StackPanel>34                         <Label Content="用户邮箱" Target="{Binding ElementName=UserEmail}"/>35                         <TextBox Width="150" Text="{Binding ValidateUI.UserEmail, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}" />36                     </StackPanel>37                     <StackPanel>38                         <Label Content="用户电话" Target="{Binding ElementName=UserPhone}"/>39                         <TextBox Width="150" Text="{Binding ValidateUI.UserPhone,UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}" />40                     </StackPanel>41                     <StackPanel>42                         <Label Foreground="Red" Content="提示:验证全部通过的时候提交按钮可操作!" Width="250"  ></Label>43                     </StackPanel>                    44                     <Button Content="提交" Margin="100,16,0,0" HorizontalAlignment="Left" Command="{Binding SubmitCmd}" />45                 </StackPanel>46             </GroupBox>            47            48             <StackPanel>49                 <DataGrid x:Name="dg1" ItemsSource="{Binding List}" AutoGenerateColumns="False" CanUserAddRows="False" 50                                       CanUserSortColumns="False" Margin="10" AllowDrop="True" IsReadOnly="True" >51                     <DataGrid.Columns>52                         <DataGridTextColumn Header="用户姓名" Binding="{Binding UserName}" Width="100" />53                         <DataGridTextColumn Header="邮箱"  Binding="{Binding UserEmail}" Width="400" />54                         <DataGridTextColumn Header="电话" Binding="{Binding UserPhone}" Width="100" />55                     </DataGrid.Columns>56                 </DataGrid>57 58             </StackPanel>59 60         </StackPanel>61       62     </Grid>63 </Window>

 ViewModel代码:

这边需要注意的是:用户在界面上点击提交按钮的时候去ViewModel 里面寻找名为SubmitCmd的 RelayCommand命令对象,如果找不到,则执行无效果,所以名称一定要对应上,而且需要是公开的访问级别。

CanExcute方法这边用表单是否验证通过来判断命令是否执行,如果返回的是false,则该命令不执行,这时候提交按钮也是不可用(Disable)的。

 1 using GalaSoft.MvvmLight; 2 using GalaSoft.MvvmLight.CommandWpf; 3 using MVVMLightDemo.Model; 4 using System.Collections.ObjectModel; 5  6 namespace MVVMLightDemo.ViewModel 7 { 8     public class CommandViewModel:ViewModelBase 9     {10         public CommandViewModel()11         {12             //构造函数13             ValidateUI = new ValidateUserInfo();14             List = new ObservableCollection<ValidateUserInfo>();15         }16 17         #region 全局属性18         private ObservableCollection<ValidateUserInfo> list;19         /// <summary>20         /// 用户数据列表21         /// </summary>22         public ObservableCollection<ValidateUserInfo> List23         {24             get { return list; }25             set { list = value; }26         }27 28         private ValidateUserInfo validateUI;29         /// <summary>30         /// 当前操作的用户信息31         /// </summary>32         public ValidateUserInfo ValidateUI33         {34             get { return validateUI; }35             set36             {37                 validateUI = value;38                 RaisePropertyChanged(() => ValidateUI);39             }40         }41         #endregion42 43         #region 全局命令44         private RelayCommand submitCmd;45         /// <summary>46         /// 执行提交命令的方法47         /// </summary>48         public RelayCommand SubmitCmd49         {50             get51             {52                 if (submitCmd == null) return new RelayCommand(() => ExcuteValidForm(),CanExcute);53                 return submitCmd;54             }55             set { submitCmd = value; }56         }57         #endregion58 59         #region 附属方法60         /// <summary>61         /// 执行提交方法62         /// </summary>63         private void ExcuteValidForm()64         {65             List.Add(new ValidateUserInfo(){ UserEmail= ValidateUI.UserEmail, UserName = ValidateUI.UserName, UserPhone = ValidateUI.UserPhone });66         }67 68         /// <summary>69         /// 是否可执行(这边用表单是否验证通过来判断命令是否执行)70         /// </summary>71         /// <returns></returns>72         private bool CanExcute()73         {74             return ValidateUI.IsValidated;75         }76         #endregion77 78     }79 }

 结果如下:

技术分享

 这是最简单的命令操作了,下篇我们来深入了解下命令和EventToCommand的相关内容。

示例代码下载

转载请标明出处,谢谢

利刃 MVVMLight 6:命令基础