首页 > 解决方案 > WPF ICommand 等效于 Click -= Button_Click

问题描述

对于 WPF ICommand,事件处理程序 -= 的等价物是什么?

我们有一个用户控制按钮,用于管理提交购买请求的功能,但需要用户先在工作站上注册。

WinForms 的用法是:

this.Click -= this.btnRegister_Click;
this.Click -= this.btnSubmit_Click;

WPF 中的现代化版本具有单个 ICommand 属性和典型声明。

public ICommand ClickCommand { get; set; }
....
ClickCommand = new DelegateCommand(RegisterClicked);

一旦用户注册,用户控件将重新初始化到提交过程。存在一个简单的触发器来检查按钮的状态。从视觉上看,触发器工作得很好,这表明程序正在处理初始化代码块。

<Label.Style>
    <Style TargetType="Label" BasedOn="{StaticResource ButtonLabel}">
        <Setter Property="Content" Value="Approvals"/>
        <Style.Triggers>
            <DataTrigger Binding="{Binding Path=ButtonState}" Value="{x:Static locdata:WorkflowButtonState.Register}">
                <Setter Property="Content" Value="Register"/>
            </DataTrigger>
            <DataTrigger Binding="{Binding Path=ButtonState}" Value="{x:Static locdata:WorkflowButtonState.Submit}">
                <Setter Property="Content" Value="Submit"/>
            </DataTrigger>
        </Style.Triggers>
    </Style>
</Label.Style>

初始化代码块

private void InitRegister()
{
    ButtonState = WorkflowButtonState.Register;
    ClickCommand = new DelegateCommand(RegisterClicked);
}
private void InitSubmit()
{
    ButtonState = WorkflowButtonState.Submit;
    ClickCommand = new DelegateCommand(SubmitClicked);
}

private void Reset()
{
    ClickCommand = null;
    ButtonState = WorkflowButtonState.Default;
}

调试模式清楚地显示 ClickCommand 执行方法是 SubmitClicked,但是 RegisterClicked 事件仍然会触发。这仅在注册过程首先发生时发生,因为用户之前没有登录工作站。

复制 EventHandler -= 行为需要什么,因为设置 ClickCommand = null 没有帮助?

更新

到目前为止,我的应用程序中的 ICommand 用法一直是单次使用。正如安迪的回答所说,它应该被视为一个属性并引发属性更改事件。

正如我多次为属性所做的那样,更新后的 ICommand 声明是所有需要更改的。

private ICommand _clickCommand;
public ICommand ClickCommand { get => _clickCommand; set { _clickCommand = value; OnPropertyChanged(); } }

标签: wpfeventhandlericommanddelegatecommand

解决方案


这里要考虑两件事:

  1. 按钮上的命令不是事件,而是依赖属性。 https://docs.microsoft.com/en-us/dotnet/api/system.windows.controls.primitives.buttonbase.command?view=netcore-3.1

  2. 你如何告诉视图中的依赖属性应该从视图模型中再次读取它的值?

您通知属性已更改。

一些示例标记:

    <Window.DataContext>
        <local:MainWindowViewModel/>
    </Window.DataContext>
    <StackPanel>
        <Button Command="{Binding TestCommand}" Content="Test"/>
        <Button Command="{Binding ChangeCommand}" Content="Change Test Command"/>
    </StackPanel>
</Window>

有一个测试命令将作为一个命令开始,然后当您单击更改测试命令按钮时,它将更改为另一个命令。

我的视图模型:

    public class MainWindowViewModel : BindableBase
    {
        private DelegateCommand testCommand;
        public DelegateCommand TestCommand 
        {
            get => testCommand;
            set { testCommand = value; RaisePropertyChanged(); }
        }
        public DelegateCommand ChangeCommand { get; set; }
        public MainWindowViewModel()
        {
            TestCommand = new DelegateCommand(() => { MessageBox.Show("Original Command"); });
            ChangeCommand = new DelegateCommand(() => 
            {
                TestCommand = new DelegateCommand(() => { MessageBox.Show("Replacement is used"); });
            });
        }
    }

当调用 changecommand 时,testcommand 的委托命令被替换为新命令。

和。

调用已更改的属性。

它告诉 UI 去获取那个新命令。

如果我然后删除从 TestCommand 更改的 raise 属性:

        public DelegateCommand TestCommand 
        {
            get => testCommand;
            set { testCommand = value;  }
        }

单击更改按钮,然后单击我的测试按钮,我得到原始命令运行。这是你看到的问题。因为你的命令看起来像:

public ICommand ClickCommand { get; set; }

没有 raise 属性改变。

笔记:

如果我的更改命令是:

            ChangeCommand = new DelegateCommand(() => 
            {
                TestCommand = null;
            });

然后单击该更改按钮,单击测试按钮,没有任何反应。它的命令现在为空。


推荐阅读