1

我已经使用 WPF 有一段时间了,但我是 Commands 的新手,但我想开始正确使用它们一次。在一个代码示例之后,我建立了一个单独的静态命令类来保存我的所有命令,它看起来像这样。

public static class Commands
{
    public static RoutedUICommand OpenDocument { get; set; }

    static Commands()
    {
        OpenDocument = new RoutedUICommand("Open Document", "OpenDocument", typeof(Commands));
    }

    public static void BindCommands(Window window)
    {
        window.CommandBindings.Add(new CommandBinding(OpenDocument, OpenDocument_Executed, OpenDocument_CanExecute));
    }

    private static void OpenDocument_CanExecute(object sender, CanExecuteRoutedEventArgs e)
    {
        // Should be set to true if an item is selected in the datagrid.
    }

    private static void OpenDocument_Executed(object sender, ExecutedRoutedEventArgs e)
    {

    }
}

我的问题是,虽然命令将绑定到 MainWindow.xaml 中的 Button 控件,但 OpenDocument_CanExecute 方法需要查看 MainWindow.xaml 中的 DataGrid 以查看是否选择了项目。

我怎样才能把东西连接起来,以便该方法可以看到 DataGrid?

解决方案

受到 Ken 的回复的启发(再次感谢!),我将以下内容放在适当的位置,效果很好。

主窗口.xaml.cs

public partial class MainWindow
{
    public MainWindow()
    {
        InitializeComponent();

        Loaded += delegate
        {
            DataContext = ViewModel.Current;
            Commands.BindCommands(this);
        };
    }
}

视图模型.cs

public class ViewModel
{
    private static ViewModel _current;
    public static ViewModel Current
    {
        get { return _current ?? (_current = new ViewModel()); }
        set { _current = value; }
    }

    public object SelectedItem { get; set; }
}

命令.cs

public static class Commands
{
    public static RoutedUICommand OpenDocument { get; set; }

    static Commands()
    {
        OpenDocument = new RoutedUICommand("Open Document", "OpenDocument", typeof(Commands));
    }

    public static void BindCommands(Window window)
    {
        window.CommandBindings.Add(new CommandBinding(OpenDocument, OpenDocument_Executed, OpenDocument_CanExecute));
    }

    private static void OpenDocument_CanExecute(object sender, CanExecuteRoutedEventArgs e)
    {
        e.CanExecute = ViewModel.Current.SelectedItem != null;
    }

    private static void OpenDocument_Executed(object sender, ExecutedRoutedEventArgs e)
    {

    }
}
4

3 回答 3

3

ICommand实现在MVVM模式中效果最好:

class ViewModel : INotifyPropertyChanged {
    class OpenDocumentCommand : ICommand {
        public bool CanExecute(object parameter) {
            return ViewModel.ItemIsSelected;
        }
        public OpenDocumentCommand(ViewModel viewModel) {
            viewModel.PropertyChanged += (s, e) => {
                if ("ItemIsSelected" == e.PropertyName) {
                    RaiseCanExecuteChanged();
                }
            };
        }
    }

    private bool _ItemIsSelected;

    public bool ItemIsSelected {
        get { return _ItemIsSelected; }
        set {
            if (value == _ItemIsSelected) return;
            _ItemIsSelected = value;
            RaisePropertyChanged("ItemIsSelected");
        }
    }

    public ICommand OpenDocument { 
        get { return new OpenDocumentCommand(this); } 
    }
}

显然,我遗漏了一大堆东西。但这种模式在过去对我来说效果很好。

于 2012-04-26T19:22:42.777 回答
0

如果您将命令与 UI 实现紧密耦合,为什么还要实现它呢?只需响应 datagrid.SelectionChanged 并编写应该发生的代码。

否则,将其放在 ViewModel 中。让 ViewModel 监控它的状态并评估 CanExe 何时为真。

编辑

另一方面,您可以将参数传递给您的命令,以及 Exe() 和 CanExe() 方法

//where T is the type you want to operate on
public static RoutedUICommand<T> OpenDocument { get; set; }
于 2012-04-26T17:51:00.900 回答
0

如果您正在执行 MVVM 解决方案,这将是实现发布/订阅聚合器的最佳时机,该聚合器允许控件相互“对话”。它背后的要点是数据网格将发布一个事件“打开文档”。后续控件可以订阅该事件并对“打开文档”的调用做出反应。发布/订阅模式防止数据网格和控件紧密耦合。对事件聚合器进行一些搜索,我想你会在路上。

于 2012-04-26T19:21:58.993 回答