mirror of
				https://github.com/godotengine/godot.git
				synced 2025-10-30 21:21:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1167 lines
		
	
	
	
		
			39 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1167 lines
		
	
	
	
		
			39 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*************************************************************************/
 | |
| /*  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);
 | |
| 
 | |
| 		MouseButton mouse_buttons[9] = { MOUSE_BUTTON_LEFT, MOUSE_BUTTON_RIGHT, MOUSE_BUTTON_MIDDLE, MOUSE_BUTTON_WHEEL_UP, MOUSE_BUTTON_WHEEL_DOWN, MOUSE_BUTTON_WHEEL_LEFT, MOUSE_BUTTON_WHEEL_RIGHT, MOUSE_BUTTON_XBUTTON1, MOUSE_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);
 | |
| }
 | 
