WPF: выполнение привязки команды в поле поиска при нажатии кнопки enter
У меня есть поле поиска в моем приложении WPF с кнопкой поиска, которая содержит привязку команды. Это отлично работает, но как я могу использовать ту же команду привязка текстового поля при нажатии Enter на клавиатуре? Примеры, которые я видел, все используют код позади с обработчиком событий KeyDown. Есть ли умный способ сделать эту работу только с XAML и привязкой команд?
4 ответов
вы можете использовать свойство IsDefault кнопки:
<Button Command="SearchCommand" IsDefault="{Binding ElementName=SearchTextBox,
Path=IsKeyboardFocused}">
Search!
</Button>
принятый ответ работает только если у вас уже есть кнопка, связанная с командой.
чтобы избежать этого ограничения, используйте TextBox.InputBindings:
<TextBox.InputBindings>
<KeyBinding Key="Enter" Command="{Binding Path=MyCommand}"></KeyBinding>
</TextBox.InputBindings>
на Присм справочная реализация содержит реализацию именно того, что вам нужно.
основные шаги:
- создать статический класс EnterKey
- зарегистрированное присоединенное свойство "Command" типа ICommand на EnterKey
- зарегистрированное присоединенное свойство "EnterKeyCommandBehavior" типа EnterKeyCommandBehavior на EnterKey
- когда значение" Command " изменится, прикрепите "EnterKeyCommandBehavior" для элемента управления в качестве нового экземпляра EnterKeyCommandBehavior и присвоить ICommand командного свойства поведения.
- если поведение уже подключено, используйте существующий экземпляр
- EnterKeyCommandBehavior принимает UIElement в конструкторе и присоединяется к PreviewKeyDown (или KeyDown, если вы хотите оставаться совместимым с Silverlight).
- в обработчике событий, если ключ Enter, выполните ICommand (если CanExecute имеет значение true).
Это позволяет использовать поведение так:
<TextBox prefix:EnterKey.Command="{Binding Path=SearchCommand}" />
Я пробовал текстовое поле.Вводит решение Грега Самсона, но получил ошибку, сказав, что я могу привязываться к textinputs только через свойство зависимостей. В конце концов я нашел следующее решение.
создайте класс CommandReference, который выглядит следующим образом:
public class CommandReference : Freezable, ICommand
{
public CommandReference()
{
//
}
public static readonly DependencyProperty CommandProperty = DependencyProperty.Register("Command", typeof(ICommand), typeof(CommandReference), new PropertyMetadata(new PropertyChangedCallback(OnCommandChanged)));
public ICommand Command
{
get { return (ICommand)GetValue(CommandProperty); }
set { SetValue(CommandProperty, value); }
}
#region ICommand Members
public bool CanExecute(object parameter)
{
if (Command != null)
return Command.CanExecute(parameter);
return false;
}
public void Execute(object parameter)
{
Command.Execute(parameter);
}
public event EventHandler CanExecuteChanged;
private static void OnCommandChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
CommandReference commandReference = d as CommandReference;
ICommand oldCommand = e.OldValue as ICommand;
ICommand newCommand = e.NewValue as ICommand;
if (oldCommand != null)
{
oldCommand.CanExecuteChanged -= commandReference.CanExecuteChanged;
}
if (newCommand != null)
{
newCommand.CanExecuteChanged += commandReference.CanExecuteChanged;
}
}
#endregion
#region Freezable
protected override Freezable CreateInstanceCore()
{
throw new NotImplementedException();
}
#endregion
}
в Xaml добавьте это в ресурсы UserControl:
<UserControl.Resources>
<Base:CommandReference x:Key="SearchCommandRef" Command="{Binding Path = SomeCommand}"/>
фактическое текстовое поле выглядит так:
<TextBox Text="{Binding Path=SomeText}">
<TextBox.InputBindings>
<KeyBinding Command="{StaticResource SearchCommandRef}" Key="Enter"/>
</TextBox.InputBindings>
</TextBox>
Я не помню, откуда у меня этот код, но этот сайт также объясняет это;