summaryrefslogtreecommitdiffstats
path: root/editor/input_event_configuration_dialog.cpp
diff options
context:
space:
mode:
authorMel Collins <sigh@ambimist.com>2023-08-03 15:18:26 +0200
committerMel Collins <sigh@ambimist.com>2024-01-26 14:42:28 +0100
commit8406e60522bb8d09649193be43c1c819edc1d059 (patch)
tree3c9ad2268b5c0c4ab50387e8d745528b2856daed /editor/input_event_configuration_dialog.cpp
parent4b6ad349886288405890b07d4a8da425eb3c97ec (diff)
downloadredot-engine-8406e60522bb8d09649193be43c1c819edc1d059.tar.gz
Add InputEventKey.location to tell left from right
This adds a new enum `KeyLocation` and associated property `InputEventKey.location`, which indicates the left/right location of key events which may come from one of two physical keys, eg. Shift, Ctrl. It also adds simulation of missing Shift KEYUP events for Windows. When multiple Shifts are held down at the same time, Windows natively only sends a KEYUP for the last one to be released.
Diffstat (limited to 'editor/input_event_configuration_dialog.cpp')
-rw-r--r--editor/input_event_configuration_dialog.cpp46
1 files changed, 44 insertions, 2 deletions
diff --git a/editor/input_event_configuration_dialog.cpp b/editor/input_event_configuration_dialog.cpp
index 0f483fcaef..22673bec64 100644
--- a/editor/input_event_configuration_dialog.cpp
+++ b/editor/input_event_configuration_dialog.cpp
@@ -64,6 +64,7 @@ void InputEventConfigurationDialog::_set_event(const Ref<InputEvent> &p_event, c
bool show_mods = false;
bool show_device = false;
bool show_key = false;
+ bool show_location = false;
if (mod.is_valid()) {
show_mods = true;
@@ -77,12 +78,17 @@ void InputEventConfigurationDialog::_set_event(const Ref<InputEvent> &p_event, c
if (k.is_valid()) {
show_key = true;
- if (k->get_keycode() == Key::NONE && k->get_physical_keycode() == Key::NONE && k->get_key_label() != Key::NONE) {
+ Key phys_key = k->get_physical_keycode();
+ if (k->get_keycode() == Key::NONE && phys_key == Key::NONE && k->get_key_label() != Key::NONE) {
key_mode->select(KEYMODE_UNICODE);
} else if (k->get_keycode() != Key::NONE) {
key_mode->select(KEYMODE_KEYCODE);
- } else if (k->get_physical_keycode() != Key::NONE) {
+ } else if (phys_key != Key::NONE) {
key_mode->select(KEYMODE_PHY_KEYCODE);
+ if (phys_key == Key::SHIFT || phys_key == Key::CTRL || phys_key == Key::ALT || phys_key == Key::META) {
+ key_location->select((int)k->get_location());
+ show_location = true;
+ }
} else {
// Invalid key.
event = Ref<InputEvent>();
@@ -103,6 +109,7 @@ void InputEventConfigurationDialog::_set_event(const Ref<InputEvent> &p_event, c
mod_container->set_visible(show_mods);
device_container->set_visible(show_device);
key_mode->set_visible(show_key);
+ location_container->set_visible(show_location);
additional_options_container->show();
// Update mode selector based on original key event.
@@ -240,6 +247,9 @@ void InputEventConfigurationDialog::_on_listen_input_changed(const Ref<InputEven
k->set_physical_keycode(Key::NONE);
k->set_keycode(Key::NONE);
}
+ if (key_location->get_selected_id() == (int)KeyLocation::UNSPECIFIED) {
+ k->set_location(KeyLocation::UNSPECIFIED);
+ }
}
Ref<InputEventWithModifiers> mod = received_event;
@@ -433,6 +443,17 @@ void InputEventConfigurationDialog::_key_mode_selected(int p_mode) {
_set_event(k, original_event);
}
+void InputEventConfigurationDialog::_key_location_selected(int p_location) {
+ Ref<InputEventKey> k = event;
+ if (k.is_null()) {
+ return;
+ }
+
+ k->set_location((KeyLocation)p_location);
+
+ _set_event(k, original_event);
+}
+
void InputEventConfigurationDialog::_input_list_item_selected() {
TreeItem *selected = input_list_tree->get_selected();
@@ -594,6 +615,8 @@ void InputEventConfigurationDialog::popup_and_configure(const Ref<InputEvent> &p
// Select "All Devices" by default.
device_id_option->select(0);
+ // Also "all locations".
+ key_location->select(0);
}
if (!p_current_action_name.is_empty()) {
@@ -726,5 +749,24 @@ InputEventConfigurationDialog::InputEventConfigurationDialog() {
key_mode->hide();
additional_options_container->add_child(key_mode);
+ // Key Location Selection
+
+ location_container = memnew(HBoxContainer);
+ location_container->hide();
+
+ Label *location_label = memnew(Label);
+ location_label->set_text(TTR("Physical location"));
+ location_container->add_child(location_label);
+
+ key_location = memnew(OptionButton);
+ key_location->set_h_size_flags(Control::SIZE_EXPAND_FILL);
+ key_location->add_item(TTR("Any"), (int)KeyLocation::UNSPECIFIED);
+ key_location->add_item(TTR("Left"), (int)KeyLocation::LEFT);
+ key_location->add_item(TTR("Right"), (int)KeyLocation::RIGHT);
+ key_location->connect("item_selected", callable_mp(this, &InputEventConfigurationDialog::_key_location_selected));
+
+ location_container->add_child(key_location);
+ additional_options_container->add_child(location_container);
+
main_vbox->add_child(additional_options_container);
}