Как я могу назначить поведение «Закрыть при нажатии клавиши Escape» для всех окон WPF в проекте?

Есть ли какой-нибудь простой способ заставить целое приложение WPF реагировать на нажатия клавиш Escape, пытаясь закрыть текущую вдову? Ручная настройка привязок команд и ввода не является большой проблемой, но мне интересно, является ли повторение этого XAML во всех окнах наиболее элегантным подходом?

<Window.CommandBindings>
        <CommandBinding Command="Close" Executed="CommandBinding_Executed" />
</Window.CommandBindings>
<Window.InputBindings>
        <KeyBinding Key="Escape" Command="Close" />
</Window.InputBindings>

Любые конструктивные предложения приветствуются!

 CharithJ19 мар. 2019 г., 01:47
Подобный вопрос здесь:stackoverflow.com/questions/7691713/...

Ответы на вопрос(7)

НаWindowпоказано сShowDialog() ты можешь использовать:

<!-- Button to close on Esc -->
<Button IsCancel="True" Width="0" Height="0"/>

Или вы можете просто добавить кнопку с Отмена как текст и установитьIsCancel = True, Тогда Escape будет работать как команда по умолчанию для закрытия.

 Peter Perháč05 нояб. 2010 г., 22:27
не всегда идеально, но все же нравится предложение. Достаточно просто.

СоздайтеRoutedUICommand как ниже

 private static RoutedUICommand EscUICommand = new RoutedUICommand("EscBtnCommand"
       , "EscBtnCommand"
       , typeof(WindowName)
       , new InputGestureCollection(new InputGesture[] 
           { new KeyGesture(Key.Escape, ModifierKeys.None, "Close") }));

и добавить его привязку команды в конструкторе

CommandBindings.Add(new CommandBinding(EscUICommand, (sender, e) => { this.Hide(); }));
 DonBoitnott11 февр. 2019 г., 16:37
Это работает довольно хорошо, и когда у вас уже есть класс для вашего окна, он хранит все прямо там.

Другой возможный способ - использовать прикрепленные свойства.

Ниже приведен суть кода:

<script src="https://gist.github.com/meziantou/1e98d7d7aa6aa859d916.js"></script>

Решение Вопроса

Все, что я могу предложить для улучшения, это устранить необходимость в обработчике событий путем привязки к статическому экземпляру команды.

Примечание. Это будет работать только в .NET 4 и более поздних версиях, поскольку для этого требуется возможность привязки кKeyBinding свойства.

Сначала создайте команду, которая принимает Window в качестве параметра и вызываетClose в пределахExecute метод:

public class CloseThisWindowCommand : ICommand
{
    #region ICommand Members

    public bool CanExecute(object parameter)
    {
        //we can only close Windows
        return (parameter is Window);
    }

    public event EventHandler CanExecuteChanged;

    public void Execute(object parameter)
    {
        if (this.CanExecute(parameter))
        {
            ((Window)parameter).Close();
        }
    }

    #endregion

    private CloseThisWindowCommand()
    {

    }

    public static readonly ICommand Instance = new CloseThisWindowCommand();
}

Тогда вы можете связать свойKeyBinding в статикуInstance имущество:

<Window.InputBindings>
    <KeyBinding Key="Escape" Command="{x:Static local:CloseThisWindowCommand.Instance}" CommandParameter="{Binding RelativeSource={RelativeSource FindAncestor, AncestorType=Window}}" />
</Window.InputBindings>

Я не знаю, что это обязательнолучше чем ваш подход, но это означает, что на вершине каждогоWindow и что вам не нужно включать обработчик событий в каждом

 Maslow07 апр. 2016 г., 15:56
это может привести к утечке памяти, так как синглтон имеетEventHandler?
 Steve Greatrex07 апр. 2016 г., 16:00
Это не должно делать, но, как мы не заботимся оCanExecute функциональность, которую вы могли бы защитить от этого, имеяCanExecute всегда возвращать истину и заменитьCanExecuteChanged обработчик с пустыми реализациями вложений (например,public event EventHandler CanExecuteChanged { add {} remove {} })

Вы также можете использовать PreviewKeyDown Event

PreviewKeyDown="UserControl_PreviewKeyDown"

Код за вызовом вам закрыть команду

private void UserControl_PreviewKeyDown(object sender, System.Windows.Input.KeyEventArgs e)
        {
            if (e.Key == Key.Escape)
            {
                _vm.OnCloseCommand(sender);
            }
        }

Ничто из вышеперечисленного не помогло мне, кроме Кая. Я изменил его ответ: я добавил 'btn_close.IsCancel = true;' для конструктора. SettingsWindow - это мое второе окно, а главное окно (по умолчанию) MainWindow.

  public partial class SettingsWindow : Window {
    public SettingsWindow() {
      InitializeComponent();
      btn_close.IsCancel = true;
    }
    private void btn_close_Click(object sender, RoutedEventArgs e) {
      this.Close();
    }
  }

Надеюсь, поможет,

Simon S люблю Нию

Ваш ответ на вопрос