summaryrefslogtreecommitdiffstats
path: root/editor/import_dock.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'editor/import_dock.cpp')
-rw-r--r--editor/import_dock.cpp97
1 files changed, 84 insertions, 13 deletions
diff --git a/editor/import_dock.cpp b/editor/import_dock.cpp
index 824a65c443..fcd2d8f908 100644
--- a/editor/import_dock.cpp
+++ b/editor/import_dock.cpp
@@ -35,6 +35,7 @@
#include "editor/editor_resource_preview.h"
#include "editor/editor_scale.h"
#include "editor/editor_settings.h"
+#include "editor/editor_undo_redo_manager.h"
class ImportDockParameters : public Object {
GDCLASS(ImportDockParameters, Object);
@@ -157,6 +158,13 @@ void ImportDock::_add_keep_import_option(const String &p_importer_name) {
}
void ImportDock::_update_options(const String &p_path, const Ref<ConfigFile> &p_config) {
+ // Set the importer class to fetch the correct class in the XML class reference.
+ // This allows tooltips to display when hovering properties.
+ if (params->importer != nullptr) {
+ // Null check to avoid crashing if the "Keep File (No Import)" mode is selected.
+ import_opts->set_object_class(params->importer->get_class_name());
+ }
+
List<ResourceImporter::ImportOption> options;
if (params->importer.is_valid()) {
@@ -474,7 +482,6 @@ static bool _find_owners(EditorFileSystemDirectory *efsd, const String &p_path)
}
void ImportDock::_reimport_attempt() {
- bool need_restart = false;
bool used_in_resources = false;
String importer_name;
@@ -491,14 +498,15 @@ void ImportDock::_reimport_attempt() {
String imported_with = config->get_value("remap", "importer");
if (imported_with != importer_name) {
- need_restart = true;
+ need_cleanup.push_back(params->paths[i]);
if (_find_owners(EditorFileSystem::get_singleton()->get_filesystem(), params->paths[i])) {
used_in_resources = true;
}
}
}
- if (need_restart) {
+ if (!need_cleanup.is_empty() || used_in_resources) {
+ cleanup_warning->set_visible(!need_cleanup.is_empty());
label_warning->set_visible(used_in_resources);
reimport_confirm->popup_centered();
return;
@@ -507,11 +515,42 @@ void ImportDock::_reimport_attempt() {
_reimport();
}
-void ImportDock::_reimport_and_restart() {
- EditorNode::get_singleton()->save_all_scenes();
- EditorResourcePreview::get_singleton()->stop(); //don't try to re-create previews after import
+void ImportDock::_reimport_and_cleanup() {
+ HashMap<String, Ref<Resource>> old_resources;
+
+ for (const String &path : need_cleanup) {
+ Ref<Resource> res = ResourceLoader::load(path);
+ res->set_path("");
+ res->set_meta(SNAME("_skip_save_"), true);
+ old_resources[path] = res;
+ }
+
+ EditorResourcePreview::get_singleton()->stop(); // Don't try to re-create previews after import.
_reimport();
- EditorNode::get_singleton()->restart_editor();
+
+ if (need_cleanup.is_empty()) {
+ return;
+ }
+
+ // After changing resource type we need to make sure that all old instances are unloaded or replaced.
+ EditorNode::get_singleton()->push_item(nullptr);
+ EditorUndoRedoManager::get_singleton()->clear_history();
+
+ List<Ref<Resource>> external_resources;
+ ResourceCache::get_cached_resources(&external_resources);
+
+ for (const String &path : need_cleanup) {
+ Ref<Resource> old_res = old_resources[path];
+ Ref<Resource> new_res = ResourceLoader::load(path);
+
+ for (int j = 0; j < EditorNode::get_editor_data().get_edited_scene_count(); j++) {
+ _replace_resource_in_object(EditorNode::get_editor_data().get_edited_scene_root(j), old_res, new_res);
+ }
+ for (Ref<Resource> res : external_resources) {
+ _replace_resource_in_object(res.ptr(), old_res, new_res);
+ }
+ }
+ need_cleanup.clear();
}
void ImportDock::_advanced_options() {
@@ -576,11 +615,41 @@ void ImportDock::_reimport() {
_set_dirty(false);
}
+void ImportDock::_replace_resource_in_object(Object *p_object, const Ref<Resource> &old_resource, const Ref<Resource> &new_resource) {
+ ERR_FAIL_NULL(p_object);
+
+ List<PropertyInfo> props;
+ p_object->get_property_list(&props);
+
+ for (const PropertyInfo &p : props) {
+ if (p.type != Variant::OBJECT || p.hint != PROPERTY_HINT_RESOURCE_TYPE) {
+ continue;
+ }
+
+ Ref<Resource> res = p_object->get(p.name);
+ if (res.is_null()) {
+ continue;
+ }
+
+ if (res == old_resource) {
+ p_object->set(p.name, new_resource);
+ } else {
+ _replace_resource_in_object(res.ptr(), old_resource, new_resource);
+ }
+ }
+
+ Node *n = Object::cast_to<Node>(p_object);
+ if (n) {
+ for (int i = 0; i < n->get_child_count(); i++) {
+ _replace_resource_in_object(n->get_child(i), old_resource, new_resource);
+ }
+ }
+}
+
void ImportDock::_notification(int p_what) {
switch (p_what) {
case EditorSettings::NOTIFICATION_EDITOR_SETTINGS_CHANGED: {
imported->add_theme_style_override("normal", get_theme_stylebox(SNAME("normal"), SNAME("LineEdit")));
- import_opts->set_property_name_style(EditorPropertyNameProcessor::get_settings_style());
} break;
case NOTIFICATION_ENTER_TREE: {
@@ -658,9 +727,11 @@ ImportDock::ImportDock() {
import_opts = memnew(EditorInspector);
content->add_child(import_opts);
import_opts->set_v_size_flags(SIZE_EXPAND_FILL);
- import_opts->set_property_name_style(EditorPropertyNameProcessor::get_settings_style());
import_opts->connect("property_edited", callable_mp(this, &ImportDock::_property_edited));
import_opts->connect("property_toggled", callable_mp(this, &ImportDock::_property_toggled));
+ // Make it possible to display tooltips stored in the XML class reference.
+ // The object name is set when the importer changes in `_update_options()`.
+ import_opts->set_use_doc_hints(true);
hb = memnew(HBoxContainer);
content->add_child(hb);
@@ -690,13 +761,13 @@ ImportDock::ImportDock() {
advanced->connect("pressed", callable_mp(this, &ImportDock::_advanced_options));
reimport_confirm = memnew(ConfirmationDialog);
- reimport_confirm->set_ok_button_text(TTR("Save Scenes, Re-Import, and Restart"));
content->add_child(reimport_confirm);
- reimport_confirm->connect("confirmed", callable_mp(this, &ImportDock::_reimport_and_restart));
+ reimport_confirm->connect("confirmed", callable_mp(this, &ImportDock::_reimport_and_cleanup));
VBoxContainer *vbc_confirm = memnew(VBoxContainer());
- vbc_confirm->add_child(memnew(Label(TTR("Changing the type of an imported file requires editor restart."))));
- label_warning = memnew(Label(TTR("WARNING: Assets exist that use this resource, they may stop loading properly.")));
+ cleanup_warning = memnew(Label(TTR("The imported resource is currently loaded. All instances will be replaced and undo history will be cleared.")));
+ vbc_confirm->add_child(cleanup_warning);
+ label_warning = memnew(Label(TTR("WARNING: Assets exist that use this resource. They may stop loading properly after changing type.")));
vbc_confirm->add_child(label_warning);
reimport_confirm->add_child(vbc_confirm);