godot/editor/groups_editor.cpp
Trevor Davenport 39ae82623d
Use ObjectID to track cached scene groups.
This prevents a crash that could occur when using the Node pointer
would reference a Node which had been freed after node_removal.
2024-01-09 11:14:38 +01:00

859 lines
28 KiB
C++

/**************************************************************************/
/* groups_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 "groups_editor.h"
#include "editor/editor_node.h"
#include "editor/editor_scale.h"
#include "editor/editor_settings.h"
#include "editor/editor_undo_redo_manager.h"
#include "editor/gui/editor_validation_panel.h"
#include "editor/project_settings_editor.h"
#include "editor/scene_tree_dock.h"
#include "scene/gui/box_container.h"
#include "scene/gui/check_button.h"
#include "scene/gui/grid_container.h"
#include "scene/gui/label.h"
#include "scene/resources/packed_scene.h"
static bool can_edit(Node *p_node, const String &p_group) {
Node *n = p_node;
bool can_edit = true;
while (n) {
Ref<SceneState> ss = (n == EditorNode::get_singleton()->get_edited_scene()) ? n->get_scene_inherited_state() : n->get_scene_instance_state();
if (ss.is_valid()) {
int path = ss->find_node_by_path(n->get_path_to(p_node));
if (path != -1) {
if (ss->is_node_in_group(path, p_group)) {
can_edit = false;
break;
}
}
}
n = n->get_owner();
}
return can_edit;
}
struct _GroupInfoComparator {
bool operator()(const Node::GroupInfo &p_a, const Node::GroupInfo &p_b) const {
return p_a.name.operator String() < p_b.name.operator String();
}
};
void GroupsEditor::_add_scene_group(const String &p_name) {
scene_groups[p_name] = true;
}
void GroupsEditor::_remove_scene_group(const String &p_name) {
scene_groups.erase(p_name);
ProjectSettingsEditor::get_singleton()->get_group_settings()->remove_node_references(scene_root_node, p_name);
}
void GroupsEditor::_rename_scene_group(const String &p_old_name, const String &p_new_name) {
scene_groups[p_new_name] = scene_groups[p_old_name];
scene_groups.erase(p_old_name);
ProjectSettingsEditor::get_singleton()->get_group_settings()->rename_node_references(scene_root_node, p_old_name, p_new_name);
}
void GroupsEditor::_set_group_checked(const String &p_name, bool p_checked) {
TreeItem *ti = tree->get_item_with_text(p_name);
if (!ti) {
return;
}
ti->set_checked(0, p_checked);
}
bool GroupsEditor::_has_group(const String &p_name) {
return global_groups.has(p_name) || scene_groups.has(p_name);
}
void GroupsEditor::_modify_group(Object *p_item, int p_column, int p_id, MouseButton p_mouse_button) {
if (p_mouse_button != MouseButton::LEFT) {
return;
}
if (!node) {
return;
}
TreeItem *ti = Object::cast_to<TreeItem>(p_item);
if (!ti) {
return;
}
if (p_id == COPY_GROUP) {
DisplayServer::get_singleton()->clipboard_set(ti->get_text(p_column));
}
}
void GroupsEditor::_load_scene_groups(Node *p_node) {
List<Node::GroupInfo> groups;
p_node->get_groups(&groups);
for (const GroupInfo &gi : groups) {
if (!gi.persistent) {
continue;
}
if (global_groups.has(gi.name)) {
continue;
}
bool is_editable = can_edit(p_node, gi.name);
if (scene_groups.has(gi.name)) {
scene_groups[gi.name] = scene_groups[gi.name] && is_editable;
} else {
scene_groups[gi.name] = is_editable;
}
}
for (int i = 0; i < p_node->get_child_count(); i++) {
_load_scene_groups(p_node->get_child(i));
}
}
void GroupsEditor::_update_groups() {
if (!is_visible_in_tree()) {
groups_dirty = true;
return;
}
if (updating_groups) {
return;
}
updating_groups = true;
global_groups = ProjectSettings::get_singleton()->get_global_groups_list();
_load_scene_groups(scene_root_node);
for (const KeyValue<StringName, bool> &E : scene_groups) {
if (global_groups.has(E.key)) {
scene_groups.erase(E.key);
}
}
updating_groups = false;
}
void GroupsEditor::_update_tree() {
if (!is_visible_in_tree()) {
groups_dirty = true;
return;
}
if (!node) {
return;
}
if (updating_tree) {
return;
}
updating_tree = true;
tree->clear();
List<Node::GroupInfo> groups;
node->get_groups(&groups);
groups.sort_custom<_GroupInfoComparator>();
List<StringName> current_groups;
for (const Node::GroupInfo &gi : groups) {
current_groups.push_back(gi.name);
}
TreeItem *root = tree->create_item();
TreeItem *local_root = tree->create_item(root);
local_root->set_text(0, "Scene Groups");
local_root->set_icon(0, get_editor_theme_icon(SNAME("PackedScene")));
local_root->set_custom_bg_color(0, get_theme_color(SNAME("prop_subsection"), SNAME("Editor")));
local_root->set_selectable(0, false);
List<StringName> scene_keys;
for (const KeyValue<StringName, bool> &E : scene_groups) {
scene_keys.push_back(E.key);
}
scene_keys.sort_custom<NoCaseComparator>();
for (const StringName &E : scene_keys) {
if (!filter->get_text().is_subsequence_ofn(E)) {
continue;
}
TreeItem *item = tree->create_item(local_root);
item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK);
item->set_editable(0, can_edit(node, E));
item->set_checked(0, current_groups.find(E) != nullptr);
item->set_text(0, E);
item->set_meta("__local", true);
item->set_meta("__name", E);
item->set_meta("__description", "");
if (!scene_groups[E]) {
item->add_button(0, get_editor_theme_icon(SNAME("Lock")), -1, true, TTR("This group belongs to another scene and can't be edited."));
}
item->add_button(0, get_editor_theme_icon(SNAME("ActionCopy")), COPY_GROUP, false, TTR("Copy group name to clipboard."));
}
List<StringName> keys;
for (const KeyValue<StringName, String> &E : global_groups) {
keys.push_back(E.key);
}
keys.sort_custom<NoCaseComparator>();
TreeItem *global_root = tree->create_item(root);
global_root->set_text(0, "Global Groups");
global_root->set_icon(0, get_editor_theme_icon(SNAME("Environment")));
global_root->set_custom_bg_color(0, get_theme_color(SNAME("prop_subsection"), SNAME("Editor")));
global_root->set_selectable(0, false);
for (const StringName &E : keys) {
if (!filter->get_text().is_subsequence_ofn(E)) {
continue;
}
TreeItem *item = tree->create_item(global_root);
item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK);
item->set_editable(0, can_edit(node, E));
item->set_checked(0, current_groups.find(E) != nullptr);
item->set_text(0, E);
item->set_meta("__local", false);
item->set_meta("__name", E);
item->set_meta("__description", global_groups[E]);
if (!global_groups[E].is_empty()) {
item->set_tooltip_text(0, vformat("%s\n\n%s", E, global_groups[E]));
}
item->add_button(0, get_editor_theme_icon(SNAME("ActionCopy")), COPY_GROUP, false, TTR("Copy group name to clipboard."));
}
updating_tree = false;
}
void GroupsEditor::_queue_update_groups_and_tree() {
if (update_groups_and_tree_queued) {
return;
}
update_groups_and_tree_queued = true;
callable_mp(this, &GroupsEditor::_update_groups_and_tree).call_deferred();
}
void GroupsEditor::_update_groups_and_tree() {
update_groups_and_tree_queued = false;
_update_groups();
_update_tree();
}
void GroupsEditor::_update_scene_groups(const ObjectID &p_id) {
HashMap<ObjectID, HashMap<StringName, bool>>::Iterator I = scene_groups_cache.find(p_id);
if (I) {
scene_groups = I->value;
scene_groups_cache.remove(I);
} else {
scene_groups = HashMap<StringName, bool>();
}
}
void GroupsEditor::_cache_scene_groups(const ObjectID &p_id) {
const int edited_scene_count = EditorNode::get_editor_data().get_edited_scene_count();
for (int i = 0; i < edited_scene_count; i++) {
Node *edited_scene_root = EditorNode::get_editor_data().get_edited_scene_root(i);
if (edited_scene_root && p_id == edited_scene_root->get_instance_id()) {
scene_groups_cache[p_id] = scene_groups_for_caching;
break;
}
}
}
void GroupsEditor::set_current(Node *p_node) {
if (node == p_node) {
return;
}
node = p_node;
if (!node) {
return;
}
if (scene_tree->get_edited_scene_root() != scene_root_node) {
scene_root_node = scene_tree->get_edited_scene_root();
_update_scene_groups(scene_root_node->get_instance_id());
_update_groups();
}
_update_tree();
}
void GroupsEditor::_item_edited() {
TreeItem *ti = tree->get_edited();
if (!ti) {
return;
}
String name = ti->get_text(0);
EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();
if (ti->is_checked(0)) {
undo_redo->create_action(TTR("Add to Group"));
undo_redo->add_do_method(node, "add_to_group", name, true);
undo_redo->add_undo_method(node, "remove_from_group", name);
undo_redo->add_do_method(this, "_set_group_checked", name, true);
undo_redo->add_undo_method(this, "_set_group_checked", name, false);
// To force redraw of scene tree.
undo_redo->add_do_method(SceneTreeDock::get_singleton()->get_tree_editor(), "update_tree");
undo_redo->add_undo_method(SceneTreeDock::get_singleton()->get_tree_editor(), "update_tree");
undo_redo->commit_action();
} else {
undo_redo->create_action(TTR("Remove from Group"));
undo_redo->add_do_method(node, "remove_from_group", name);
undo_redo->add_undo_method(node, "add_to_group", name, true);
undo_redo->add_do_method(this, "_set_group_checked", name, false);
undo_redo->add_undo_method(this, "_set_group_checked", name, true);
// To force redraw of scene tree.
undo_redo->add_do_method(SceneTreeDock::get_singleton()->get_tree_editor(), "update_tree");
undo_redo->add_undo_method(SceneTreeDock::get_singleton()->get_tree_editor(), "update_tree");
undo_redo->commit_action();
}
}
void GroupsEditor::_notification(int p_what) {
switch (p_what) {
case NOTIFICATION_READY: {
get_tree()->connect("node_added", callable_mp(this, &GroupsEditor::_load_scene_groups));
get_tree()->connect("node_removed", callable_mp(this, &GroupsEditor::_node_removed));
} break;
case NOTIFICATION_THEME_CHANGED: {
filter->set_right_icon(get_editor_theme_icon("Search"));
add->set_icon(get_editor_theme_icon("Add"));
} break;
case NOTIFICATION_VISIBILITY_CHANGED: {
if (groups_dirty && is_visible_in_tree()) {
groups_dirty = false;
_update_groups_and_tree();
}
} break;
}
}
void GroupsEditor::_menu_id_pressed(int p_id) {
TreeItem *ti = tree->get_selected();
if (!ti) {
return;
}
bool is_local = ti->get_meta("__local");
String group_name = ti->get_meta("__name");
switch (p_id) {
case DELETE_GROUP: {
if (!is_local || scene_groups[group_name]) {
_show_remove_group_dialog();
}
} break;
case RENAME_GROUP: {
if (!is_local || scene_groups[group_name]) {
_show_rename_group_dialog();
}
} break;
case CONVERT_GROUP: {
String description = ti->get_meta("__description");
String property_name = GLOBAL_GROUP_PREFIX + group_name;
EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();
if (is_local) {
undo_redo->create_action(TTR("Convert to Global Group"));
undo_redo->add_do_property(ProjectSettings::get_singleton(), property_name, "");
undo_redo->add_undo_property(ProjectSettings::get_singleton(), property_name, Variant());
undo_redo->add_do_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_undo_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_undo_method(this, "_add_scene_group", group_name);
undo_redo->add_do_method(this, "_update_groups_and_tree");
undo_redo->add_undo_method(this, "_update_groups_and_tree");
undo_redo->commit_action();
} else {
undo_redo->create_action(TTR("Convert to Scene Group"));
undo_redo->add_do_property(ProjectSettings::get_singleton(), property_name, Variant());
undo_redo->add_undo_property(ProjectSettings::get_singleton(), property_name, description);
undo_redo->add_do_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_undo_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_do_method(this, "_add_scene_group", group_name);
undo_redo->add_do_method(this, "_update_groups_and_tree");
undo_redo->add_undo_method(this, "_update_groups_and_tree");
undo_redo->commit_action();
}
} break;
}
}
void GroupsEditor::_item_mouse_selected(const Vector2 &p_pos, MouseButton p_mouse_button) {
TreeItem *ti = tree->get_selected();
if (!ti) {
return;
}
if (p_mouse_button == MouseButton::LEFT) {
callable_mp(this, &GroupsEditor::_item_edited).call_deferred();
} else if (p_mouse_button == MouseButton::RIGHT) {
// Restore the previous state after clicking RMB.
if (ti->is_editable(0)) {
ti->set_checked(0, !ti->is_checked(0));
}
menu->clear();
if (ti->get_meta("__local")) {
menu->add_icon_item(get_editor_theme_icon(SNAME("Environment")), TTR("Convert to Global Group"), CONVERT_GROUP);
} else {
menu->add_icon_item(get_editor_theme_icon(SNAME("PackedScene")), TTR("Convert to Scene Group"), CONVERT_GROUP);
}
String group_name = ti->get_meta("__name");
if (global_groups.has(group_name) || scene_groups[group_name]) {
menu->add_separator();
menu->add_icon_shortcut(get_editor_theme_icon(SNAME("Rename")), ED_GET_SHORTCUT("groups_editor/rename"), RENAME_GROUP);
menu->add_icon_shortcut(get_editor_theme_icon(SNAME("Remove")), ED_GET_SHORTCUT("groups_editor/delete"), DELETE_GROUP);
}
menu->set_position(tree->get_screen_position() + p_pos);
menu->reset_size();
menu->popup();
}
}
void GroupsEditor::_confirm_add() {
String name = add_group_name->get_text().strip_edges();
String description = add_group_description->get_text();
EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();
undo_redo->create_action(TTR("Add to Group"));
undo_redo->add_do_method(node, "add_to_group", name, true);
undo_redo->add_undo_method(node, "remove_from_group", name);
bool is_local = !global_group_button->is_pressed();
if (is_local) {
undo_redo->add_do_method(this, "_add_scene_group", name);
undo_redo->add_undo_method(this, "_remove_scene_group", name);
} else {
String property_name = GLOBAL_GROUP_PREFIX + name;
undo_redo->add_do_property(ProjectSettings::get_singleton(), property_name, description);
undo_redo->add_undo_property(ProjectSettings::get_singleton(), property_name, Variant());
undo_redo->add_do_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_undo_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_do_method(this, "_update_groups");
undo_redo->add_undo_method(this, "_update_groups");
}
undo_redo->add_do_method(this, "_update_tree");
undo_redo->add_undo_method(this, "_update_tree");
// To force redraw of scene tree.
undo_redo->add_do_method(SceneTreeDock::get_singleton()->get_tree_editor(), "update_tree");
undo_redo->add_undo_method(SceneTreeDock::get_singleton()->get_tree_editor(), "update_tree");
undo_redo->commit_action();
tree->grab_focus();
}
void GroupsEditor::_confirm_rename() {
TreeItem *ti = tree->get_selected();
if (!ti) {
return;
}
String old_name = ti->get_meta("__name");
String new_name = rename_group->get_text().strip_edges();
if (old_name == new_name) {
return;
}
EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();
undo_redo->create_action(TTR("Rename Group"));
if (!global_groups.has(old_name)) {
undo_redo->add_do_method(this, "_rename_scene_group", old_name, new_name);
undo_redo->add_undo_method(this, "_rename_scene_group", new_name, old_name);
} else {
String property_new_name = GLOBAL_GROUP_PREFIX + new_name;
String property_old_name = GLOBAL_GROUP_PREFIX + old_name;
String description = ti->get_meta("__description");
undo_redo->add_do_property(ProjectSettings::get_singleton(), property_new_name, description);
undo_redo->add_undo_property(ProjectSettings::get_singleton(), property_new_name, Variant());
undo_redo->add_do_property(ProjectSettings::get_singleton(), property_old_name, Variant());
undo_redo->add_undo_property(ProjectSettings::get_singleton(), property_old_name, description);
if (rename_check_box->is_pressed()) {
undo_redo->add_do_method(ProjectSettingsEditor::get_singleton()->get_group_settings(), "rename_references", old_name, new_name);
undo_redo->add_undo_method(ProjectSettingsEditor::get_singleton()->get_group_settings(), "rename_references", new_name, old_name);
}
undo_redo->add_do_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_undo_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_do_method(this, "_update_groups");
undo_redo->add_undo_method(this, "_update_groups");
}
undo_redo->add_do_method(this, "_update_tree");
undo_redo->add_undo_method(this, "_update_tree");
undo_redo->commit_action();
tree->grab_focus();
}
void GroupsEditor::_confirm_delete() {
TreeItem *ti = tree->get_selected();
if (!ti) {
return;
}
String name = ti->get_meta("__name");
bool is_local = ti->get_meta("__local");
EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();
undo_redo->create_action(TTR("Remove Group"));
if (is_local) {
undo_redo->add_do_method(this, "_remove_scene_group", name);
undo_redo->add_undo_method(this, "_add_scene_group", name);
} else {
String property_name = GLOBAL_GROUP_PREFIX + name;
String description = ti->get_meta("__description");
undo_redo->add_do_property(ProjectSettings::get_singleton(), property_name, Variant());
undo_redo->add_undo_property(ProjectSettings::get_singleton(), property_name, description);
if (remove_check_box->is_pressed()) {
undo_redo->add_do_method(ProjectSettingsEditor::get_singleton()->get_group_settings(), "remove_references", name);
}
undo_redo->add_do_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_undo_method(ProjectSettings::get_singleton(), "save");
undo_redo->add_do_method(this, "_update_groups");
undo_redo->add_undo_method(this, "_update_groups");
}
undo_redo->add_do_method(this, "_update_tree");
undo_redo->add_undo_method(this, "_update_tree");
undo_redo->commit_action();
tree->grab_focus();
}
void GroupsEditor::_show_add_group_dialog() {
if (!add_group_dialog) {
add_group_dialog = memnew(ConfirmationDialog);
add_group_dialog->set_title(TTR("Create New Group"));
add_group_dialog->connect("confirmed", callable_mp(this, &GroupsEditor::_confirm_add));
VBoxContainer *vbc = memnew(VBoxContainer);
add_group_dialog->add_child(vbc);
GridContainer *gc = memnew(GridContainer);
gc->set_columns(2);
vbc->add_child(gc);
Label *label_name = memnew(Label(TTR("Name:")));
label_name->set_h_size_flags(SIZE_SHRINK_BEGIN);
gc->add_child(label_name);
HBoxContainer *hbc = memnew(HBoxContainer);
hbc->set_h_size_flags(SIZE_EXPAND_FILL);
gc->add_child(hbc);
add_group_name = memnew(LineEdit);
add_group_name->set_custom_minimum_size(Size2(200 * EDSCALE, 0));
add_group_name->set_h_size_flags(SIZE_EXPAND_FILL);
hbc->add_child(add_group_name);
global_group_button = memnew(CheckButton);
global_group_button->set_text(TTR("Global"));
hbc->add_child(global_group_button);
Label *label_description = memnew(Label(TTR("Description:")));
label_name->set_h_size_flags(SIZE_SHRINK_BEGIN);
gc->add_child(label_description);
add_group_description = memnew(LineEdit);
add_group_description->set_h_size_flags(SIZE_EXPAND_FILL);
add_group_description->set_editable(false);
gc->add_child(add_group_description);
global_group_button->connect("toggled", callable_mp(add_group_description, &LineEdit::set_editable));
add_group_dialog->register_text_enter(add_group_name);
add_group_dialog->register_text_enter(add_group_description);
add_validation_panel = memnew(EditorValidationPanel);
add_validation_panel->add_line(EditorValidationPanel::MSG_ID_DEFAULT, TTR("Group name is valid."));
add_validation_panel->set_update_callback(callable_mp(this, &GroupsEditor::_check_add));
add_validation_panel->set_accept_button(add_group_dialog->get_ok_button());
add_group_name->connect("text_changed", callable_mp(add_validation_panel, &EditorValidationPanel::update).unbind(1));
vbc->add_child(add_validation_panel);
add_child(add_group_dialog);
}
add_group_name->clear();
add_group_description->clear();
global_group_button->set_pressed(false);
add_validation_panel->update();
add_group_dialog->popup_centered();
add_group_name->grab_focus();
}
void GroupsEditor::_show_rename_group_dialog() {
if (!rename_group_dialog) {
rename_group_dialog = memnew(ConfirmationDialog);
rename_group_dialog->set_title(TTR("Rename Group"));
rename_group_dialog->connect("confirmed", callable_mp(this, &GroupsEditor::_confirm_rename));
VBoxContainer *vbc = memnew(VBoxContainer);
rename_group_dialog->add_child(vbc);
HBoxContainer *hbc = memnew(HBoxContainer);
hbc->add_child(memnew(Label(TTR("Name:"))));
rename_group = memnew(LineEdit);
rename_group->set_custom_minimum_size(Size2(300 * EDSCALE, 1));
hbc->add_child(rename_group);
vbc->add_child(hbc);
rename_group_dialog->register_text_enter(rename_group);
rename_validation_panel = memnew(EditorValidationPanel);
rename_validation_panel->add_line(EditorValidationPanel::MSG_ID_DEFAULT, TTR("Group name is valid."));
rename_validation_panel->set_update_callback(callable_mp(this, &GroupsEditor::_check_rename));
rename_validation_panel->set_accept_button(rename_group_dialog->get_ok_button());
rename_group->connect("text_changed", callable_mp(rename_validation_panel, &EditorValidationPanel::update).unbind(1));
vbc->add_child(rename_validation_panel);
rename_check_box = memnew(CheckBox);
rename_check_box->set_text(TTR("Rename references in all scenes"));
vbc->add_child(rename_check_box);
add_child(rename_group_dialog);
}
TreeItem *ti = tree->get_selected();
if (!ti) {
return;
}
bool is_global = !ti->get_meta("__local");
rename_check_box->set_visible(is_global);
rename_check_box->set_pressed(false);
String name = ti->get_meta("__name");
rename_group->set_text(name);
rename_group_dialog->set_meta("__name", name);
rename_validation_panel->update();
rename_group_dialog->reset_size();
rename_group_dialog->popup_centered();
rename_group->select_all();
rename_group->grab_focus();
}
void GroupsEditor::_show_remove_group_dialog() {
if (!remove_group_dialog) {
remove_group_dialog = memnew(ConfirmationDialog);
remove_group_dialog->connect("confirmed", callable_mp(this, &GroupsEditor::_confirm_delete));
VBoxContainer *vbox = memnew(VBoxContainer);
remove_label = memnew(Label);
vbox->add_child(remove_label);
remove_check_box = memnew(CheckBox);
remove_check_box->set_text(TTR("Delete references from all scenes"));
vbox->add_child(remove_check_box);
remove_group_dialog->add_child(vbox);
add_child(remove_group_dialog);
}
TreeItem *ti = tree->get_selected();
if (!ti) {
return;
}
bool is_global = !ti->get_meta("__local");
remove_check_box->set_visible(is_global);
remove_check_box->set_pressed(false);
remove_label->set_text(vformat(TTR("Delete group \"%s\" and all its references?"), ti->get_text(0)));
remove_group_dialog->reset_size();
remove_group_dialog->popup_centered();
}
void GroupsEditor::_check_add() {
String group_name = add_group_name->get_text().strip_edges();
_validate_name(group_name, add_validation_panel);
}
void GroupsEditor::_check_rename() {
String group_name = rename_group->get_text().strip_edges();
String old_name = rename_group_dialog->get_meta("__name");
if (group_name == old_name) {
return;
}
_validate_name(group_name, rename_validation_panel);
}
void GroupsEditor::_validate_name(const String &p_name, EditorValidationPanel *p_validation_panel) {
if (p_name.is_empty()) {
p_validation_panel->set_message(EditorValidationPanel::MSG_ID_DEFAULT, TTR("Group can't be empty."), EditorValidationPanel::MSG_ERROR);
} else if (_has_group(p_name)) {
p_validation_panel->set_message(EditorValidationPanel::MSG_ID_DEFAULT, TTR("Group already exists."), EditorValidationPanel::MSG_ERROR);
}
}
void GroupsEditor::_groups_gui_input(Ref<InputEvent> p_event) {
Ref<InputEventKey> key = p_event;
if (key.is_valid() && key->is_pressed() && !key->is_echo()) {
if (ED_IS_SHORTCUT("groups_editor/delete", p_event)) {
_menu_id_pressed(DELETE_GROUP);
} else if (ED_IS_SHORTCUT("groups_editor/rename", p_event)) {
_menu_id_pressed(RENAME_GROUP);
} else {
return;
}
accept_event();
}
}
void GroupsEditor::_bind_methods() {
ClassDB::bind_method("_update_tree", &GroupsEditor::_update_tree);
ClassDB::bind_method("_update_groups", &GroupsEditor::_update_groups);
ClassDB::bind_method("_update_groups_and_tree", &GroupsEditor::_update_groups_and_tree);
ClassDB::bind_method("_add_scene_group", &GroupsEditor::_add_scene_group);
ClassDB::bind_method("_rename_scene_group", &GroupsEditor::_rename_scene_group);
ClassDB::bind_method("_remove_scene_group", &GroupsEditor::_remove_scene_group);
ClassDB::bind_method("_set_group_checked", &GroupsEditor::_set_group_checked);
}
void GroupsEditor::_node_removed(Node *p_node) {
if (scene_root_node == p_node) {
scene_groups_for_caching = scene_groups;
callable_mp(this, &GroupsEditor::_cache_scene_groups).call_deferred(p_node->get_instance_id());
scene_root_node = nullptr;
}
if (scene_root_node && scene_root_node == p_node->get_owner()) {
_queue_update_groups_and_tree();
}
}
GroupsEditor::GroupsEditor() {
node = nullptr;
scene_tree = SceneTree::get_singleton();
ED_SHORTCUT("groups_editor/delete", TTR("Delete"), Key::KEY_DELETE);
ED_SHORTCUT("groups_editor/rename", TTR("Rename"), Key::F2);
ED_SHORTCUT_OVERRIDE("groups_editor/rename", "macos", Key::ENTER);
HBoxContainer *hbc = memnew(HBoxContainer);
add_child(hbc);
add = memnew(Button);
add->set_flat(true);
add->set_tooltip_text(TTR("Add a new group."));
add->connect("pressed", callable_mp(this, &GroupsEditor::_show_add_group_dialog));
hbc->add_child(add);
filter = memnew(LineEdit);
filter->set_clear_button_enabled(true);
filter->set_placeholder(TTR("Filter Groups"));
filter->set_h_size_flags(SIZE_EXPAND_FILL);
filter->connect("text_changed", callable_mp(this, &GroupsEditor::_update_tree).unbind(1));
hbc->add_child(filter);
tree = memnew(Tree);
tree->set_hide_root(true);
tree->set_v_size_flags(SIZE_EXPAND_FILL);
tree->set_allow_rmb_select(true);
tree->set_select_mode(Tree::SelectMode::SELECT_SINGLE);
tree->connect("button_clicked", callable_mp(this, &GroupsEditor::_modify_group));
tree->connect("item_mouse_selected", callable_mp(this, &GroupsEditor::_item_mouse_selected));
tree->connect("gui_input", callable_mp(this, &GroupsEditor::_groups_gui_input));
add_child(tree);
menu = memnew(PopupMenu);
menu->connect("id_pressed", callable_mp(this, &GroupsEditor::_menu_id_pressed));
tree->add_child(menu);
ProjectSettingsEditor::get_singleton()->get_group_settings()->connect("group_changed", callable_mp(this, &GroupsEditor::_update_groups_and_tree));
}
GroupsEditor::~GroupsEditor() {
}