Страницы

Поиск по вопросам

пятница, 5 апреля 2019 г.

Звук при нажатии горячих клавиш

Создал сочетание клавиш Ctrl+O и Ctrl+А. При нажатии все работает, как я хотел, но проигрывается стандартный звук «Восклицание». Откуда он берется и как его устранить?
Я заметил что при запуске формы у меня в фокусе текстбокс. И если сделать его неактивным звук пропадает.


Ответ

TextBox - наиболее часто используемый элемент WinForms, прямую конкуренцию может составить только Label. И именно в нем есть досадный баг, который исправили только в .NET 4.6.1
Почти все контролы, когда они находятся в фокусе, перехватывают на себя обработку нажатий на клавиши и передают событие родительскому контейнеру, только если это явно указать в обработчике. В TextBox по-умолчанию определено много горячих клавиш для работы именно с ним, но если он находится в режиме MultyLine, то большинство сочетаний клавиш Ctrl+some key работают некорректно. Если быть точным, то единственное что отрабатывает для этих сочетаний - это проигрывание системного звука "Восклицание" (не помню как он называется в английской версии), который обычно сигнализирует об ошибочном действии пользователя (например при клике в основное окно приложения, при открытом модальном диалоге).
Решений два, радикальное и "костыль". Решений три:
Мягкое решение - добавить на форму MenuStrip и настроить горячие клавиши в MenuStripItem-ах. Меню перехватит все заданные сочетания, они не попадут в TextBox, следовательно не будет проигрываться звук. Хорошо, но подходит не всегда.
Радикальное решение - пересобрать проект на платформе .NET 4.6.1. В этой редакции исправили работу некоторых сочетаний клавиш, например Ctrl+A (выделить всё), но поведение по-умолчанию для не используемых сочетаний клавиш не изменилось. Для нас это означает, что звуки отдельных сочетаний клавиш придется экранировать отдельно. Либо с помощью меню (см. выше), либо как описано ниже.
Исправляем баг руками - когда ни добавление меню, ни повышение версии платформы не допустимо, придется исправлять баг своими силами. Для этого вешаем обработчик на событие TextBox.KeyDown, в котором прописываем необходимые нам сочетания клавиш следующим образом (для примера Ctrl+A):
private void textBox1_KeyDown(object sender, KeyEventArgs e) { if (e.KeyData == (Keys.Control | Keys.A)) { textBox1.SelectAll(); //убираем звуковое сопровождение при нажатии клавиш e.Handled = e.SuppressKeyPress = true; } }
Отдельно замечу, что нельзя использовать для отключения звуков вариант:
private void textBox1_KeyDown(object sender, KeyEventArgs e) { e.Handled = e.SuppressKeyPress = true; }
Так как это не только заглушит звуки, но и превратит TextBox в тыкву, отключив обработку клавиш полностью.

Комментариев нет:

Отправить комментарий