I have a form with a textbox and a button.
When that textbox has it's value changed, the button command doesn't call the CanExecute method of it's command.
The command parameter is set but doesn't seem to change. After load the window, the button remains disabled.
<TextBox Text="{Binding Name, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />
<Button Content="Save" Command="{Binding SaveChangesCommand}" CommandParameter="{Binding Name}" />
I know the binding is working because I created a behavior that receives a target binding and raise the CanExecute when the the binding changes.
With this behavior, the CanExecute is called normally.
<Button Content="Save" Command="{Binding SaveChangesCommand}">
<i:Interaction.Behaviors>
<behaviors:CallCommandCanExecuteWhenBindingChange Target="{Binding Name}" />
</i:Interaction.Behaviors>
</Button>
ViewModel:
public class EditViewModel : INotifyPropertyChanged
{
private string _name;
public EditViewModel()
{
SaveChangesCommand = new DelegateCommand(p => SaveChanges(), p => CanSaveChanges());
}
public string Name
{
get { return _name; }
set
{
if (value == _name) return;
_name = value;
OnPropertyChanged();
}
}
public DelegateCommand SaveChangesCommand { get; private set; }
private void SaveChanges()
{
}
private bool CanSaveChanges()
{
return !string.IsNullOrWhiteSpace(Name);
}
}
DelegateCommand:
public interface IBaseCommand : ICommand
{
void OnCanExecuteChanged();
}
public class DelegateCommand : IBaseCommand
{
private readonly Action<object> _execute;
private readonly Func<object, bool> _canExecute;
public DelegateCommand(Action<object> execute, Func<object, bool> canExecute)
{
_execute = execute;
_canExecute = canExecute;
}
public event EventHandler CanExecuteChanged;
public bool CanExecute(object parameter)
{
return _canExecute(parameter);
}
public void Execute(object parameter)
{
_execute(parameter);
OnCanExecuteChanged();
}
public void OnCanExecuteChanged()
{
var handler = CanExecuteChanged;
if (handler != null)
handler(this, EventArgs.Empty);
}
}
CallCommandCanExecuteWhenBindingChange:
public class CallCommandCanExecuteWhenBindingChange : Behavior<FrameworkElement>
{
public static readonly DependencyProperty<CallCommandCanExecuteWhenBindingChange, object> TargetProperty;
private ICommandBase _command;
static CallCommandCanExecuteWhenBindingChange()
{
var dependency = new DependencyRegistry<CallCommandCanExecuteWhenBindingChange>();
TargetProperty = dependency.Register(b => b.Target, s => s.OnTargetChange());
}
public object Target
{
get { return TargetProperty.Get(this); }
set { TargetProperty.Set(this, value); }
}
private void OnTargetChange()
{
if (_command == null && AssociatedObject != null)
{
var field = AssociatedObject.GetType().GetProperty("Command");
_command = (IBaseCommand)field.GetValue(AssociatedObject);
}
if (_command != null)
_command.OnCanExecuteChanged();
}
}
Does anyone know why the button doesn't call the CanExecute?
In your DelegateCommand
implementation CanExecuteChanged
should add/remove to CommandManager.RequerySuggested
event
Occurs when the CommandManager detects conditions that might change the ability of a command to execute.
change it to
public event EventHandler CanExecuteChanged
{
add { CommandManager.RequerySuggested += value; }
remove { CommandManager.RequerySuggested -= value; }
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With