1

我正在尝试使用一些命令制作用户控件。如果我使用此处显示的方法连接 xaml 中的命令http://msdn.microsoft.com/en-us/library/vstudio/ms771361(v=vs.90).aspx它可以工作,但如果我使用来自的 DelegateCommand Prism 库 CanExecuteChanged 不会在用户控件上触发,我不知道为什么。我很抱歉,我意识到这是很多代码。Execute 正确触发,但 CanExecute 永远不会。

提前致谢。

自定义控件 Xaml

<UserControl x:Class="Controls.LinkButton"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
    <TextBlock>
        <Hyperlink x:Name="hyperLink" Click="Hyperlink_Click">
            <Run x:Name="textRun"
                 Text="Click Me"/>
        </Hyperlink>
    </TextBlock>
</UserControl>

自定义控制代码

public partial class LinkButton : UserControl, ICommandSource
{
    public LinkButton()
        : base()
    {
        InitializeComponent();
        textRun.DataContext = this;
        hyperLink.DataContext = this;
    }

    public ICommand Command
    {
        get { return (ICommand)GetValue(CommandProperty); }
        set { SetValue(CommandProperty, value); }
    }

    public static readonly DependencyProperty CommandProperty =
        DependencyProperty.Register("Command", typeof(ICommand), typeof(LinkButton), new PropertyMetadata(null, new PropertyChangedCallback(CommandChanged)));

    public object CommandParameter
    {
        get { return (object)GetValue(CommandParameterProperty); }
        set { SetValue(CommandParameterProperty, value); }
    }

    public static readonly DependencyProperty CommandParameterProperty =
        DependencyProperty.Register("CommandParameter", typeof(object), typeof(LinkButton), new PropertyMetadata(null));

    public IInputElement CommandTarget
    {
        get { return (IInputElement)GetValue(CommandTargetProperty); }
        set { SetValue(CommandTargetProperty, value); }
    }

    public static readonly DependencyProperty CommandTargetProperty =
        DependencyProperty.Register("CommandTarget", typeof(IInputElement), typeof(LinkButton), new PropertyMetadata(null));

    private void Hyperlink_Click(object sender, RoutedEventArgs e)
    {
        if (Command != null)
        {
            RoutedCommand command = Command as RoutedCommand;

            if (command != null)
            {
                command.Execute(CommandParameter, CommandTarget);
            }
            else
            {
                ((ICommand)Command).Execute(CommandParameter);
            }
        }
    }

    public static EventHandler canExecuteChangedHandler;

    private static void CommandChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
    {
        LinkButton lb = (LinkButton)d;
        lb.HookUpCommand((ICommand)e.OldValue, (ICommand)e.NewValue);
    }

    // Add a new command to the Command Property. 
    private void HookUpCommand(ICommand oldCommand, ICommand newCommand)
    {
        // If oldCommand is not null, then we need to remove the handlers. 
        if (oldCommand != null)
        {
            RemoveCommand(oldCommand, newCommand);
        }
        AddCommand(oldCommand, newCommand);
    }

    // Remove an old command from the Command Property. 
    private void RemoveCommand(ICommand oldCommand, ICommand newCommand)
    {
        EventHandler handler = CanExecuteChanged;
        oldCommand.CanExecuteChanged -= handler;
    }

    // Add the command. 
    private void AddCommand(ICommand oldCommand, ICommand newCommand)
    {
        EventHandler handler = new EventHandler(CanExecuteChanged);
        canExecuteChangedHandler = handler;
        if (newCommand != null)
        {
            newCommand.CanExecuteChanged += canExecuteChangedHandler;
        }
    }

    private void CanExecuteChanged(object sender, EventArgs e)
    {
        if (this.Command != null)
        {
            RoutedCommand command = this.Command as RoutedCommand;

            // If a RoutedCommand. 
            if (command != null)
            {
                if (command.CanExecute(CommandParameter, CommandTarget))
                {
                    this.IsEnabled = true;
                }
                else
                {
                    this.IsEnabled = false;
                }
            }
            // If a not RoutedCommand. 
            else
            {
                if (Command.CanExecute(CommandParameter))
                {
                    this.IsEnabled = true;
                }
                else
                {
                    this.IsEnabled = false;
                }
            }
        }
    }
}

窗户

<ctrl:LinkButton Command="{Binding LinkClicked}"/>


public partial class MainWindow : Window
{
    public MainWindow()
    {
        LinkClicked = new DelegateCommand(DoSomething, CanDoSomething);

        InitializeComponent();
        DataContext = this;
        LinkClicked.RaiseCanExecuteChanged();
    }

    public DelegateCommand LinkClicked { get; set; }

    public void DoSomething()
    {
        MessageBox.Show("Did Something");
    }

    public bool CanDoSomething()
    {
        return false;
    }
}
4

1 回答 1

1

这里的问题是您正在调用 LinkClicked.RaiseCanExecuteChanged(); 在设置 DataContext 之后,直到那时 LinkedCommand 未绑定,因此 DelegateCommand 的 CanExecuteChanged 事件为空,因此 RaiseCanExecuteChanged() 不执行任何操作。因此要避免在 Window 的加载事件中调用 LinkClicked.RaiseCanExecuteChanged() ,因为在此之前绑定将被更新。尽管这是一个肮脏的解决方案,因为您必须在使用此 LinkBut​​ton 并绑定其命令的任何地方执行此操作。

RaiseCanExecuteChanged 的​​实现是这样的

    public void RaiseCanExecuteChanged()
    {
        if (CanExecuteChanged != null)  //CanExecuteChanged is coming null in your case so the event is not fired.
            CanExecuteChanged(this, new EventArgs()); 
    }

或者更好的解决方案是你没有在 AddCommand 方法中调用 CanExecute,在实际的命令实现中调用 CanExecute

if (newCommand != null)
        {
            newCommand.CanExecuteChanged += CanExecuteChanged;
            newCommand.CanExecute(CommandParameter); //you are missing this.
        }

如果您这样做,则无需调用 RaiseCanExecuteChanged。

于 2014-08-15T17:11:13.473 回答