123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167 |
- /*************************************************************************/
- /* action_map_editor.cpp */
- /*************************************************************************/
- /* This file is part of: */
- /* GODOT ENGINE */
- /* https://godotengine.org */
- /*************************************************************************/
- /* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
- /* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
- /* */
- /* Permission is hereby granted, free of charge, to any person obtaining */
- /* a copy of this software and associated documentation files (the */
- /* "Software"), to deal in the Software without restriction, including */
- /* without limitation the rights to use, copy, modify, merge, publish, */
- /* distribute, sublicense, and/or sell copies of the Software, and to */
- /* permit persons to whom the Software is furnished to do so, subject to */
- /* the following conditions: */
- /* */
- /* The above copyright notice and this permission notice shall be */
- /* included in all copies or substantial portions of the Software. */
- /* */
- /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
- /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
- /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
- /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
- /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
- /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
- /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
- /*************************************************************************/
- #include "action_map_editor.h"
- #include "core/input/input_map.h"
- #include "core/os/keyboard.h"
- #include "editor/editor_scale.h"
- #include "scene/gui/center_container.h"
- /////////////////////////////////////////
- // Maps to 2*axis if value is neg, or + 1 if value is pos.
- static const char *_joy_axis_descriptions[JOY_AXIS_MAX * 2] = {
- TTRC("Left Stick Left, Joystick 0 Left"),
- TTRC("Left Stick Right, Joystick 0 Right"),
- TTRC("Left Stick Up, Joystick 0 Up"),
- TTRC("Left Stick Down, Joystick 0 Down"),
- TTRC("Right Stick Left, Joystick 1 Left"),
- TTRC("Right Stick Right, Joystick 1 Right"),
- TTRC("Right Stick Up, Joystick 1 Up"),
- TTRC("Right Stick Down, Joystick 1 Down"),
- TTRC("Joystick 2 Left"),
- TTRC("Left Trigger, Sony L2, Xbox LT, Joystick 2 Right"),
- TTRC("Joystick 2 Up"),
- TTRC("Right Trigger, Sony R2, Xbox RT, Joystick 2 Down"),
- TTRC("Joystick 3 Left"),
- TTRC("Joystick 3 Right"),
- TTRC("Joystick 3 Up"),
- TTRC("Joystick 3 Down"),
- TTRC("Joystick 4 Left"),
- TTRC("Joystick 4 Right"),
- TTRC("Joystick 4 Up"),
- TTRC("Joystick 4 Down"),
- };
- String InputEventConfigurationDialog::get_event_text(const Ref<InputEvent> &p_event) {
- ERR_FAIL_COND_V_MSG(p_event.is_null(), String(), "Provided event is not a valid instance of InputEvent");
- // Joypad motion events will display slighlty differently than what the event->as_text() provides. See #43660.
- Ref<InputEventJoypadMotion> jpmotion = p_event;
- if (jpmotion.is_valid()) {
- String desc = TTR("Unknown Joypad Axis");
- if (jpmotion->get_axis() < JOY_AXIS_MAX) {
- desc = RTR(_joy_axis_descriptions[2 * jpmotion->get_axis() + (jpmotion->get_axis_value() < 0 ? 0 : 1)]);
- }
- return vformat("Joypad Axis %s %s (%s)", itos(jpmotion->get_axis()), jpmotion->get_axis_value() < 0 ? "-" : "+", desc);
- } else {
- return p_event->as_text();
- }
- }
- void InputEventConfigurationDialog::_set_event(const Ref<InputEvent> &p_event) {
- if (p_event.is_valid()) {
- event = p_event;
- // Update Label
- event_as_text->set_text(get_event_text(event));
- Ref<InputEventKey> k = p_event;
- Ref<InputEventMouseButton> mb = p_event;
- Ref<InputEventJoypadButton> joyb = p_event;
- Ref<InputEventJoypadMotion> joym = p_event;
- Ref<InputEventWithModifiers> mod = p_event;
- // Update option values and visibility
- bool show_mods = false;
- bool show_device = false;
- bool show_phys_key = false;
- if (mod.is_valid()) {
- show_mods = true;
- mod_checkboxes[MOD_ALT]->set_pressed(mod->get_alt());
- mod_checkboxes[MOD_SHIFT]->set_pressed(mod->get_shift());
- mod_checkboxes[MOD_COMMAND]->set_pressed(mod->get_command());
- mod_checkboxes[MOD_CONTROL]->set_pressed(mod->get_control());
- mod_checkboxes[MOD_META]->set_pressed(mod->get_metakey());
- store_command_checkbox->set_pressed(mod->is_storing_command());
- }
- if (k.is_valid()) {
- show_phys_key = true;
- physical_key_checkbox->set_pressed(k->get_physical_keycode() != 0 && k->get_keycode() == 0);
- } else if (joyb.is_valid() || joym.is_valid() || mb.is_valid()) {
- show_device = true;
- _set_current_device(event->get_device());
- }
- mod_container->set_visible(show_mods);
- device_container->set_visible(show_device);
- physical_key_checkbox->set_visible(show_phys_key);
- additional_options_container->show();
- // Update selected item in input list for keys, joybuttons and joyaxis only (since the mouse cannot be "listened" for).
- if (k.is_valid() || joyb.is_valid() || joym.is_valid()) {
- TreeItem *category = input_list_tree->get_root()->get_children();
- while (category) {
- TreeItem *input_item = category->get_children();
- // has_type this should be always true, unless the tree structure has been misconfigured.
- bool has_type = input_item->get_parent()->has_meta("__type");
- int input_type = input_item->get_parent()->get_meta("__type");
- if (!has_type) {
- return;
- }
- // If event type matches input types of this category.
- if ((k.is_valid() && input_type == INPUT_KEY) || (joyb.is_valid() && input_type == INPUT_JOY_BUTTON) || (joym.is_valid() && input_type == INPUT_JOY_MOTION)) {
- // Loop through all items of this category until one matches.
- while (input_item) {
- bool key_match = k.is_valid() && (Variant(k->get_keycode()) == input_item->get_meta("__keycode") || Variant(k->get_physical_keycode()) == input_item->get_meta("__keycode"));
- bool joyb_match = joyb.is_valid() && Variant(joyb->get_button_index()) == input_item->get_meta("__index");
- bool joym_match = joym.is_valid() && Variant(joym->get_axis()) == input_item->get_meta("__axis") && joym->get_axis_value() == (float)input_item->get_meta("__value");
- if (key_match || joyb_match || joym_match) {
- category->set_collapsed(false);
- input_item->select(0);
- input_list_tree->ensure_cursor_is_visible();
- return;
- }
- input_item = input_item->get_next();
- }
- }
- category->set_collapsed(true); // Event not in this category, so collapse;
- category = category->get_next();
- }
- }
- } else {
- // Event is not valid, reset dialog
- event = p_event;
- Vector<String> strings;
- // Reset message, promp for input according to which input types are allowed.
- String text = TTR("Perform an Input (%s).");
- if (allowed_input_types & INPUT_KEY) {
- strings.append(TTR("Key"));
- }
- // We don't check for INPUT_MOUSE_BUTTON since it is ignored in the "Listen Window Input" method.
- if (allowed_input_types & INPUT_JOY_BUTTON) {
- strings.append(TTR("Joypad Button"));
- }
- if (allowed_input_types & INPUT_JOY_MOTION) {
- strings.append(TTR("Joypad Axis"));
- }
- if (strings.size() == 0) {
- text = TTR("Input Event dialog has been misconfigured: No input types are allowed.");
- event_as_text->set_text(text);
- } else {
- String insert_text = String(", ").join(strings);
- event_as_text->set_text(vformat(text, insert_text));
- }
- additional_options_container->hide();
- input_list_tree->deselect_all();
- _update_input_list();
- }
- }
- void InputEventConfigurationDialog::_tab_selected(int p_tab) {
- Callable signal_method = callable_mp(this, &InputEventConfigurationDialog::_listen_window_input);
- if (p_tab == 0) {
- // Start Listening.
- if (!is_connected("window_input", signal_method)) {
- connect("window_input", signal_method);
- }
- } else {
- // Stop Listening.
- if (is_connected("window_input", signal_method)) {
- disconnect("window_input", signal_method);
- }
- input_list_tree->call_deferred("ensure_cursor_is_visible");
- if (input_list_tree->get_selected() == nullptr) {
- // If nothing selected, scroll to top.
- input_list_tree->scroll_to_item(input_list_tree->get_root());
- }
- }
- }
- void InputEventConfigurationDialog::_listen_window_input(const Ref<InputEvent> &p_event) {
- // Ignore if echo or not pressed
- if (p_event->is_echo() || !p_event->is_pressed()) {
- return;
- }
- // Ignore mouse
- Ref<InputEventMouse> m = p_event;
- if (m.is_valid()) {
- return;
- }
- // Check what the type is and if it is allowed.
- Ref<InputEventKey> k = p_event;
- Ref<InputEventJoypadButton> joyb = p_event;
- Ref<InputEventJoypadMotion> joym = p_event;
- int type = k.is_valid() ? INPUT_KEY : joyb.is_valid() ? INPUT_JOY_BUTTON :
- joym.is_valid() ? INPUT_JOY_MOTION :
- 0;
- if (!(allowed_input_types & type)) {
- return;
- }
- if (joym.is_valid()) {
- float axis_value = joym->get_axis_value();
- if (ABS(axis_value) < 0.9) {
- // Ignore motion below 0.9 magnitude to avoid accidental touches
- return;
- } else {
- // Always make the value 1 or -1 for display consistency
- joym->set_axis_value(SGN(axis_value));
- }
- }
- if (k.is_valid()) {
- k->set_pressed(false); // to avoid serialisation of 'pressed' property - doesn't matter for actions anyway.
- // Maintain physical keycode option state
- if (physical_key_checkbox->is_pressed()) {
- k->set_physical_keycode(k->get_keycode());
- k->set_keycode(0);
- } else {
- k->set_keycode(k->get_physical_keycode());
- k->set_physical_keycode(0);
- }
- }
- Ref<InputEventWithModifiers> mod = p_event;
- if (mod.is_valid()) {
- // Maintain store command option state
- mod->set_store_command(store_command_checkbox->is_pressed());
- mod->set_window_id(0);
- }
- _set_event(p_event);
- set_input_as_handled();
- }
- void InputEventConfigurationDialog::_search_term_updated(const String &) {
- _update_input_list();
- }
- void InputEventConfigurationDialog::_update_input_list() {
- input_list_tree->clear();
- TreeItem *root = input_list_tree->create_item();
- String search_term = input_list_search->get_text();
- bool collapse = input_list_search->get_text().is_empty();
- if (allowed_input_types & INPUT_KEY) {
- TreeItem *kb_root = input_list_tree->create_item(root);
- kb_root->set_text(0, TTR("Keyboard Keys"));
- kb_root->set_icon(0, icon_cache.keyboard);
- kb_root->set_collapsed(collapse);
- kb_root->set_meta("__type", INPUT_KEY);
- for (int i = 0; i < keycode_get_count(); i++) {
- String name = keycode_get_name_by_index(i);
- if (!search_term.is_empty() && name.findn(search_term) == -1) {
- continue;
- }
- TreeItem *item = input_list_tree->create_item(kb_root);
- item->set_text(0, name);
- item->set_meta("__keycode", keycode_get_value_by_index(i));
- }
- }
- if (allowed_input_types & INPUT_MOUSE_BUTTON) {
- TreeItem *mouse_root = input_list_tree->create_item(root);
- mouse_root->set_text(0, TTR("Mouse Buttons"));
- mouse_root->set_icon(0, icon_cache.mouse);
- mouse_root->set_collapsed(collapse);
- mouse_root->set_meta("__type", INPUT_MOUSE_BUTTON);
- int mouse_buttons[9] = { BUTTON_LEFT, BUTTON_RIGHT, BUTTON_MIDDLE, BUTTON_WHEEL_UP, BUTTON_WHEEL_DOWN, BUTTON_WHEEL_LEFT, BUTTON_WHEEL_RIGHT, BUTTON_XBUTTON1, BUTTON_XBUTTON2 };
- for (int i = 0; i < 9; i++) {
- Ref<InputEventMouseButton> mb;
- mb.instance();
- mb->set_button_index(mouse_buttons[i]);
- String desc = get_event_text(mb);
- if (!search_term.is_empty() && desc.findn(search_term) == -1) {
- continue;
- }
- TreeItem *item = input_list_tree->create_item(mouse_root);
- item->set_text(0, desc);
- item->set_meta("__index", mouse_buttons[i]);
- }
- }
- if (allowed_input_types & INPUT_JOY_BUTTON) {
- TreeItem *joyb_root = input_list_tree->create_item(root);
- joyb_root->set_text(0, TTR("Joypad Buttons"));
- joyb_root->set_icon(0, icon_cache.joypad_button);
- joyb_root->set_collapsed(collapse);
- joyb_root->set_meta("__type", INPUT_JOY_BUTTON);
- for (int i = 0; i < JOY_BUTTON_MAX; i++) {
- Ref<InputEventJoypadButton> joyb;
- joyb.instance();
- joyb->set_button_index(i);
- String desc = get_event_text(joyb);
- if (!search_term.is_empty() && desc.findn(search_term) == -1) {
- continue;
- }
- TreeItem *item = input_list_tree->create_item(joyb_root);
- item->set_text(0, desc);
- item->set_meta("__index", i);
- }
- }
- if (allowed_input_types & INPUT_JOY_MOTION) {
- TreeItem *joya_root = input_list_tree->create_item(root);
- joya_root->set_text(0, TTR("Joypad Axes"));
- joya_root->set_icon(0, icon_cache.joypad_axis);
- joya_root->set_collapsed(collapse);
- joya_root->set_meta("__type", INPUT_JOY_MOTION);
- for (int i = 0; i < JOY_AXIS_MAX * 2; i++) {
- int axis = i / 2;
- int direction = (i & 1) ? 1 : -1;
- Ref<InputEventJoypadMotion> joym;
- joym.instance();
- joym->set_axis(axis);
- joym->set_axis_value(direction);
- String desc = get_event_text(joym);
- if (!search_term.is_empty() && desc.findn(search_term) == -1) {
- continue;
- }
- TreeItem *item = input_list_tree->create_item(joya_root);
- item->set_text(0, desc);
- item->set_meta("__axis", i >> 1);
- item->set_meta("__value", (i & 1) ? 1 : -1);
- }
- }
- }
- void InputEventConfigurationDialog::_mod_toggled(bool p_checked, int p_index) {
- Ref<InputEventWithModifiers> ie = event;
- // Not event with modifiers
- if (ie.is_null()) {
- return;
- }
- if (p_index == 0) {
- ie->set_alt(p_checked);
- } else if (p_index == 1) {
- ie->set_shift(p_checked);
- } else if (p_index == 2) {
- ie->set_command(p_checked);
- } else if (p_index == 3) {
- ie->set_control(p_checked);
- } else if (p_index == 4) {
- ie->set_metakey(p_checked);
- }
- _set_event(ie);
- }
- void InputEventConfigurationDialog::_store_command_toggled(bool p_checked) {
- Ref<InputEventWithModifiers> ie = event;
- if (ie.is_valid()) {
- ie->set_store_command(p_checked);
- _set_event(ie);
- }
- if (p_checked) {
- // If storing Command, show it's checkbox and hide Control (Win/Lin) or Meta (Mac)
- #ifdef APPLE_STYLE_KEYS
- mod_checkboxes[MOD_META]->hide();
- mod_checkboxes[MOD_COMMAND]->show();
- mod_checkboxes[MOD_COMMAND]->set_text("Meta (Command)");
- #else
- mod_checkboxes[MOD_CONTROL]->hide();
- mod_checkboxes[MOD_COMMAND]->show();
- mod_checkboxes[MOD_COMMAND]->set_text("Control (Command)");
- #endif
- } else {
- // If not, hide Command, show Control and Meta.
- mod_checkboxes[MOD_COMMAND]->hide();
- mod_checkboxes[MOD_CONTROL]->show();
- mod_checkboxes[MOD_META]->show();
- }
- }
- void InputEventConfigurationDialog::_physical_keycode_toggled(bool p_checked) {
- Ref<InputEventKey> k = event;
- if (k.is_null()) {
- return;
- }
- if (p_checked) {
- k->set_physical_keycode(k->get_keycode());
- k->set_keycode(0);
- } else {
- k->set_keycode(k->get_physical_keycode());
- k->set_physical_keycode(0);
- }
- _set_event(k);
- }
- void InputEventConfigurationDialog::_input_list_item_selected() {
- TreeItem *selected = input_list_tree->get_selected();
- // Invalid tree selection - type only exists on the "category" items, which are not a valid selection.
- if (selected->has_meta("__type")) {
- return;
- }
- int input_type = selected->get_parent()->get_meta("__type");
- switch (input_type) {
- case InputEventConfigurationDialog::INPUT_KEY: {
- int kc = selected->get_meta("__keycode");
- Ref<InputEventKey> k;
- k.instance();
- if (physical_key_checkbox->is_pressed()) {
- k->set_physical_keycode(kc);
- k->set_keycode(0);
- } else {
- k->set_physical_keycode(0);
- k->set_keycode(kc);
- }
- // Maintain modifier state from checkboxes
- k->set_alt(mod_checkboxes[MOD_ALT]->is_pressed());
- k->set_shift(mod_checkboxes[MOD_SHIFT]->is_pressed());
- k->set_command(mod_checkboxes[MOD_COMMAND]->is_pressed());
- k->set_control(mod_checkboxes[MOD_CONTROL]->is_pressed());
- k->set_metakey(mod_checkboxes[MOD_META]->is_pressed());
- k->set_store_command(store_command_checkbox->is_pressed());
- _set_event(k);
- } break;
- case InputEventConfigurationDialog::INPUT_MOUSE_BUTTON: {
- int idx = selected->get_meta("__index");
- Ref<InputEventMouseButton> mb;
- mb.instance();
- mb->set_button_index(idx);
- // Maintain modifier state from checkboxes
- mb->set_alt(mod_checkboxes[MOD_ALT]->is_pressed());
- mb->set_shift(mod_checkboxes[MOD_SHIFT]->is_pressed());
- mb->set_command(mod_checkboxes[MOD_COMMAND]->is_pressed());
- mb->set_control(mod_checkboxes[MOD_CONTROL]->is_pressed());
- mb->set_metakey(mod_checkboxes[MOD_META]->is_pressed());
- mb->set_store_command(store_command_checkbox->is_pressed());
- _set_event(mb);
- } break;
- case InputEventConfigurationDialog::INPUT_JOY_BUTTON: {
- int idx = selected->get_meta("__index");
- Ref<InputEventJoypadButton> jb = InputEventJoypadButton::create_reference(idx);
- _set_event(jb);
- } break;
- case InputEventConfigurationDialog::INPUT_JOY_MOTION: {
- int axis = selected->get_meta("__axis");
- int value = selected->get_meta("__value");
- Ref<InputEventJoypadMotion> jm;
- jm.instance();
- jm->set_axis(axis);
- jm->set_axis_value(value);
- _set_event(jm);
- } break;
- default:
- break;
- }
- }
- void InputEventConfigurationDialog::_set_current_device(int i_device) {
- device_id_option->select(i_device + 1);
- }
- int InputEventConfigurationDialog::_get_current_device() const {
- return device_id_option->get_selected() - 1;
- }
- String InputEventConfigurationDialog::_get_device_string(int i_device) const {
- if (i_device == InputMap::ALL_DEVICES) {
- return TTR("All Devices");
- }
- return TTR("Device") + " " + itos(i_device);
- }
- void InputEventConfigurationDialog::_notification(int p_what) {
- switch (p_what) {
- case NOTIFICATION_ENTER_TREE:
- case NOTIFICATION_THEME_CHANGED: {
- input_list_search->set_right_icon(input_list_search->get_theme_icon("Search", "EditorIcons"));
- physical_key_checkbox->set_icon(get_theme_icon("KeyboardPhysical", "EditorIcons"));
- icon_cache.keyboard = get_theme_icon("Keyboard", "EditorIcons");
- icon_cache.mouse = get_theme_icon("Mouse", "EditorIcons");
- icon_cache.joypad_button = get_theme_icon("JoyButton", "EditorIcons");
- icon_cache.joypad_axis = get_theme_icon("JoyAxis", "EditorIcons");
- _update_input_list();
- } break;
- default:
- break;
- }
- }
- void InputEventConfigurationDialog::popup_and_configure(const Ref<InputEvent> &p_event) {
- if (p_event.is_valid()) {
- _set_event(p_event);
- } else {
- // Clear Event
- _set_event(p_event);
- // Clear Checkbox Values
- for (int i = 0; i < MOD_MAX; i++) {
- mod_checkboxes[i]->set_pressed(false);
- }
- physical_key_checkbox->set_pressed(false);
- store_command_checkbox->set_pressed(true);
- _set_current_device(0);
- // Switch to "Listen" tab
- tab_container->set_current_tab(0);
- }
- popup_centered();
- }
- Ref<InputEvent> InputEventConfigurationDialog::get_event() const {
- return event;
- }
- void InputEventConfigurationDialog::set_allowed_input_types(int p_type_masks) {
- allowed_input_types = p_type_masks;
- }
- InputEventConfigurationDialog::InputEventConfigurationDialog() {
- allowed_input_types = INPUT_KEY | INPUT_MOUSE_BUTTON | INPUT_JOY_BUTTON | INPUT_JOY_MOTION;
- set_title("Event Configuration");
- set_min_size(Size2i(550 * EDSCALE, 0)); // Min width
- VBoxContainer *main_vbox = memnew(VBoxContainer);
- add_child(main_vbox);
- tab_container = memnew(TabContainer);
- tab_container->set_tab_align(TabContainer::TabAlign::ALIGN_LEFT);
- tab_container->set_use_hidden_tabs_for_min_size(true);
- tab_container->set_v_size_flags(Control::SIZE_EXPAND_FILL);
- tab_container->connect("tab_selected", callable_mp(this, &InputEventConfigurationDialog::_tab_selected));
- main_vbox->add_child(tab_container);
- CenterContainer *cc = memnew(CenterContainer);
- cc->set_name("Listen for Input");
- event_as_text = memnew(Label);
- event_as_text->set_align(Label::ALIGN_CENTER);
- cc->add_child(event_as_text);
- tab_container->add_child(cc);
- // List of all input options to manually select from.
- VBoxContainer *manual_vbox = memnew(VBoxContainer);
- manual_vbox->set_name("Manual Selection");
- manual_vbox->set_v_size_flags(Control::SIZE_EXPAND_FILL);
- tab_container->add_child(manual_vbox);
- input_list_search = memnew(LineEdit);
- input_list_search->set_h_size_flags(Control::SIZE_EXPAND_FILL);
- input_list_search->set_placeholder(TTR("Filter Inputs"));
- input_list_search->set_clear_button_enabled(true);
- input_list_search->connect("text_changed", callable_mp(this, &InputEventConfigurationDialog::_search_term_updated));
- manual_vbox->add_child(input_list_search);
- input_list_tree = memnew(Tree);
- input_list_tree->set_custom_minimum_size(Size2(0, 100 * EDSCALE)); // Min height for tree
- input_list_tree->connect("item_selected", callable_mp(this, &InputEventConfigurationDialog::_input_list_item_selected));
- input_list_tree->set_v_size_flags(Control::SIZE_EXPAND_FILL);
- manual_vbox->add_child(input_list_tree);
- input_list_tree->set_hide_root(true);
- input_list_tree->set_columns(1);
- _update_input_list();
- // Additional Options
- additional_options_container = memnew(VBoxContainer);
- additional_options_container->hide();
- Label *opts_label = memnew(Label);
- opts_label->set_text("Additional Options");
- additional_options_container->add_child(opts_label);
- // Device Selection
- device_container = memnew(HBoxContainer);
- device_container->set_h_size_flags(Control::SIZE_EXPAND_FILL);
- Label *device_label = memnew(Label);
- device_label->set_text("Device:");
- device_container->add_child(device_label);
- device_id_option = memnew(OptionButton);
- device_id_option->set_h_size_flags(Control::SIZE_EXPAND_FILL);
- device_container->add_child(device_id_option);
- for (int i = -1; i < 8; i++) {
- device_id_option->add_item(_get_device_string(i));
- }
- _set_current_device(0);
- device_container->hide();
- additional_options_container->add_child(device_container);
- // Modifier Selection
- mod_container = memnew(HBoxContainer);
- for (int i = 0; i < MOD_MAX; i++) {
- String name = mods[i];
- mod_checkboxes[i] = memnew(CheckBox);
- mod_checkboxes[i]->connect("toggled", callable_mp(this, &InputEventConfigurationDialog::_mod_toggled), varray(i));
- mod_checkboxes[i]->set_text(name);
- mod_container->add_child(mod_checkboxes[i]);
- }
- mod_container->add_child(memnew(VSeparator));
- store_command_checkbox = memnew(CheckBox);
- store_command_checkbox->connect("toggled", callable_mp(this, &InputEventConfigurationDialog::_store_command_toggled));
- store_command_checkbox->set_pressed(true);
- store_command_checkbox->set_text(TTR("Store Command"));
- #ifdef APPLE_STYLE_KEYS
- store_command_checkbox->set_tooltip(TTR("Toggles between serializing 'command' and 'meta'. Used for compatibility with Windows/Linux style keyboard."));
- #else
- store_command_checkbox->set_tooltip(TTR("Toggles between serializing 'command' and 'control'. Used for compatibility with Apple Style keyboards."));
- #endif
- mod_container->add_child(store_command_checkbox);
- mod_container->hide();
- additional_options_container->add_child(mod_container);
- // Physical Key Checkbox
- physical_key_checkbox = memnew(CheckBox);
- physical_key_checkbox->set_text(TTR("Use Physical Keycode"));
- physical_key_checkbox->set_tooltip(TTR("Stores the physical position of the key on the keyboard rather than the keys value. Used for compatibility with non-latin layouts."));
- physical_key_checkbox->connect("toggled", callable_mp(this, &InputEventConfigurationDialog::_physical_keycode_toggled));
- physical_key_checkbox->hide();
- additional_options_container->add_child(physical_key_checkbox);
- main_vbox->add_child(additional_options_container);
- // Default to first tab
- tab_container->set_current_tab(0);
- }
- /////////////////////////////////////////
- static bool _is_action_name_valid(const String &p_name) {
- const char32_t *cstr = p_name.get_data();
- for (int i = 0; cstr[i]; i++) {
- if (cstr[i] == '/' || cstr[i] == ':' || cstr[i] == '"' ||
- cstr[i] == '=' || cstr[i] == '\\' || cstr[i] < 32) {
- return false;
- }
- }
- return true;
- }
- void ActionMapEditor::_event_config_confirmed() {
- Ref<InputEvent> ev = event_config_dialog->get_event();
- Dictionary new_action = current_action.duplicate();
- Array events = new_action["events"];
- if (current_action_event_index == -1) {
- // Add new event
- events.push_back(ev);
- } else {
- // Edit existing event
- events[current_action_event_index] = ev;
- }
- new_action["events"] = events;
- emit_signal("action_edited", current_action_name, new_action);
- }
- void ActionMapEditor::_add_action_pressed() {
- _add_action(add_edit->get_text());
- }
- void ActionMapEditor::_add_action(const String &p_name) {
- if (!allow_editing_actions) {
- return;
- }
- if (p_name == "" || !_is_action_name_valid(p_name)) {
- show_message(TTR("Invalid action name. it cannot be.is_empty()() nor contain '/', ':', '=', '\\' or '\"'"));
- return;
- }
- add_edit->clear();
- emit_signal("action_added", p_name);
- }
- void ActionMapEditor::_action_edited() {
- if (!allow_editing_actions) {
- return;
- }
- TreeItem *ti = action_tree->get_edited();
- if (!ti) {
- return;
- }
- if (action_tree->get_selected_column() == 0) {
- // Name Edited
- String new_name = ti->get_text(0);
- String old_name = ti->get_meta("__name");
- if (new_name == old_name) {
- return;
- }
- if (new_name == "" || !_is_action_name_valid(new_name)) {
- ti->set_text(0, old_name);
- show_message(TTR("Invalid action name. it cannot be.is_empty()() nor contain '/', ':', '=', '\\' or '\"'"));
- return;
- }
- emit_signal("action_renamed", old_name, new_name);
- } else if (action_tree->get_selected_column() == 1) {
- // Deadzone Edited
- String name = ti->get_meta("__name");
- Dictionary old_action = ti->get_meta("__action");
- Dictionary new_action = old_action.duplicate();
- new_action["deadzone"] = ti->get_range(1);
- // Call deferred so that input can finish propagating through tree, allowing re-making of tree to occur.
- call_deferred("emit_signal", "action_edited", name, new_action);
- }
- }
- void ActionMapEditor::_tree_button_pressed(Object *p_item, int p_column, int p_id) {
- ItemButton option = (ItemButton)p_id;
- TreeItem *item = Object::cast_to<TreeItem>(p_item);
- if (!item) {
- return;
- }
- switch (option) {
- case ActionMapEditor::BUTTON_ADD_EVENT: {
- current_action = item->get_meta("__action");
- current_action_name = item->get_meta("__name");
- current_action_event_index = -1;
- event_config_dialog->popup_and_configure();
- } break;
- case ActionMapEditor::BUTTON_EDIT_EVENT: {
- // Action and Action name is located on the parent of the event.
- current_action = item->get_parent()->get_meta("__action");
- current_action_name = item->get_parent()->get_meta("__name");
- current_action_event_index = item->get_meta("__index");
- Ref<InputEvent> ie = item->get_meta("__event");
- if (ie.is_valid()) {
- event_config_dialog->popup_and_configure(ie);
- }
- } break;
- case ActionMapEditor::BUTTON_REMOVE_ACTION: {
- if (!allow_editing_actions) {
- break;
- }
- // Send removed action name
- String name = item->get_meta("__name");
- emit_signal("action_removed", name);
- } break;
- case ActionMapEditor::BUTTON_REMOVE_EVENT: {
- // Remove event and send updated action
- Dictionary action = item->get_parent()->get_meta("__action");
- String action_name = item->get_parent()->get_meta("__name");
- int event_index = item->get_meta("__index");
- Array events = action["events"];
- events.remove(event_index);
- action["events"] = events;
- emit_signal("action_edited", action_name, action);
- } break;
- default:
- break;
- }
- }
- void ActionMapEditor::_tree_item_activated() {
- TreeItem *item = action_tree->get_selected();
- if (!item || !item->has_meta("__event")) {
- return;
- }
- _tree_button_pressed(item, 2, BUTTON_EDIT_EVENT);
- }
- void ActionMapEditor::set_show_uneditable(bool p_show) {
- show_uneditable = p_show;
- show_uneditable_actions_checkbox->set_pressed(p_show);
- // Prevent unnecessary updates of action list when cache is.is_empty()().
- if (!actions_cache.is_empty()) {
- update_action_list();
- }
- }
- void ActionMapEditor::_search_term_updated(const String &) {
- update_action_list();
- }
- Variant ActionMapEditor::get_drag_data_fw(const Point2 &p_point, Control *p_from) {
- TreeItem *selected = action_tree->get_selected();
- if (!selected) {
- return Variant();
- }
- String name = selected->get_text(0);
- Label *label = memnew(Label(name));
- label->set_modulate(Color(1, 1, 1, 1.0f));
- action_tree->set_drag_preview(label);
- Dictionary drag_data;
- if (selected->has_meta("__action")) {
- drag_data["input_type"] = "action";
- }
- if (selected->has_meta("__event")) {
- drag_data["input_type"] = "event";
- }
- action_tree->set_drop_mode_flags(Tree::DROP_MODE_INBETWEEN);
- return drag_data;
- }
- bool ActionMapEditor::can_drop_data_fw(const Point2 &p_point, const Variant &p_data, Control *p_from) const {
- Dictionary d = p_data;
- if (!d.has("input_type")) {
- return false;
- }
- TreeItem *selected = action_tree->get_selected();
- TreeItem *item = action_tree->get_item_at_position(p_point);
- if (!selected || !item || item == selected) {
- return false;
- }
- // Don't allow moving an action in-between events.
- if (d["input_type"] == "action" && item->has_meta("__event")) {
- return false;
- }
- // Don't allow moving an event to a different action.
- if (d["input_type"] == "event" && item->get_parent() != selected->get_parent()) {
- return false;
- }
- return true;
- }
- void ActionMapEditor::drop_data_fw(const Point2 &p_point, const Variant &p_data, Control *p_from) {
- if (!can_drop_data_fw(p_point, p_data, p_from)) {
- return;
- }
- TreeItem *selected = action_tree->get_selected();
- TreeItem *target = action_tree->get_item_at_position(p_point);
- bool drop_above = action_tree->get_drop_section_at_position(p_point) == -1;
- if (!target) {
- return;
- }
- Dictionary d = p_data;
- if (d["input_type"] == "action") {
- // Change action order.
- String relative_to = target->get_meta("__name");
- String action_name = selected->get_meta("__name");
- emit_signal("action_reordered", action_name, relative_to, drop_above);
- } else if (d["input_type"] == "event") {
- // Change event order
- int current_index = selected->get_meta("__index");
- int target_index = target->get_meta("__index");
- // Construct new events array.
- Dictionary new_action = selected->get_parent()->get_meta("__action");
- Array events = new_action["events"];
- Array new_events;
- // The following method was used to perform the array changes since `remove` followed by `insert` was not working properly at time of writing.
- // Loop thought existing events
- for (int i = 0; i < events.size(); i++) {
- // If you come across the current index, just skip it, as it has been moved.
- if (i == current_index) {
- continue;
- } else if (i == target_index) {
- // We are at the target index. If drop above, add selected event there first, then target, so moved event goes on top.
- if (drop_above) {
- new_events.push_back(events[current_index]);
- new_events.push_back(events[target_index]);
- } else {
- new_events.push_back(events[target_index]);
- new_events.push_back(events[current_index]);
- }
- } else {
- new_events.push_back(events[i]);
- }
- }
- new_action["events"] = new_events;
- emit_signal("action_edited", selected->get_parent()->get_meta("__name"), new_action);
- }
- }
- void ActionMapEditor::_notification(int p_what) {
- switch (p_what) {
- case NOTIFICATION_ENTER_TREE:
- case NOTIFICATION_THEME_CHANGED: {
- action_list_search->set_right_icon(get_theme_icon("Search", "EditorIcons"));
- } break;
- default:
- break;
- }
- }
- void ActionMapEditor::_bind_methods() {
- ClassDB::bind_method(D_METHOD("get_drag_data_fw"), &ActionMapEditor::get_drag_data_fw);
- ClassDB::bind_method(D_METHOD("can_drop_data_fw"), &ActionMapEditor::can_drop_data_fw);
- ClassDB::bind_method(D_METHOD("drop_data_fw"), &ActionMapEditor::drop_data_fw);
- ADD_SIGNAL(MethodInfo("action_added", PropertyInfo(Variant::STRING, "name")));
- ADD_SIGNAL(MethodInfo("action_edited", PropertyInfo(Variant::STRING, "name"), PropertyInfo(Variant::DICTIONARY, "new_action")));
- ADD_SIGNAL(MethodInfo("action_removed", PropertyInfo(Variant::STRING, "name")));
- ADD_SIGNAL(MethodInfo("action_renamed", PropertyInfo(Variant::STRING, "old_name"), PropertyInfo(Variant::STRING, "new_name")));
- ADD_SIGNAL(MethodInfo("action_reordered", PropertyInfo(Variant::STRING, "action_name"), PropertyInfo(Variant::STRING, "relative_to"), PropertyInfo(Variant::BOOL, "before")));
- }
- LineEdit *ActionMapEditor::get_search_box() const {
- return action_list_search;
- }
- InputEventConfigurationDialog *ActionMapEditor::get_configuration_dialog() {
- return event_config_dialog;
- }
- void ActionMapEditor::update_action_list(const Vector<ActionInfo> &p_action_infos) {
- if (!p_action_infos.is_empty()) {
- actions_cache = p_action_infos;
- }
- action_tree->clear();
- TreeItem *root = action_tree->create_item();
- int uneditable_count = 0;
- for (int i = 0; i < actions_cache.size(); i++) {
- ActionInfo action_info = actions_cache[i];
- if (!action_info.editable) {
- uneditable_count++;
- }
- String search_term = action_list_search->get_text();
- if (!search_term.is_empty() && action_info.name.findn(search_term) == -1) {
- continue;
- }
- if (!action_info.editable && !show_uneditable) {
- continue;
- }
- const Array events = action_info.action["events"];
- const Variant deadzone = action_info.action["deadzone"];
- // Update Tree...
- TreeItem *action_item = action_tree->create_item(root);
- action_item->set_meta("__action", action_info.action);
- action_item->set_meta("__name", action_info.name);
- // First Column - Action Name
- action_item->set_text(0, action_info.name);
- action_item->set_editable(0, action_info.editable);
- action_item->set_icon(0, action_info.icon);
- // Second Column - Deadzone
- action_item->set_editable(1, true);
- action_item->set_cell_mode(1, TreeItem::CELL_MODE_RANGE);
- action_item->set_range_config(1, 0.0, 1.0, 0.01);
- action_item->set_range(1, deadzone);
- // Third column - buttons
- action_item->add_button(2, action_tree->get_theme_icon("Add", "EditorIcons"), BUTTON_ADD_EVENT, false, TTR("Add Event"));
- action_item->add_button(2, action_tree->get_theme_icon("Remove", "EditorIcons"), BUTTON_REMOVE_ACTION, !action_info.editable, action_info.editable ? "Remove Action" : "Cannot Remove Action");
- action_item->set_custom_bg_color(0, action_tree->get_theme_color("prop_subsection", "Editor"));
- action_item->set_custom_bg_color(1, action_tree->get_theme_color("prop_subsection", "Editor"));
- for (int evnt_idx = 0; evnt_idx < events.size(); evnt_idx++) {
- Ref<InputEvent> event = events[evnt_idx];
- if (event.is_null()) {
- continue;
- }
- TreeItem *event_item = action_tree->create_item(action_item);
- // First Column - Text
- event_item->set_text(0, event_config_dialog->get_event_text(event)); // Need to us the special description for JoypadMotion here, so don't use as_text() directly.
- event_item->set_meta("__event", event);
- event_item->set_meta("__index", evnt_idx);
- // Third Column - Buttons
- event_item->add_button(2, action_tree->get_theme_icon("Edit", "EditorIcons"), BUTTON_EDIT_EVENT, false, TTR("Edit Event"));
- event_item->add_button(2, action_tree->get_theme_icon("Remove", "EditorIcons"), BUTTON_REMOVE_EVENT, false, TTR("Remove Event"));
- event_item->set_button_color(2, 0, Color(1, 1, 1, 0.75));
- event_item->set_button_color(2, 1, Color(1, 1, 1, 0.75));
- }
- }
- }
- void ActionMapEditor::show_message(const String &p_message) {
- message->set_text(p_message);
- message->popup_centered(Size2(300, 100) * EDSCALE);
- }
- void ActionMapEditor::set_allow_editing_actions(bool p_allow) {
- allow_editing_actions = p_allow;
- add_hbox->set_visible(p_allow);
- }
- void ActionMapEditor::set_toggle_editable_label(const String &p_label) {
- show_uneditable_actions_checkbox->set_text(p_label);
- }
- void ActionMapEditor::use_external_search_box(LineEdit *p_searchbox) {
- memdelete(action_list_search);
- action_list_search = p_searchbox;
- action_list_search->connect("text_changed", callable_mp(this, &ActionMapEditor::_search_term_updated));
- }
- ActionMapEditor::ActionMapEditor() {
- allow_editing_actions = true;
- show_uneditable = true;
- // Main Vbox Container
- VBoxContainer *main_vbox = memnew(VBoxContainer);
- main_vbox->set_anchors_and_offsets_preset(PRESET_WIDE);
- add_child(main_vbox);
- HBoxContainer *top_hbox = memnew(HBoxContainer);
- main_vbox->add_child(top_hbox);
- action_list_search = memnew(LineEdit);
- action_list_search->set_h_size_flags(Control::SIZE_EXPAND_FILL);
- action_list_search->set_placeholder(TTR("Filter Actions"));
- action_list_search->set_clear_button_enabled(true);
- action_list_search->connect("text_changed", callable_mp(this, &ActionMapEditor::_search_term_updated));
- top_hbox->add_child(action_list_search);
- show_uneditable_actions_checkbox = memnew(CheckBox);
- show_uneditable_actions_checkbox->set_pressed(false);
- show_uneditable_actions_checkbox->set_text(TTR("Show Uneditable Actions"));
- show_uneditable_actions_checkbox->connect("toggled", callable_mp(this, &ActionMapEditor::set_show_uneditable));
- top_hbox->add_child(show_uneditable_actions_checkbox);
- // Adding Action line edit + button
- add_hbox = memnew(HBoxContainer);
- add_hbox->set_h_size_flags(Control::SIZE_EXPAND_FILL);
- add_edit = memnew(LineEdit);
- add_edit->set_h_size_flags(Control::SIZE_EXPAND_FILL);
- add_edit->set_placeholder(TTR("Add New Action"));
- add_edit->set_clear_button_enabled(true);
- add_edit->connect("text_entered", callable_mp(this, &ActionMapEditor::_add_action));
- add_hbox->add_child(add_edit);
- Button *add_button = memnew(Button);
- add_button->set_text("Add");
- add_button->connect("pressed", callable_mp(this, &ActionMapEditor::_add_action_pressed));
- add_hbox->add_child(add_button);
- main_vbox->add_child(add_hbox);
- // Action Editor Tree
- action_tree = memnew(Tree);
- action_tree->set_v_size_flags(Control::SIZE_EXPAND_FILL);
- action_tree->set_columns(3);
- action_tree->set_hide_root(true);
- action_tree->set_column_titles_visible(true);
- action_tree->set_column_title(0, TTR("Action"));
- action_tree->set_column_title(1, TTR("Deadzone"));
- action_tree->set_column_expand(1, false);
- action_tree->set_column_min_width(1, 80 * EDSCALE);
- action_tree->set_column_expand(2, false);
- action_tree->set_column_min_width(2, 50 * EDSCALE);
- action_tree->connect("item_edited", callable_mp(this, &ActionMapEditor::_action_edited));
- action_tree->connect("item_activated", callable_mp(this, &ActionMapEditor::_tree_item_activated));
- action_tree->connect("button_pressed", callable_mp(this, &ActionMapEditor::_tree_button_pressed));
- main_vbox->add_child(action_tree);
- action_tree->set_drag_forwarding(this);
- // Adding event dialog
- event_config_dialog = memnew(InputEventConfigurationDialog);
- event_config_dialog->connect("confirmed", callable_mp(this, &ActionMapEditor::_event_config_confirmed));
- add_child(event_config_dialog);
- message = memnew(AcceptDialog);
- add_child(message);
- }
|