1、概述html
有時候,單純的命令綁定不必定能知足咱們的開發需求,好比咱們須要在命令綁定的時候傳遞一個參數,這個時候,咱們就須要使用RelayCommand的泛型版本了。express
RelayCommand的泛型版本的構造函數如下:mvvm
public RelayCommand(Action<T> execute, bool keepTargetAlive = false);
public RelayCommand(Action<T> execute, Func<T, bool> canExecute, bool keepTargetAlive = false);ide
構造函數傳入的是委託類型的參數,Execute 和 CanExecute執行委託方法。函數
2、帶一個參數的命令綁定spa
代碼片斷以下:code
<StackPanel> <GroupBox Header="帶string類型參數的命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FFCDAA0C" Margin="2"> <StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserList:" VerticalContentAlignment="Center" FontSize="20" ></Label> <Label Content="{Binding Path=UserList,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}" FontSize="20" /> </StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserName:" VerticalContentAlignment="Center" FontSize="20" ></Label> <TextBox Width="200" Name="tbUser"></TextBox> <Button Content="AddUser" Command="{Binding AddUserCommand}" CommandParameter="{Binding ElementName=tbUser,Path=Text}"></Button> <CheckBox Content="IsCanAdd" VerticalAlignment="Center" FontSize="16" IsChecked="{Binding IsCanAddUser, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}"></CheckBox> </StackPanel> </StackPanel> </GroupBox> </StackPanel>
private RelayCommand<string> addUserCommand; public RelayCommand<string> AddUserCommand { get { if (addUserCommand == null) { addUserCommand = new RelayCommand<string>(AddUser, (string p) => { return IsCanAddUser; }); } return addUserCommand; } set { addUserCommand = value; } } private void AddUser(string par) { UserList = UserList + " " + par; }
3、帶多個參數的命令綁定component
給命令傳遞多個參數,建議使用如下方式:orm
使用MultiBinding將多綁定的各個值轉換成咱們所需的對象或者實例模型,再傳遞給ViewModel中的命令。xml
代碼片斷以下:
<Window x:Class="MvvmLightDemo1.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:local="clr-namespace:MvvmLightDemo1" xmlns:cvt="clr-namespace:MvvmLightDemo1.Converter" xmlns:mvvm="http://www.galasoft.ch/mvvmlight" xmlns:i="clr-namespace:System.Windows.Interactivity;assembly=System.Windows.Interactivity" mc:Ignorable="d" Title="MVVMLIghtDemo1" Height="500" Width="700" > <Window.Resources> <cvt:UserInfoConverter x:Key="userInfoConverter"></cvt:UserInfoConverter> </Window.Resources> <Window.DataContext> <Binding Path="Main" Source="{StaticResource Locator}"></Binding> </Window.DataContext> <StackPanel> <StackPanel> <GroupBox Header="帶string類型參數的命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FFCDAA0C" Margin="2"> <StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserList:" VerticalContentAlignment="Center" FontSize="20" ></Label> <Label Content="{Binding Path=UserList,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}" FontSize="20" /> </StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserName:" VerticalContentAlignment="Center" FontSize="20" ></Label> <TextBox Width="200" Name="tbUser"></TextBox> <Button Content="AddUser" Command="{Binding AddUserCommand}" CommandParameter="{Binding ElementName=tbUser,Path=Text}"></Button> <CheckBox Content="IsCanAdd" VerticalAlignment="Center" FontSize="16" IsChecked="{Binding IsCanAddUser, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}"></CheckBox> </StackPanel> </StackPanel> </GroupBox> </StackPanel> <StackPanel> <GroupBox Header="帶對象類型參數的命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FF127C0D" Margin="2"> <StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserName:" FontSize="16" ></Label> <TextBox Width="200" Name="tbxUser" FontSize="16" /> <Label Content="Password:" FontSize="16" ></Label> <TextBox Width="200" Name="tbxPwd" FontSize="16" /> <Button Content="AddUser" Command="{Binding AddUserCommandWithObjPar}"> <Button.CommandParameter> <MultiBinding Converter="{StaticResource userInfoConverter}"> <Binding ElementName="tbxUser" Path="Text"/> <Binding ElementName="tbxPwd" Path="Text"/> </MultiBinding> </Button.CommandParameter> </Button> </StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="Parameter:" FontSize="16" ></Label> <Label Content="{Binding ObjParameter}" FontSize="16" ></Label> </StackPanel> </StackPanel> </GroupBox> </StackPanel> <StackPanel> <GroupBox Header="事件轉命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FFCDAA0C" Margin="2"> <StackPanel> <StackPanel> <ListBox x:Name="lb" ItemsSource="{Binding ListBoxData}" BorderThickness="0" SelectedIndex="{Binding SelectIndex}" > <i:Interaction.Triggers> <i:EventTrigger EventName="SelectionChanged"> <mvvm:EventToCommand Command="{Binding SelectionChangedCommand}"/> </i:EventTrigger> </i:Interaction.Triggers> <ListBox.ItemsPanel> <ItemsPanelTemplate> <WrapPanel Width="{Binding ActualWidth,RelativeSource={RelativeSource AncestorType={x:Type ListBox}}}"/> </ItemsPanelTemplate> </ListBox.ItemsPanel> <ListBox.ItemTemplate> <DataTemplate> <Border BorderBrush="AntiqueWhite" BorderThickness="1"> <StackPanel Margin="2"> <Image Source="{Binding Img}" Width="96" Height="96"/> <TextBlock HorizontalAlignment="Center" Text="{Binding Info}"/> </StackPanel> </Border> </DataTemplate> </ListBox.ItemTemplate> </ListBox> <Label Content="您選擇的是:" FontSize="16" ></Label> <Label Content="{Binding Path=SelResult,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}" FontSize="16" /> </StackPanel> </StackPanel> </GroupBox> </StackPanel> </StackPanel> </Window>
1 using GalaSoft.MvvmLight; 2 3 namespace MvvmLightDemo1.ViewModel 4 { 5 public class UserModel: ObservableObject 6 { 7 private string userName; 8 9 public string UserName 10 { 11 get { return userName; } 12 set 13 { 14 userName = value; 15 RaisePropertyChanged(); 16 } 17 } 18 19 private string passWord; 20 21 public string PassWord 22 { 23 get { return passWord; } 24 set 25 { 26 passWord = value; 27 RaisePropertyChanged(); 28 } 29 } 30 31 32 } 33 }
using MvvmLightDemo1.ViewModel; using System; using System.Linq; using System.Windows.Data; namespace MvvmLightDemo1.Converter { public class UserInfoConverter: IMultiValueConverter { public object Convert(object[] values, Type targetType, object parameter, System.Globalization.CultureInfo culture) { if (!values.Cast<string>().Any(text => string.IsNullOrEmpty(text)) && values.Count() == 2) { UserModel userModel = new UserModel() { UserName = values[0].ToString(), PassWord = values[1].ToString()}; return userModel; } return null; } public object[] ConvertBack(object value, Type[] targetTypes, object parameter, System.Globalization.CultureInfo culture) { throw new NotImplementedException(); } } }
private RelayCommand<UserModel> addUserCommandWithObjPar; public RelayCommand<UserModel> AddUserCommandWithObjPar { get { if (addUserCommandWithObjPar == null) { addUserCommandWithObjPar = new RelayCommand<UserModel>(AddUserWithObjPar); } return addUserCommandWithObjPar; } set { addUserCommandWithObjPar = value; } } private void AddUserWithObjPar(UserModel par) { ObjParameter = "UserName: "+ par.UserName + " Password: " + par.PassWord; }
後記:
從MVVM的模式來講,其實命令中的參數傳遞未必是必要的。與其維護一段額外的參數代碼,還不如把全部的交互參數細化成爲當前DataContext下的全局屬性。View開發人員和ViewModel開發人員共同維護好這份命令清單和屬性清單便可。
4、EventToCommand
在WPF中,並非全部控件都有Command,例如TextBox,那麼當文本改變,咱們須要處理一些邏輯,這些邏輯在ViewModel中,沒有Command如何綁定呢?
這個時候咱們就用到EventToCommand,事件轉命令,能夠將一些事件例如TextChanged,Checked等事件轉換成命令的方式。
接下來咱們就以ListBox爲例子,來看看具體的實例:
View代碼:(這邊聲明瞭i特性和mvvm特性,一個是爲了擁有觸發器和行爲附加屬性的能力,當事件觸發時,會去調用相應的命令,EventName表明觸發的事件名稱;一個是爲了使用MVVMLight中 EventToCommand功能。)
這邊就是當ListBox執行SelectionChanged事件的時候,會相應去執行ViewModel中 SelectionChangedCommand命令。
代碼片斷以下:
<Window x:Class="MvvmLightDemo1.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:local="clr-namespace:MvvmLightDemo1" xmlns:cvt="clr-namespace:MvvmLightDemo1.Converter" xmlns:mvvm="http://www.galasoft.ch/mvvmlight" xmlns:i="clr-namespace:System.Windows.Interactivity;assembly=System.Windows.Interactivity" mc:Ignorable="d" Title="MVVMLIghtDemo1" Height="500" Width="700" > <Window.Resources> <cvt:UserInfoConverter x:Key="userInfoConverter"></cvt:UserInfoConverter> </Window.Resources> <Window.DataContext> <Binding Path="Main" Source="{StaticResource Locator}"></Binding> </Window.DataContext> <StackPanel> <StackPanel> <GroupBox Header="帶string類型參數的命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FFCDAA0C" Margin="2"> <StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserList:" VerticalContentAlignment="Center" FontSize="20" ></Label> <Label Content="{Binding Path=UserList,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}" FontSize="20" /> </StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserName:" VerticalContentAlignment="Center" FontSize="20" ></Label> <TextBox Width="200" Name="tbUser"></TextBox> <Button Content="AddUser" Command="{Binding AddUserCommand}" CommandParameter="{Binding ElementName=tbUser,Path=Text}"></Button> <CheckBox Content="IsCanAdd" VerticalAlignment="Center" FontSize="16" IsChecked="{Binding IsCanAddUser, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}"></CheckBox> </StackPanel> </StackPanel> </GroupBox> </StackPanel> <StackPanel> <GroupBox Header="帶對象類型參數的命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FF127C0D" Margin="2"> <StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="UserName:" FontSize="16" ></Label> <TextBox Width="200" Name="tbxUser" FontSize="16" /> <Label Content="Password:" FontSize="16" ></Label> <TextBox Width="200" Name="tbxPwd" FontSize="16" /> <Button Content="AddUser" Command="{Binding AddUserCommandWithObjPar}"> <Button.CommandParameter> <MultiBinding Converter="{StaticResource userInfoConverter}"> <Binding ElementName="tbxUser" Path="Text"/> <Binding ElementName="tbxPwd" Path="Text"/> </MultiBinding> </Button.CommandParameter> </Button> </StackPanel> <StackPanel Orientation="Horizontal"> <Label Content="Parameter:" FontSize="16" ></Label> <Label Content="{Binding ObjParameter}" FontSize="16" ></Label> </StackPanel> </StackPanel> </GroupBox> </StackPanel> <StackPanel> <GroupBox Header="事件轉命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FFCDAA0C" Margin="2"> <StackPanel> <StackPanel> <ListBox x:Name="lb" ItemsSource="{Binding ListBoxData}" BorderThickness="0" SelectedIndex="{Binding SelectIndex}" > <i:Interaction.Triggers> <i:EventTrigger EventName="SelectionChanged"> <mvvm:EventToCommand Command="{Binding SelectionChangedCommand}"/> </i:EventTrigger> </i:Interaction.Triggers> <ListBox.ItemsPanel> <ItemsPanelTemplate> <WrapPanel Width="{Binding ActualWidth,RelativeSource={RelativeSource AncestorType={x:Type ListBox}}}"/> </ItemsPanelTemplate> </ListBox.ItemsPanel> <ListBox.ItemTemplate> <DataTemplate> <Border BorderBrush="AntiqueWhite" BorderThickness="1"> <StackPanel Margin="2"> <Image Source="{Binding Img}" Width="96" Height="96"/> <TextBlock HorizontalAlignment="Center" Text="{Binding Info}"/> </StackPanel> </Border> </DataTemplate> </ListBox.ItemTemplate> </ListBox> <Label Content="您選擇的是:" FontSize="16" ></Label> <Label Content="{Binding Path=SelResult,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}" FontSize="16" /> </StackPanel> </StackPanel> </GroupBox> </StackPanel> </StackPanel> </Window>
using GalaSoft.MvvmLight; using GalaSoft.MvvmLight.CommandWpf; using MvvmLightDemo1.Model; using System.Collections; using System.Collections.ObjectModel; using System.Windows.Input; namespace MvvmLightDemo1.ViewModel { public class MainViewModel : ViewModelBase { private WelcomeModel welcomeModel; public WelcomeModel WelcomeModel { get { return welcomeModel; } set { welcomeModel = value; RaisePropertyChanged(() => WelcomeModel); } } private string objParameter = ""; public string ObjParameter { get { return objParameter; } set { objParameter = value; RaisePropertyChanged(); } } private int selectIndex = -1; public int SelectIndex { get { return selectIndex; } set { selectIndex = value; RaisePropertyChanged(); } } private string selResult = ""; public string SelResult { get { return selResult; } set { selResult = value; RaisePropertyChanged(); } } private IEnumerable listBoxData = new ObservableCollection<DataModel>() { new DataModel(){ Img="/MvvmLightDemo1;component/Img/1.png",Info=" Honey Peach " }, new DataModel(){ Img="/MvvmLightDemo1;component/Img/2.png",Info="Tomato" }, new DataModel(){ Img="/MvvmLightDemo1;component/Img/3.png",Info="Banana" }, new DataModel(){ Img="/MvvmLightDemo1;component/Img/4.png",Info="Chilli " }, new DataModel(){ Img="/MvvmLightDemo1;component/Img/5.png",Info="Apple" }, }; /// <summary> /// LisBox數據模板 /// </summary> public IEnumerable ListBoxData { get { return listBoxData; } set { listBoxData = value; RaisePropertyChanged(() => ListBoxData); } } /// <summary> /// Initializes a new instance of the MainViewModel class. /// </summary> public MainViewModel() { WelcomeModel = new WelcomeModel() { WelcomeMsg = "Welcome to MVVMLight World!" }; } private string userList = "Mary"; public string UserList { get { return userList; } set { userList = value; RaisePropertyChanged(); } } private string user = ""; public string User { get { return user; } set { user = value; } } private bool isCanAddUser = true; public bool IsCanAddUser { get { return isCanAddUser; } set { isCanAddUser = value; } } #region Command private RelayCommand<string> addUserCommand; public RelayCommand<string> AddUserCommand { get { if (addUserCommand == null) { addUserCommand = new RelayCommand<string>(AddUser, (string p) => { return IsCanAddUser; }); } return addUserCommand; } set { addUserCommand = value; } } private void AddUser(string par) { UserList = UserList + " " + par; } private RelayCommand<UserModel> addUserCommandWithObjPar; public RelayCommand<UserModel> AddUserCommandWithObjPar { get { if (addUserCommandWithObjPar == null) { addUserCommandWithObjPar = new RelayCommand<UserModel>(AddUserWithObjPar); } return addUserCommandWithObjPar; } set { addUserCommandWithObjPar = value; } } private void AddUserWithObjPar(UserModel par) { ObjParameter = "UserName: "+ par.UserName + " Password: " + par.PassWord; } private RelayCommand selectionChangedCommand; public RelayCommand SelectionChangedCommand { get { if (selectionChangedCommand == null) { selectionChangedCommand = new RelayCommand(SelectChange); } return selectionChangedCommand; } set { selectionChangedCommand = value; } } private void SelectChange() { int i = 0; foreach(var a in ListBoxData) { if(i == SelectIndex) { SelResult = (a as DataModel).Info; break; } i++; } } #endregion } }
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace MvvmLightDemo1.Model { public class DataModel { private string img; public string Img { get { return img; } set { img = value; } } private string info; public string Info { get { return info; } set { info = value; } } } }
運行結果以下:
5、帶原事件參數的命令綁定
代碼片斷以下:
<StackPanel> <GroupBox Header="帶事件參數的命令" BorderBrush="#FF11519C" BorderThickness="1" FontSize="16" Foreground="#FF127C0D" Margin="2"> <StackPanel Orientation="Horizontal"> <Button Content="拖拽上傳文件" AllowDrop="True"> <i:Interaction.Triggers> <i:EventTrigger EventName="Drop"> <mvvm:EventToCommand PassEventArgsToCommand="True" Command="{Binding DropCommand}" /> </i:EventTrigger> </i:Interaction.Triggers> </Button> <Label Content="FilePath:" FontSize="16" ></Label> <Label Content="{Binding DraggedFilePath}" FontSize="16" ></Label> </StackPanel> </GroupBox> </StackPanel>
private RelayCommand<DragEventArgs> dropCommand; /// <summary> /// 傳遞原事件參數 /// </summary> public RelayCommand<DragEventArgs> DropCommand { get { if (dropCommand == null) dropCommand = new RelayCommand<DragEventArgs>(e => ExecuteDrop(e)); return dropCommand; } set { dropCommand = value; } } private void ExecuteDrop(DragEventArgs e) { DraggedFilePath = ((System.Array)e.Data.GetData(System.Windows.DataFormats.FileDrop)).GetValue(0).ToString(); }
注:本系列文章參考