Подтвердить что ты не робот

Как связать кнопку WPF с командой в ViewModelBase?

У меня есть представление AttributeView, которое содержит всевозможные атрибуты. Там также кнопка, которая при нажатии, должна устанавливать значения по умолчанию для атрибутов. У меня также есть класс ViewModelBase, который является базовым классом для всех ViewModels, которые у меня есть. Проблема в том, что я не могу заставить кнопку привязываться к команде с 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, так что это может быть очень простой вопрос)?

4b9b3361

Ответ 1

 <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();
    }
}

Я надеюсь, что это даст вам идею.

Ответ 2

Решение, предложенное здесь этической логикой, работает как шарм. Только одно замечание - вместо реализации вашего собственного CommandHandler, рассмотрите возможность использования Microsoft.Practices.Prism.Commands.DelegateCommand (который практически одинаков, за исключением дополнительного ctor, если ваша кнопка всегда включена).