如何将WPF按钮绑定到ViewModelBase中的命令?

时间:2012-09-14 10:45:36

标签: c# wpf mvvm command viewmodel

我有一个包含各种属性的视图AttributeView。还有一个按钮,按下时,它应该为属性设置默认值。我还有一个ViewModelBase类,它是我所有ViewModel的基类。 问题是我似乎无法使用WPF将命令绑定到命令。

我试过这个,但它没有做任何事情:

<Button Command="{Binding DataInitialization}" Content="{x:Static localProperties:Resources.BtnReinitializeData}"></Button>

该命令(在ViewModelBase中)定义如下:

public CommandBase DataInitialization { get; protected set; }

并在应用程序启动时为该命令创建一个新实例:

DataInitialization = new DataInitializationCommand()

然而,WPF绑定似乎没有“找到”命令(按下按钮什么都不做)。当前视图中使用的ViewModel派生自ViewModelBase。我还能尝试什么(我对WPF很新,所以这可能是一个非常简单的问题)?

2 个答案:

答案 0 :(得分:104)

 <Grid >
    <Grid.ColumnDefinitions>
        <ColumnDefinition Width="*"/>
    </Grid.ColumnDefinitions>
    <Button Command="{Binding ClickCommand}" Width="100" Height="100" Content="wefwfwef"/>
</Grid>

窗口背后的代码:

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
        DataContext = new ViewModelBase();
    }
}

ViewModel:

public class ViewModelBase
{
    private ICommand _clickCommand;
    public ICommand ClickCommand
    {
        get
        {
            return _clickCommand ?? (_clickCommand = new CommandHandler(() => MyAction(), ()=> CanExecute));
        }
    }
     public bool CanExecute
     {
        get
        {
            // check if executing is allowed, i.e., validate, check if a process is running, etc. 
            return true/false;
        }
     }

    public void MyAction()
    {

    }
}

命令处理程序:

 public class CommandHandler : ICommand
{
    private Action _action;
    private Func<bool> _canExecute;

    /// <summary>
    /// Creates instance of the command handler
    /// </summary>
    /// <param name="action">Action to be executed by the command</param>
    /// <param name="canExecute">A bolean property to containing current permissions to execute the command</param>
    public CommandHandler(Action action, Func<bool> canExecute)
    {
        _action = action;
        _canExecute = canExecute;
    }

    /// <summary>
    /// Wires CanExecuteChanged event 
    /// </summary>
    public event EventHandler CanExecuteChanged
    {
        add { CommandManager.RequerySuggested += value; }
        remove { CommandManager.RequerySuggested -= value; }
    }

    /// <summary>
    /// Forcess checking if execute is allowed
    /// </summary>
    /// <param name="parameter"></param>
    /// <returns></returns>
    public bool CanExecute(object parameter)
    {
        return _canExecute.Invoke();
    }

    public void Execute(object parameter)
    {
        _action();
    }
}

我希望这会给你这个想法。

答案 1 :(得分:0)

伦理学家在这里建议的解决方案就像一个魅力。 只是一句话-考虑使用Microsoft.Practices.Prism.Commands.DelegateCommand(实际上是相同的,只是在始终启用按钮的情况下增加了ctor除外),而不是实现自己的CommandHandler。