/**************************************************************************/ /* openxr_action_map_editor.cpp */ /**************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* https://godotengine.org */ /**************************************************************************/ /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */ /* */ /* 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 "openxr_action_map_editor.h" #include "core/config/project_settings.h" #include "editor/editor_file_dialog.h" #include "editor/editor_node.h" #include "editor/editor_scale.h" #include "editor/editor_settings.h" // TODO implement redo/undo system void OpenXRActionMapEditor::_bind_methods() { ClassDB::bind_method("_add_action_set_editor", &OpenXRActionMapEditor::_add_action_set_editor); ClassDB::bind_method("_add_interaction_profile_editor", &OpenXRActionMapEditor::_add_interaction_profile_editor); ClassDB::bind_method(D_METHOD("_add_action_set", "name"), &OpenXRActionMapEditor::_add_action_set); ClassDB::bind_method(D_METHOD("_set_focus_on_action_set", "action_set"), &OpenXRActionMapEditor::_set_focus_on_action_set); ClassDB::bind_method(D_METHOD("_remove_action_set", "name"), &OpenXRActionMapEditor::_remove_action_set); ClassDB::bind_method(D_METHOD("_do_add_action_set_editor", "action_set_editor"), &OpenXRActionMapEditor::_do_add_action_set_editor); ClassDB::bind_method(D_METHOD("_do_remove_action_set_editor", "action_set_editor"), &OpenXRActionMapEditor::_do_remove_action_set_editor); ClassDB::bind_method(D_METHOD("_do_add_interaction_profile_editor", "interaction_profile_editor"), &OpenXRActionMapEditor::_do_add_interaction_profile_editor); ClassDB::bind_method(D_METHOD("_do_remove_interaction_profile_editor", "interaction_profile_editor"), &OpenXRActionMapEditor::_do_remove_interaction_profile_editor); } void OpenXRActionMapEditor::_notification(int p_what) { switch (p_what) { case NOTIFICATION_ENTER_TREE: case NOTIFICATION_THEME_CHANGED: { for (int i = 0; i < tabs->get_child_count(); i++) { Control *tab = Object::cast_to(tabs->get_child(i)); if (tab) { tab->add_theme_style_override("panel", get_theme_stylebox(SNAME("panel"), SNAME("Tree"))); } } } break; case NOTIFICATION_READY: { _create_action_sets(); _create_interaction_profiles(); } break; } } OpenXRActionSetEditor *OpenXRActionMapEditor::_add_action_set_editor(Ref p_action_set) { ERR_FAIL_COND_V(p_action_set.is_null(), nullptr); OpenXRActionSetEditor *action_set_editor = memnew(OpenXRActionSetEditor(action_map, p_action_set)); action_set_editor->connect("remove", callable_mp(this, &OpenXRActionMapEditor::_on_remove_action_set)); action_set_editor->connect("action_removed", callable_mp(this, &OpenXRActionMapEditor::_on_action_removed)); actionsets_vb->add_child(action_set_editor); return action_set_editor; } void OpenXRActionMapEditor::_create_action_sets() { if (action_map.is_valid()) { Array action_sets = action_map->get_action_sets(); for (int i = 0; i < action_sets.size(); i++) { Ref action_set = action_sets[i]; _add_action_set_editor(action_set); } } } OpenXRInteractionProfileEditorBase *OpenXRActionMapEditor::_add_interaction_profile_editor(Ref p_interaction_profile) { ERR_FAIL_COND_V(p_interaction_profile.is_null(), nullptr); String profile_path = p_interaction_profile->get_interaction_profile_path(); // need to instance the correct editor for our profile OpenXRInteractionProfileEditorBase *new_profile_editor = nullptr; if (profile_path == "placeholder_text") { // instance specific editor for this type } else { // instance generic editor new_profile_editor = memnew(OpenXRInteractionProfileEditor(action_map, p_interaction_profile)); } // now add it in.. ERR_FAIL_NULL_V(new_profile_editor, nullptr); tabs->add_child(new_profile_editor); new_profile_editor->add_theme_style_override("panel", get_theme_stylebox(SNAME("panel"), SNAME("Tree"))); tabs->set_tab_button_icon(tabs->get_tab_count() - 1, get_theme_icon(SNAME("close"), SNAME("TabBar"))); return new_profile_editor; } void OpenXRActionMapEditor::_create_interaction_profiles() { if (action_map.is_valid()) { Array new_interaction_profiles = action_map->get_interaction_profiles(); for (int i = 0; i < new_interaction_profiles.size(); i++) { Ref interaction_profile = new_interaction_profiles[i]; _add_interaction_profile_editor(interaction_profile); } } } OpenXRActionSetEditor *OpenXRActionMapEditor::_add_action_set(String p_name) { ERR_FAIL_COND_V(action_map.is_null(), nullptr); Ref new_action_set; // add our new action set new_action_set.instantiate(); new_action_set->set_name(p_name); new_action_set->set_localized_name(p_name); action_map->add_action_set(new_action_set); action_map->set_edited(true); // update our editor right away OpenXRActionSetEditor *action_set_editor = _add_action_set_editor(new_action_set); undo_redo->create_action(TTR("Add action set")); undo_redo->add_do_method(this, "_do_add_action_set_editor", action_set_editor); undo_redo->add_undo_method(this, "_do_remove_action_set_editor", action_set_editor); undo_redo->commit_action(false); return action_set_editor; } void OpenXRActionMapEditor::_remove_action_set(String p_name) { ERR_FAIL_COND(action_map.is_null()); Ref action_set = action_map->find_action_set(p_name); ERR_FAIL_COND(action_set.is_null()); for (int i = 0; i < actionsets_vb->get_child_count(); i++) { OpenXRActionSetEditor *action_set_editor = Object::cast_to(actionsets_vb->get_child(i)); if (action_set_editor && action_set_editor->get_action_set() == action_set) { _on_remove_action_set(action_set_editor); } } } void OpenXRActionMapEditor::_on_add_action_set() { ERR_FAIL_COND(action_map.is_null()); String new_name = "New"; int count = 0; while (action_map->find_action_set(new_name).is_valid()) { new_name = "New_" + itos(count++); } OpenXRActionSetEditor *new_action_set_editor = _add_action_set(new_name); // Make sure our action set is the current tab tabs->set_current_tab(0); call_deferred("_set_focus_on_action_set", new_action_set_editor); } void OpenXRActionMapEditor::_set_focus_on_action_set(OpenXRActionSetEditor *p_action_set_editor) { // Scroll down to our new entry actionsets_scroll->ensure_control_visible(p_action_set_editor); // Set focus on this entry p_action_set_editor->set_focus_on_entry(); } void OpenXRActionMapEditor::_on_remove_action_set(Object *p_action_set_editor) { ERR_FAIL_COND(action_map.is_null()); OpenXRActionSetEditor *action_set_editor = Object::cast_to(p_action_set_editor); ERR_FAIL_NULL(action_set_editor); ERR_FAIL_COND(action_set_editor->get_parent() != actionsets_vb); Ref action_set = action_set_editor->get_action_set(); ERR_FAIL_COND(action_set.is_null()); action_set_editor->remove_all_actions(); undo_redo->create_action(TTR("Remove action set")); undo_redo->add_do_method(this, "_do_remove_action_set_editor", action_set_editor); undo_redo->add_undo_method(this, "_do_add_action_set_editor", action_set_editor); undo_redo->commit_action(true); action_map->set_edited(true); } void OpenXRActionMapEditor::_on_action_removed(Ref p_action) { for (int i = 0; i < tabs->get_tab_count(); i++) { // First tab won't be an interaction profile editor, but being thorough.. OpenXRInteractionProfileEditorBase *interaction_profile_editor = Object::cast_to(tabs->get_tab_control(i)); if (interaction_profile_editor) { interaction_profile_editor->remove_all_bindings_for_action(p_action); } } } void OpenXRActionMapEditor::_on_add_interaction_profile() { ERR_FAIL_COND(action_map.is_null()); PackedStringArray already_selected; for (int i = 0; i < action_map->get_interaction_profile_count(); i++) { already_selected.push_back(action_map->get_interaction_profile(i)->get_interaction_profile_path()); } select_interaction_profile_dialog->open(already_selected); } void OpenXRActionMapEditor::_on_interaction_profile_selected(const String p_path) { ERR_FAIL_COND(action_map.is_null()); Ref new_profile; new_profile.instantiate(); new_profile->set_interaction_profile_path(p_path); action_map->add_interaction_profile(new_profile); action_map->set_edited(true); OpenXRInteractionProfileEditorBase *interaction_profile_editor = _add_interaction_profile_editor(new_profile); undo_redo->create_action(TTR("Add interaction profile")); undo_redo->add_do_method(this, "_do_add_interaction_profile_editor", interaction_profile_editor); undo_redo->add_undo_method(this, "_do_remove_interaction_profile_editor", interaction_profile_editor); undo_redo->commit_action(false); tabs->set_current_tab(tabs->get_tab_count() - 1); } void OpenXRActionMapEditor::_load_action_map(const String p_path, bool p_create_new_if_missing) { Error err = OK; action_map = ResourceLoader::load(p_path, "", ResourceFormatLoader::CACHE_MODE_IGNORE, &err); if (err != OK) { if ((err == ERR_FILE_NOT_FOUND || err == ERR_CANT_OPEN) && p_create_new_if_missing) { action_map.instantiate(); action_map->create_default_action_sets(); // Save it immediately err = ResourceSaver::save(action_map, p_path); if (err != OK) { // show warning but continue EditorNode::get_singleton()->show_warning(vformat(TTR("Error saving file %s: %s"), edited_path, error_names[err])); } } else { EditorNode::get_singleton()->show_warning(vformat(TTR("Error loading %s: %s."), edited_path, error_names[err])); edited_path = ""; header_label->set_text(""); return; } } edited_path = p_path; header_label->set_text(TTR("OpenXR Action map:") + " " + edited_path.get_file()); } void OpenXRActionMapEditor::_on_save_action_map() { Error err = ResourceSaver::save(action_map, edited_path); if (err != OK) { EditorNode::get_singleton()->show_warning(vformat(TTR("Error saving file %s: %s"), edited_path, error_names[err])); return; } // TODO should clear undo/redo history // out with the old _clear_action_map(); _create_action_sets(); _create_interaction_profiles(); } void OpenXRActionMapEditor::_on_reset_to_default_layout() { // TODO should clear undo/redo history // out with the old _clear_action_map(); // create a new one action_map.unref(); action_map.instantiate(); action_map->create_default_action_sets(); action_map->set_edited(true); _create_action_sets(); _create_interaction_profiles(); } void OpenXRActionMapEditor::_on_tabs_tab_changed(int p_tab) { } void OpenXRActionMapEditor::_on_tab_button_pressed(int p_tab) { OpenXRInteractionProfileEditorBase *interaction_profile_editor = Object::cast_to(tabs->get_tab_control(p_tab)); ERR_FAIL_NULL(interaction_profile_editor); undo_redo->create_action(TTR("Remove interaction profile")); undo_redo->add_do_method(this, "_do_remove_interaction_profile_editor", interaction_profile_editor); undo_redo->add_undo_method(this, "_do_add_interaction_profile_editor", interaction_profile_editor); undo_redo->commit_action(true); action_map->set_edited(true); } void OpenXRActionMapEditor::_do_add_action_set_editor(OpenXRActionSetEditor *p_action_set_editor) { Ref action_set = p_action_set_editor->get_action_set(); ERR_FAIL_COND(action_set.is_null()); action_map->add_action_set(action_set); actionsets_vb->add_child(p_action_set_editor); } void OpenXRActionMapEditor::_do_remove_action_set_editor(OpenXRActionSetEditor *p_action_set_editor) { Ref action_set = p_action_set_editor->get_action_set(); ERR_FAIL_COND(action_set.is_null()); actionsets_vb->remove_child(p_action_set_editor); action_map->remove_action_set(action_set); } void OpenXRActionMapEditor::_do_add_interaction_profile_editor(OpenXRInteractionProfileEditorBase *p_interaction_profile_editor) { Ref interaction_profile = p_interaction_profile_editor->get_interaction_profile(); ERR_FAIL_COND(interaction_profile.is_null()); action_map->add_interaction_profile(interaction_profile); tabs->add_child(p_interaction_profile_editor); tabs->set_tab_button_icon(tabs->get_tab_count() - 1, get_theme_icon(SNAME("close"), SNAME("TabBar"))); tabs->set_current_tab(tabs->get_tab_count() - 1); } void OpenXRActionMapEditor::_do_remove_interaction_profile_editor(OpenXRInteractionProfileEditorBase *p_interaction_profile_editor) { Ref interaction_profile = p_interaction_profile_editor->get_interaction_profile(); ERR_FAIL_COND(interaction_profile.is_null()); tabs->remove_child(p_interaction_profile_editor); action_map->remove_interaction_profile(interaction_profile); } void OpenXRActionMapEditor::open_action_map(String p_path) { EditorNode::get_singleton()->make_bottom_panel_item_visible(this); // out with the old... _clear_action_map(); // now load in our new action map _load_action_map(p_path); _create_action_sets(); _create_interaction_profiles(); } void OpenXRActionMapEditor::_clear_action_map() { while (actionsets_vb->get_child_count() > 0) { Node *child = actionsets_vb->get_child(0); actionsets_vb->remove_child(child); child->queue_free(); } for (int i = tabs->get_tab_count() - 1; i >= 0; --i) { // First tab won't be an interaction profile editor, but being thorough.. OpenXRInteractionProfileEditorBase *interaction_profile_editor = Object::cast_to(tabs->get_tab_control(i)); if (interaction_profile_editor) { tabs->remove_child(interaction_profile_editor); interaction_profile_editor->queue_free(); } } } OpenXRActionMapEditor::OpenXRActionMapEditor() { undo_redo = EditorNode::get_undo_redo(); set_custom_minimum_size(Size2(0.0, 300.0)); top_hb = memnew(HBoxContainer); add_child(top_hb); header_label = memnew(Label); header_label->set_text(String(TTR("Action Map"))); header_label->set_clip_text(true); header_label->set_h_size_flags(Control::SIZE_EXPAND_FILL); top_hb->add_child(header_label); add_action_set = memnew(Button); add_action_set->set_text(TTR("Add Action Set")); add_action_set->set_tooltip_text(TTR("Add an action set.")); add_action_set->connect("pressed", callable_mp(this, &OpenXRActionMapEditor::_on_add_action_set)); top_hb->add_child(add_action_set); add_interaction_profile = memnew(Button); add_interaction_profile->set_text(TTR("Add profile")); add_interaction_profile->set_tooltip_text(TTR("Add an interaction profile.")); add_interaction_profile->connect("pressed", callable_mp(this, &OpenXRActionMapEditor::_on_add_interaction_profile)); top_hb->add_child(add_interaction_profile); VSeparator *vseparator = memnew(VSeparator); top_hb->add_child(vseparator); save_as = memnew(Button); save_as->set_text(TTR("Save")); save_as->set_tooltip_text(TTR("Save this OpenXR action map.")); save_as->connect("pressed", callable_mp(this, &OpenXRActionMapEditor::_on_save_action_map)); top_hb->add_child(save_as); _default = memnew(Button); _default->set_text(TTR("Reset to Default")); _default->set_tooltip_text(TTR("Reset to default OpenXR action map.")); _default->connect("pressed", callable_mp(this, &OpenXRActionMapEditor::_on_reset_to_default_layout)); top_hb->add_child(_default); tabs = memnew(TabContainer); tabs->set_h_size_flags(SIZE_EXPAND_FILL); tabs->set_v_size_flags(SIZE_EXPAND_FILL); tabs->set_theme_type_variation("TabContainerOdd"); tabs->connect("tab_changed", callable_mp(this, &OpenXRActionMapEditor::_on_tabs_tab_changed)); tabs->connect("tab_button_pressed", callable_mp(this, &OpenXRActionMapEditor::_on_tab_button_pressed)); add_child(tabs); actionsets_scroll = memnew(ScrollContainer); actionsets_scroll->set_h_size_flags(SIZE_EXPAND_FILL); actionsets_scroll->set_v_size_flags(SIZE_EXPAND_FILL); actionsets_scroll->set_horizontal_scroll_mode(ScrollContainer::SCROLL_MODE_DISABLED); tabs->add_child(actionsets_scroll); actionsets_scroll->set_name(TTR("Action Sets")); actionsets_vb = memnew(VBoxContainer); actionsets_vb->set_h_size_flags(SIZE_EXPAND_FILL); actionsets_scroll->add_child(actionsets_vb); select_interaction_profile_dialog = memnew(OpenXRSelectInteractionProfileDialog); select_interaction_profile_dialog->connect("interaction_profile_selected", callable_mp(this, &OpenXRActionMapEditor::_on_interaction_profile_selected)); add_child(select_interaction_profile_dialog); // Our Action map editor is only shown if openxr is enabled in project settings // So load our action map and if it doesn't exist, create it right away. _load_action_map(GLOBAL_GET("xr/openxr/default_action_map"), true); } OpenXRActionMapEditor::~OpenXRActionMapEditor() { }