一、 ICommand接口
ICommand接口定义在System.Windows.Input命名空间中,包含三个主要成员:
- Execute: 当命令被触发时执行的方法,通常包含业务逻辑。
- CanExecute: 判断命令是否可执行的方法,返回布尔值(如基于某些条件禁用按钮)。
- CanExecuteChanged: 事件,当命令的可执行状态改变时触发(如当条件变化时通知UI更新)。
使用ICommand的好处包括:
- 解耦UI和业务逻辑:命令在ViewModel中定义,而不是在View的代码后台文件。
- 支持启用/禁用状态:通过
CanExecute动态控制UI元素的可用性。 - 易于测试:命令逻辑可以独立于UI进行单元测试。
二、 实现自定义命令
实现一个自定义命令,需要创建一个类来实现ICommand接口。示例如下:
usingSystem;usingSystem.Windows.Input;publicclassRelayCommand:ICommand{privatereadonlyAction<object>_execute;privatereadonlyFunc<object,bool>_canExecute;publicRelayCommand(Action<object>execute,Func<object,bool>canExecute=null){_execute=execute??thrownewArgumentNullException(nameof(execute));_canExecute=canExecute;}publicboolCanExecute(objectparameter){return_canExecute==null||_canExecute(parameter);}publicvoidExecute(objectparameter){_execute(parameter);}publiceventEventHandlerCanExecuteChanged{add{CommandManager.RequerySuggested+=value;}remove{CommandManager.RequerySuggested-=value;}}}RelayCommand是一个通用的命令实现,可以通过传入一个Action来定义执行逻辑以及一个可选的Func来定义可执行条件。CanExecuteChanged事件使用CommandManager.RequerySuggested自动处理状态更新,简化开发。- 要使用这个命令,在ViewModel中创建实例并绑定到UI。
三、 在ViewModel中使用命令
在MVVM模式中,ViewModel持有命令实例。示例如下:
publicclassMainViewModel{publicICommandClickCommand{get;privateset;}publicMainViewModel(){ClickCommand=newRelayCommand(ExecuteClick,CanExecuteClick);}privatevoidExecuteClick(objectparameter){// 执行逻辑,例如显示消息MessageBox.Show("按钮被点击了!");}privateboolCanExecuteClick(objectparameter){// 判断命令是否可执行,例如基于某个条件returntrue;// 这里总是可执行,实际中可能检查属性值}}在这个ViewModel中:
ClickCommand是一个RelayCommand实例,绑定到ExecuteClick和CanExecuteClick方法。ExecuteClick方法包含实际业务逻辑(如弹出消息)。CanExecuteClick方法返回true表示命令始终可用;实际应用中,可能基于ViewModel的属性(如IsEnabled)动态返回。
四、 在XAML中绑定命令到UI
在WPF的XAML文件中,你可以将命令绑定到控件(如按钮)。首先,确保ViewModel被设置为View的DataContext。
<StackPanel><ButtonContent="点击我"Command="{Binding ClickCommand}"/></StackPanel>按钮的Command属性绑定到ClickCommand命令。当按钮被点击时,会自动调用Execute方法;如果CanExecute返回false,按钮会被禁用。
四、注意事项
- 命令参数:
Execute和CanExecute方法接受一个object parameter参数,可以通过XAML的CommandParameter属性传递数据,例如:<ButtonCommand="{Binding ClickCommand}"CommandParameter="Hello"/> - 内置命令: WPF提供了一些内置命令(如
ApplicationCommands.Open),可以直接使用或继承。