diff options
Diffstat (limited to 'scene/resources/resource_format_text.cpp')
| -rw-r--r-- | scene/resources/resource_format_text.cpp | 1224 | 
1 files changed, 817 insertions, 407 deletions
| diff --git a/scene/resources/resource_format_text.cpp b/scene/resources/resource_format_text.cpp index ba39593690..85b538b1d9 100644 --- a/scene/resources/resource_format_text.cpp +++ b/scene/resources/resource_format_text.cpp @@ -5,8 +5,8 @@  /*                           GODOT ENGINE                                */  /*                      https://godotengine.org                          */  /*************************************************************************/ -/* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur.                 */ -/* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md).   */ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ +/* Copyright (c) 2014-2022 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       */ @@ -30,15 +30,19 @@  #include "resource_format_text.h" +#include "core/config/project_settings.h" +#include "core/io/dir_access.h" +#include "core/io/missing_resource.h"  #include "core/io/resource_format_binary.h" -#include "core/os/dir_access.h" -#include "core/project_settings.h"  #include "core/version.h" -//version 2: changed names for basis, aabb, Vectors, etc. -#define FORMAT_VERSION 2 +// Version 2: changed names for Basis, AABB, Vectors, etc. +// Version 3: new string ID for ext/subresources, breaks forward compat. +#define FORMAT_VERSION 3 -#include "core/os/dir_access.h" +#define BINARY_FORMAT_VERSION 4 + +#include "core/io/dir_access.h"  #include "core/version.h"  #define _printerr() ERR_PRINT(String(res_path + ":" + itos(lines) + " - Parse Error: " + error_text).utf8().get_data()); @@ -56,22 +60,23 @@ Ref<Resource> ResourceLoaderText::get_resource() {  Error ResourceLoaderText::_parse_sub_resource_dummy(DummyReadData *p_data, VariantParser::Stream *p_stream, Ref<Resource> &r_res, int &line, String &r_err_str) {  	VariantParser::Token token;  	VariantParser::get_token(p_stream, token, line, r_err_str); -	if (token.type != VariantParser::TK_NUMBER) { -		r_err_str = "Expected number (sub-resource index)"; +	if (token.type != VariantParser::TK_NUMBER && token.type != VariantParser::TK_STRING) { +		r_err_str = "Expected number (old style) or string (sub-resource index)";  		return ERR_PARSE_ERROR;  	} -	int index = token.value; +	if (p_data->no_placeholders) { +		r_res.unref(); +	} else { +		String unique_id = token.value; -	if (!p_data->resource_map.has(index)) { -		Ref<DummyResource> dr; -		dr.instance(); -		dr->set_subindex(index); -		p_data->resource_map[index] = dr; -		p_data->resource_set.insert(dr); -	} +		if (!p_data->resource_map.has(unique_id)) { +			r_err_str = "Found unique_id reference before mapping, sub-resources stored out of order in resource file"; +			return ERR_PARSE_ERROR; +		} -	r_res = p_data->resource_map[index]; +		r_res = p_data->resource_map[unique_id]; +	}  	VariantParser::get_token(p_stream, token, line, r_err_str);  	if (token.type != VariantParser::TK_PARENTHESIS_CLOSE) { @@ -85,16 +90,20 @@ Error ResourceLoaderText::_parse_sub_resource_dummy(DummyReadData *p_data, Varia  Error ResourceLoaderText::_parse_ext_resource_dummy(DummyReadData *p_data, VariantParser::Stream *p_stream, Ref<Resource> &r_res, int &line, String &r_err_str) {  	VariantParser::Token token;  	VariantParser::get_token(p_stream, token, line, r_err_str); -	if (token.type != VariantParser::TK_NUMBER) { -		r_err_str = "Expected number (sub-resource index)"; +	if (token.type != VariantParser::TK_NUMBER && token.type != VariantParser::TK_STRING) { +		r_err_str = "Expected number (old style sub-resource index) or String (ext-resource ID)";  		return ERR_PARSE_ERROR;  	} -	int id = token.value; +	if (p_data->no_placeholders) { +		r_res.unref(); +	} else { +		String id = token.value; -	ERR_FAIL_COND_V(!p_data->rev_external_resources.has(id), ERR_PARSE_ERROR); +		ERR_FAIL_COND_V(!p_data->rev_external_resources.has(id), ERR_PARSE_ERROR); -	r_res = p_data->rev_external_resources[id]; +		r_res = p_data->rev_external_resources[id]; +	}  	VariantParser::get_token(p_stream, token, line, r_err_str);  	if (token.type != VariantParser::TK_PARENTHESIS_CLOSE) { @@ -108,29 +117,14 @@ Error ResourceLoaderText::_parse_ext_resource_dummy(DummyReadData *p_data, Varia  Error ResourceLoaderText::_parse_sub_resource(VariantParser::Stream *p_stream, Ref<Resource> &r_res, int &line, String &r_err_str) {  	VariantParser::Token token;  	VariantParser::get_token(p_stream, token, line, r_err_str); -	if (token.type != VariantParser::TK_NUMBER) { -		r_err_str = "Expected number (sub-resource index)"; +	if (token.type != VariantParser::TK_NUMBER && token.type != VariantParser::TK_STRING) { +		r_err_str = "Expected number (old style sub-resource index) or string";  		return ERR_PARSE_ERROR;  	} -	int index = token.value; - -	if (use_nocache) { -		r_res = int_resources[index]; -	} else { -		String path = local_path + "::" + itos(index); - -		if (!ignore_resource_parsing) { -			if (!ResourceCache::has(path)) { -				r_err_str = "Can't load cached sub-resource: " + path; -				return ERR_PARSE_ERROR; -			} - -			r_res = RES(ResourceCache::get(path)); -		} else { -			r_res = RES(); -		} -	} +	String id = token.value; +	ERR_FAIL_COND_V(!int_resources.has(id), ERR_INVALID_PARAMETER); +	r_res = int_resources[id];  	VariantParser::get_token(p_stream, token, line, r_err_str);  	if (token.type != VariantParser::TK_PARENTHESIS_CLOSE) { @@ -144,16 +138,16 @@ Error ResourceLoaderText::_parse_sub_resource(VariantParser::Stream *p_stream, R  Error ResourceLoaderText::_parse_ext_resource(VariantParser::Stream *p_stream, Ref<Resource> &r_res, int &line, String &r_err_str) {  	VariantParser::Token token;  	VariantParser::get_token(p_stream, token, line, r_err_str); -	if (token.type != VariantParser::TK_NUMBER) { -		r_err_str = "Expected number (sub-resource index)"; +	if (token.type != VariantParser::TK_NUMBER && token.type != VariantParser::TK_STRING) { +		r_err_str = "Expected number (old style sub-resource index) or String (ext-resource ID)";  		return ERR_PARSE_ERROR;  	} -	int id = token.value; +	String id = token.value;  	if (!ignore_resource_parsing) {  		if (!ext_resources.has(id)) { -			r_err_str = "Can't load cached ext-resource #" + itos(id); +			r_err_str = "Can't load cached ext-resource id: " + id;  			return ERR_PARSE_ERROR;  		} @@ -163,10 +157,10 @@ Error ResourceLoaderText::_parse_ext_resource(VariantParser::Stream *p_stream, R  		if (ext_resources[id].cache.is_valid()) {  			r_res = ext_resources[id].cache;  		} else if (use_sub_threads) { -			RES res = ResourceLoader::load_threaded_get(path); +			Ref<Resource> res = ResourceLoader::load_threaded_get(path);  			if (res.is_null()) {  				if (ResourceLoader::get_abort_on_missing_resources()) { -					error = ERR_FILE_CORRUPT; +					error = ERR_FILE_MISSING_DEPENDENCIES;  					error_text = "[ext_resource] referenced nonexistent resource at: " + path;  					_printerr();  					return error; @@ -178,13 +172,13 @@ Error ResourceLoaderText::_parse_ext_resource(VariantParser::Stream *p_stream, R  				r_res = res;  			}  		} else { -			error = ERR_FILE_CORRUPT; +			error = ERR_FILE_MISSING_DEPENDENCIES;  			error_text = "[ext_resource] referenced non-loaded resource at: " + path;  			_printerr();  			return error;  		}  	} else { -		r_res = RES(); +		r_res = Ref<Resource>();  	}  	VariantParser::get_token(p_stream, token, line, r_err_str); @@ -198,7 +192,7 @@ Error ResourceLoaderText::_parse_ext_resource(VariantParser::Stream *p_stream, R  Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourceParser &parser) {  	Ref<PackedScene> packed_scene; -	packed_scene.instance(); +	packed_scene.instantiate();  	while (true) {  		if (next_tag.name == "node") { @@ -223,7 +217,16 @@ Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourcePars  			if (next_tag.fields.has("type")) {  				type = packed_scene->get_state()->add_name(next_tag.fields["type"]);  			} else { -				type = SceneState::TYPE_INSTANCED; //no type? assume this was instanced +				type = SceneState::TYPE_INSTANCED; //no type? assume this was instantiated +			} + +			HashSet<StringName> path_properties; + +			if (next_tag.fields.has("node_paths")) { +				Vector<String> paths = next_tag.fields["node_paths"]; +				for (int i = 0; i < paths.size(); i++) { +					path_properties.insert(paths[i]); +				}  			}  			if (next_tag.fields.has("instance")) { @@ -278,7 +281,9 @@ Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourcePars  				error = VariantParser::parse_tag_assign_eof(&stream, lines, error_text, next_tag, assign, value, &parser);  				if (error) { -					if (error != ERR_FILE_EOF) { +					if (error == ERR_FILE_MISSING_DEPENDENCIES) { +						// Resource loading error, just skip it. +					} else if (error != ERR_FILE_EOF) {  						_printerr();  						return Ref<PackedScene>();  					} else { @@ -287,12 +292,13 @@ Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourcePars  					}  				} -				if (assign != String()) { -					int nameidx = packed_scene->get_state()->add_name(assign); +				if (!assign.is_empty()) { +					StringName assign_name = assign; +					int nameidx = packed_scene->get_state()->add_name(assign_name);  					int valueidx = packed_scene->get_state()->add_value(value); -					packed_scene->get_state()->add_node_property(node_id, nameidx, valueidx); +					packed_scene->get_state()->add_node_property(node_id, nameidx, valueidx, path_properties.has(assign_name));  					//it's assignment -				} else if (next_tag.name != String()) { +				} else if (!next_tag.name.is_empty()) {  					break;  				}  			} @@ -326,6 +332,7 @@ Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourcePars  			StringName method = next_tag.fields["method"];  			StringName signal = next_tag.fields["signal"];  			int flags = Object::CONNECT_PERSIST; +			int unbinds = 0;  			Array binds;  			if (next_tag.fields.has("flags")) { @@ -336,6 +343,10 @@ Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourcePars  				binds = next_tag.fields["binds"];  			} +			if (next_tag.fields.has("unbinds")) { +				unbinds = next_tag.fields["unbinds"]; +			} +  			Vector<int> bind_ints;  			for (int i = 0; i < binds.size(); i++) {  				bind_ints.push_back(packed_scene->get_state()->add_value(binds[i])); @@ -347,6 +358,7 @@ Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourcePars  					packed_scene->get_state()->add_name(signal),  					packed_scene->get_state()->add_name(method),  					flags, +					unbinds,  					bind_ints);  			error = VariantParser::parse_tag(&stream, lines, error_text, next_tag, &parser); @@ -363,7 +375,7 @@ Ref<PackedScene> ResourceLoaderText::_parse_node_tag(VariantParser::ResourcePars  		} else if (next_tag.name == "editable") {  			if (!next_tag.fields.has("path")) {  				error = ERR_FILE_CORRUPT; -				error_text = "missing 'path' field from connection tag"; +				error_text = "missing 'path' field from editable tag";  				_printerr();  				return Ref<PackedScene>();  			} @@ -424,11 +436,29 @@ Error ResourceLoaderText::load() {  		String path = next_tag.fields["path"];  		String type = next_tag.fields["type"]; -		int index = next_tag.fields["id"]; +		String id = next_tag.fields["id"]; + +		if (next_tag.fields.has("uid")) { +			String uidt = next_tag.fields["uid"]; +			ResourceUID::ID uid = ResourceUID::get_singleton()->text_to_id(uidt); +			if (uid != ResourceUID::INVALID_ID && ResourceUID::get_singleton()->has_id(uid)) { +				// If a UID is found and the path is valid, it will be used, otherwise, it falls back to the path. +				path = ResourceUID::get_singleton()->get_id_path(uid); +			} else { +#ifdef TOOLS_ENABLED +				// Silence a warning that can happen during the initial filesystem scan due to cache being regenerated. +				if (ResourceLoader::get_resource_uid(path) != uid) { +					WARN_PRINT(String(res_path + ":" + itos(lines) + " - ext_resource, invalid UUID: " + uidt + " - using text path instead: " + path).utf8().get_data()); +				} +#else +				WARN_PRINT(String(res_path + ":" + itos(lines) + " - ext_resource, invalid UUID: " + uidt + " - using text path instead: " + path).utf8().get_data()); +#endif +			} +		} -		if (path.find("://") == -1 && path.is_rel_path()) { +		if (!path.contains("://") && path.is_relative_path()) {  			// path is relative to file being loaded, so convert to a resource path -			path = ProjectSettings::get_singleton()->localize_path(local_path.get_base_dir().plus_file(path)); +			path = ProjectSettings::get_singleton()->localize_path(local_path.get_base_dir().path_join(path));  		}  		if (remaps.has(path)) { @@ -440,7 +470,7 @@ Error ResourceLoaderText::load() {  		er.type = type;  		if (use_sub_threads) { -			Error err = ResourceLoader::load_threaded_request(path, type, use_sub_threads, local_path); +			Error err = ResourceLoader::load_threaded_request(path, type, use_sub_threads, ResourceFormatLoader::CACHE_MODE_REUSE, local_path);  			if (err != OK) {  				if (ResourceLoader::get_abort_on_missing_resources()) { @@ -454,7 +484,7 @@ Error ResourceLoaderText::load() {  			}  		} else { -			RES res = ResourceLoader::load(path, type); +			Ref<Resource> res = ResourceLoader::load(path, type);  			if (res.is_null()) {  				if (ResourceLoader::get_abort_on_missing_resources()) { @@ -468,14 +498,14 @@ Error ResourceLoaderText::load() {  			} else {  #ifdef TOOLS_ENABLED  				//remember ID for saving -				res->set_id_for_path(local_path, index); +				res->set_id_for_path(local_path, id);  #endif  			}  			er.cache = res;  		} -		ext_resources[index] = er; +		ext_resources[id] = er;  		error = VariantParser::parse_tag(&stream, lines, error_text, next_tag, &rp); @@ -504,47 +534,79 @@ Error ResourceLoaderText::load() {  		if (!next_tag.fields.has("id")) {  			error = ERR_FILE_CORRUPT; -			error_text = "Missing 'index' in external resource tag"; +			error_text = "Missing 'id' in external resource tag";  			_printerr();  			return error;  		}  		String type = next_tag.fields["type"]; -		int id = next_tag.fields["id"]; +		String id = next_tag.fields["id"]; -		String path = local_path + "::" + itos(id); +		String path = local_path + "::" + id;  		//bool exists=ResourceCache::has(path);  		Ref<Resource> res; +		bool do_assign = false; -		if (use_nocache || !ResourceCache::has(path)) { //only if it doesn't exist - -			Object *obj = ClassDB::instance(type); -			if (!obj) { -				error_text += "Can't create sub resource of type: " + type; -				_printerr(); -				error = ERR_FILE_CORRUPT; -				return error; +		if (cache_mode == ResourceFormatLoader::CACHE_MODE_REPLACE && ResourceCache::has(path)) { +			//reuse existing +			Ref<Resource> cache = ResourceCache::get_ref(path); +			if (cache.is_valid() && cache->get_class() == type) { +				res = cache; +				res->reset_state(); +				do_assign = true;  			} +		} -			Resource *r = Object::cast_to<Resource>(obj); -			if (!r) { -				error_text += "Can't create sub resource of type, because not a resource: " + type; -				_printerr(); -				error = ERR_FILE_CORRUPT; -				return error; -			} +		MissingResource *missing_resource = nullptr; + +		if (res.is_null()) { //not reuse +			Ref<Resource> cache = ResourceCache::get_ref(path); +			if (cache_mode != ResourceFormatLoader::CACHE_MODE_IGNORE && cache.is_valid()) { //only if it doesn't exist +				//cached, do not assign +				res = cache; +			} else { +				//create + +				Object *obj = ClassDB::instantiate(type); +				if (!obj) { +					if (ResourceLoader::is_creating_missing_resources_if_class_unavailable_enabled()) { +						missing_resource = memnew(MissingResource); +						missing_resource->set_original_class(type); +						missing_resource->set_recording_properties(true); +						obj = missing_resource; +					} else { +						error_text += "Can't create sub resource of type: " + type; +						_printerr(); +						error = ERR_FILE_CORRUPT; +						return error; +					} +				} -			res = Ref<Resource>(r); -			int_resources[id] = res; -			if (!use_nocache) { -				res->set_path(path); +				Resource *r = Object::cast_to<Resource>(obj); +				if (!r) { +					error_text += "Can't create sub resource of type, because not a resource: " + type; +					_printerr(); +					error = ERR_FILE_CORRUPT; +					return error; +				} + +				res = Ref<Resource>(r); +				do_assign = true;  			}  		}  		resource_current++; +		int_resources[id] = res; //always assign int resources +		if (do_assign && cache_mode != ResourceFormatLoader::CACHE_MODE_IGNORE) { +			res->set_path(path, cache_mode == ResourceFormatLoader::CACHE_MODE_REPLACE); +			res->set_scene_unique_id(id); +		} + +		Dictionary missing_resource_properties; +  		while (true) {  			String assign;  			Variant value; @@ -556,12 +618,28 @@ Error ResourceLoaderText::load() {  				return error;  			} -			if (assign != String()) { -				if (res.is_valid()) { -					res->set(assign, value); +			if (!assign.is_empty()) { +				if (do_assign) { +					bool set_valid = true; + +					if (value.get_type() == Variant::OBJECT && missing_resource != nullptr) { +						// If the property being set is a missing resource (and the parent is not), +						// then setting it will most likely not work. +						// Instead, save it as metadata. + +						Ref<MissingResource> mr = value; +						if (mr.is_valid()) { +							missing_resource_properties[assign] = mr; +							set_valid = false; +						} +					} + +					if (set_valid) { +						res->set(assign, value); +					}  				}  				//it's assignment -			} else if (next_tag.name != String()) { +			} else if (!next_tag.name.is_empty()) {  				error = OK;  				break;  			} else { @@ -572,6 +650,14 @@ Error ResourceLoaderText::load() {  			}  		} +		if (missing_resource) { +			missing_resource->set_recording_properties(false); +		} + +		if (!missing_resource_properties.is_empty()) { +			res->set_meta(META_MISSING_RESOURCES, missing_resource_properties); +		} +  		if (progress && resources_total > 0) {  			*progress = resource_current / float(resources_total);  		} @@ -589,26 +675,45 @@ Error ResourceLoaderText::load() {  			return error;  		} -		Object *obj = ClassDB::instance(res_type); -		if (!obj) { -			error_text += "Can't create sub resource of type: " + res_type; -			_printerr(); -			error = ERR_FILE_CORRUPT; -			return error; +		Ref<Resource> cache = ResourceCache::get_ref(local_path); +		if (cache_mode == ResourceFormatLoader::CACHE_MODE_REPLACE && cache.is_valid() && cache->get_class() == res_type) { +			cache->reset_state(); +			resource = cache;  		} -		Resource *r = Object::cast_to<Resource>(obj); -		if (!r) { -			error_text += "Can't create sub resource of type, because not a resource: " + res_type; -			_printerr(); -			error = ERR_FILE_CORRUPT; -			return error; -		} +		MissingResource *missing_resource = nullptr; -		resource = Ref<Resource>(r); +		if (!resource.is_valid()) { +			Object *obj = ClassDB::instantiate(res_type); +			if (!obj) { +				if (ResourceLoader::is_creating_missing_resources_if_class_unavailable_enabled()) { +					missing_resource = memnew(MissingResource); +					missing_resource->set_original_class(res_type); +					missing_resource->set_recording_properties(true); +					obj = missing_resource; +				} else { +					error_text += "Can't create sub resource of type: " + res_type; +					_printerr(); +					error = ERR_FILE_CORRUPT; +					return error; +				} +			} + +			Resource *r = Object::cast_to<Resource>(obj); +			if (!r) { +				error_text += "Can't create sub resource of type, because not a resource: " + res_type; +				_printerr(); +				error = ERR_FILE_CORRUPT; +				return error; +			} + +			resource = Ref<Resource>(r); +		}  		resource_current++; +		Dictionary missing_resource_properties; +  		while (true) {  			String assign;  			Variant value; @@ -620,7 +725,7 @@ Error ResourceLoaderText::load() {  					_printerr();  				} else {  					error = OK; -					if (!use_nocache) { +					if (cache_mode != ResourceFormatLoader::CACHE_MODE_IGNORE) {  						if (!ResourceCache::has(res_path)) {  							resource->set_path(res_path);  						} @@ -630,23 +735,49 @@ Error ResourceLoaderText::load() {  				return error;  			} -			if (assign != String()) { -				resource->set(assign, value); +			if (!assign.is_empty()) { +				bool set_valid = true; + +				if (value.get_type() == Variant::OBJECT && missing_resource != nullptr) { +					// If the property being set is a missing resource (and the parent is not), +					// then setting it will most likely not work. +					// Instead, save it as metadata. + +					Ref<MissingResource> mr = value; +					if (mr.is_valid()) { +						missing_resource_properties[assign] = mr; +						set_valid = false; +					} +				} + +				if (set_valid) { +					resource->set(assign, value); +				}  				//it's assignment -			} else if (next_tag.name != String()) { +			} else if (!next_tag.name.is_empty()) {  				error = ERR_FILE_CORRUPT;  				error_text = "Extra tag found when parsing main resource file";  				_printerr();  				return error;  			} else { -				error = OK; -				if (progress && resources_total > 0) { -					*progress = resource_current / float(resources_total); -				} - -				return error; +				break;  			}  		} + +		if (missing_resource) { +			missing_resource->set_recording_properties(false); +		} + +		if (!missing_resource_properties.is_empty()) { +			resource->set_meta(META_MISSING_RESOURCES, missing_resource_properties); +		} + +		error = OK; +		if (progress && resources_total > 0) { +			*progress = resource_current / float(resources_total); +		} + +		return error;  	}  	//for scene files @@ -668,7 +799,7 @@ Error ResourceLoaderText::load() {  		error = OK;  		//get it here  		resource = packed_scene; -		if (!use_nocache && !ResourceCache::has(res_path)) { +		if (cache_mode != ResourceFormatLoader::CACHE_MODE_IGNORE && !ResourceCache::has(res_path)) {  			packed_scene->set_path(res_path);  		} @@ -699,25 +830,9 @@ void ResourceLoaderText::set_translation_remapped(bool p_remapped) {  	translation_remapped = p_remapped;  } -ResourceLoaderText::ResourceLoaderText() { -	use_nocache = false; - -	resources_total = 0; -	resource_current = 0; -	use_sub_threads = false; +ResourceLoaderText::ResourceLoaderText() {} -	progress = nullptr; -	lines = false; -	translation_remapped = false; -	use_sub_threads = false; -	error = OK; -} - -ResourceLoaderText::~ResourceLoaderText() { -	memdelete(f); -} - -void ResourceLoaderText::get_dependencies(FileAccess *p_f, List<String> *p_dependencies, bool p_add_types) { +void ResourceLoaderText::get_dependencies(Ref<FileAccess> p_f, List<String> *p_dependencies, bool p_add_types) {  	open(p_f);  	ignore_resource_parsing = true;  	ERR_FAIL_COND(error != OK); @@ -732,7 +847,7 @@ void ResourceLoaderText::get_dependencies(FileAccess *p_f, List<String> *p_depen  		if (!next_tag.fields.has("id")) {  			error = ERR_FILE_CORRUPT; -			error_text = "Missing 'index' in external resource tag"; +			error_text = "Missing 'id' in external resource tag";  			_printerr();  			return;  		} @@ -740,9 +855,20 @@ void ResourceLoaderText::get_dependencies(FileAccess *p_f, List<String> *p_depen  		String path = next_tag.fields["path"];  		String type = next_tag.fields["type"]; -		if (path.find("://") == -1 && path.is_rel_path()) { +		bool using_uid = false; +		if (next_tag.fields.has("uid")) { +			//if uid exists, return uid in text format, not the path +			String uidt = next_tag.fields["uid"]; +			ResourceUID::ID uid = ResourceUID::get_singleton()->text_to_id(uidt); +			if (uid != ResourceUID::INVALID_ID) { +				path = ResourceUID::get_singleton()->id_to_text(uid); +				using_uid = true; +			} +		} + +		if (!using_uid && !path.contains("://") && path.is_relative_path()) {  			// path is relative to file being loaded, so convert to a resource path -			path = ProjectSettings::get_singleton()->localize_path(local_path.get_base_dir().plus_file(path)); +			path = ProjectSettings::get_singleton()->localize_path(local_path.get_base_dir().path_join(path));  		}  		if (p_add_types) { @@ -762,13 +888,13 @@ void ResourceLoaderText::get_dependencies(FileAccess *p_f, List<String> *p_depen  	}  } -Error ResourceLoaderText::rename_dependencies(FileAccess *p_f, const String &p_path, const Map<String, String> &p_map) { +Error ResourceLoaderText::rename_dependencies(Ref<FileAccess> p_f, const String &p_path, const HashMap<String, String> &p_map) {  	open(p_f, true);  	ERR_FAIL_COND_V(error != OK, error);  	ignore_resource_parsing = true;  	//FileAccess -	FileAccess *fw = nullptr; +	Ref<FileAccess> fw;  	String base_path = local_path.get_base_dir(); @@ -778,23 +904,20 @@ Error ResourceLoaderText::rename_dependencies(FileAccess *p_f, const String &p_p  		Error err = VariantParser::parse_tag(&stream, lines, error_text, next_tag, &rp);  		if (err != OK) { -			if (fw) { -				memdelete(fw); -			}  			error = ERR_FILE_CORRUPT;  			ERR_FAIL_V(error);  		}  		if (next_tag.name != "ext_resource") {  			//nothing was done -			if (!fw) { +			if (fw.is_null()) {  				return OK;  			}  			break;  		} else { -			if (!fw) { +			if (fw.is_null()) {  				fw = FileAccess::open(p_path + ".depren", FileAccess::WRITE);  				if (is_scene) {  					fw->store_line("[gd_scene load_steps=" + itos(resources_total) + " format=" + itos(FORMAT_VERSION) + "]\n"); @@ -804,18 +927,25 @@ Error ResourceLoaderText::rename_dependencies(FileAccess *p_f, const String &p_p  			}  			if (!next_tag.fields.has("path") || !next_tag.fields.has("id") || !next_tag.fields.has("type")) { -				memdelete(fw);  				error = ERR_FILE_CORRUPT;  				ERR_FAIL_V(error);  			}  			String path = next_tag.fields["path"]; -			int index = next_tag.fields["id"]; +			String id = next_tag.fields["id"];  			String type = next_tag.fields["type"]; +			if (next_tag.fields.has("uid")) { +				String uidt = next_tag.fields["uid"]; +				ResourceUID::ID uid = ResourceUID::get_singleton()->text_to_id(uidt); +				if (uid != ResourceUID::INVALID_ID && ResourceUID::get_singleton()->has_id(uid)) { +					// If a UID is found and the path is valid, it will be used, otherwise, it falls back to the path. +					path = ResourceUID::get_singleton()->get_id_path(uid); +				} +			}  			bool relative = false;  			if (!path.begins_with("res://")) { -				path = base_path.plus_file(path).simplify_path(); +				path = base_path.path_join(path).simplify_path();  				relative = true;  			} @@ -829,7 +959,14 @@ Error ResourceLoaderText::rename_dependencies(FileAccess *p_f, const String &p_p  				path = base_path.path_to_file(path);  			} -			fw->store_line("[ext_resource path=\"" + path + "\" type=\"" + type + "\" id=" + itos(index) + "]"); +			String s = "[ext_resource type=\"" + type + "\""; + +			ResourceUID::ID uid = ResourceSaver::get_resource_id_for_path(path); +			if (uid != ResourceUID::INVALID_ID) { +				s += " uid=\"" + ResourceUID::get_singleton()->id_to_text(uid) + "\""; +			} +			s += " path=\"" + path + "\" id=\"" + id + "\"]"; +			fw->store_line(s); // Bundled.  			tag_end = f->get_position();  		} @@ -838,29 +975,26 @@ Error ResourceLoaderText::rename_dependencies(FileAccess *p_f, const String &p_p  	f->seek(tag_end);  	uint8_t c = f->get_8(); +	if (c == '\n' && !f->eof_reached()) { +		// Skip first newline character since we added one +		c = f->get_8(); +	} +  	while (!f->eof_reached()) {  		fw->store_8(c);  		c = f->get_8();  	} -	f->close();  	bool all_ok = fw->get_error() == OK; -	memdelete(fw); -  	if (!all_ok) {  		return ERR_CANT_CREATE;  	} -	DirAccess *da = DirAccess::create(DirAccess::ACCESS_RESOURCES); -	da->remove(p_path); -	da->rename(p_path + ".depren", p_path); -	memdelete(da); -  	return OK;  } -void ResourceLoaderText::open(FileAccess *p_f, bool p_skip_first_tag) { +void ResourceLoaderText::open(Ref<FileAccess> p_f, bool p_skip_first_tag) {  	error = OK;  	lines = 1; @@ -910,6 +1044,12 @@ void ResourceLoaderText::open(FileAccess *p_f, bool p_skip_first_tag) {  		return;  	} +	if (tag.fields.has("uid")) { +		res_uid = ResourceUID::get_singleton()->text_to_id(tag.fields["uid"]); +	} else { +		res_uid = ResourceUID::INVALID_ID; +	} +  	if (tag.fields.has("load_steps")) {  		resources_total = tag.fields["load_steps"];  	} else { @@ -928,27 +1068,26 @@ void ResourceLoaderText::open(FileAccess *p_f, bool p_skip_first_tag) {  	rp.ext_func = _parse_ext_resources;  	rp.sub_func = _parse_sub_resources; -	rp.func = nullptr;  	rp.userdata = this;  } -static void bs_save_unicode_string(FileAccess *f, const String &p_string, bool p_bit_on_len = false) { +static void bs_save_unicode_string(Ref<FileAccess> p_f, const String &p_string, bool p_bit_on_len = false) {  	CharString utf8 = p_string.utf8();  	if (p_bit_on_len) { -		f->store_32((utf8.length() + 1) | 0x80000000); +		p_f->store_32((utf8.length() + 1) | 0x80000000);  	} else { -		f->store_32(utf8.length() + 1); +		p_f->store_32(utf8.length() + 1);  	} -	f->store_buffer((const uint8_t *)utf8.get_data(), utf8.length() + 1); +	p_f->store_buffer((const uint8_t *)utf8.get_data(), utf8.length() + 1);  } -Error ResourceLoaderText::save_as_binary(FileAccess *p_f, const String &p_path) { +Error ResourceLoaderText::save_as_binary(const String &p_path) {  	if (error) {  		return error;  	} -	FileAccessRef wf = FileAccess::open(p_path, FileAccess::WRITE); -	if (!wf) { +	Ref<FileAccess> wf = FileAccess::open(p_path, FileAccess::WRITE); +	if (wf.is_null()) {  		return ERR_CANT_OPEN;  	} @@ -960,27 +1099,32 @@ Error ResourceLoaderText::save_as_binary(FileAccess *p_f, const String &p_path)  	wf->store_32(0); //64 bits file, false for now  	wf->store_32(VERSION_MAJOR);  	wf->store_32(VERSION_MINOR); -	static const int save_format_version = 3; //use format version 3 for saving +	static const int save_format_version = BINARY_FORMAT_VERSION;  	wf->store_32(save_format_version); -	bs_save_unicode_string(wf.f, is_scene ? "PackedScene" : resource_type); +	bs_save_unicode_string(wf, is_scene ? "PackedScene" : resource_type);  	wf->store_64(0); //offset to import metadata, this is no longer used -	for (int i = 0; i < 14; i++) { + +	wf->store_32(ResourceFormatSaverBinaryInstance::FORMAT_FLAG_NAMED_SCENE_IDS | ResourceFormatSaverBinaryInstance::FORMAT_FLAG_UIDS); + +	wf->store_64(res_uid); + +	for (int i = 0; i < ResourceFormatSaverBinaryInstance::RESERVED_FIELDS; i++) {  		wf->store_32(0); // reserved  	}  	wf->store_32(0); //string table size, will not be in use -	size_t ext_res_count_pos = wf->get_position(); +	uint64_t ext_res_count_pos = wf->get_position();  	wf->store_32(0); //zero ext resources, still parsing them  	//go with external resources  	DummyReadData dummy_read; -	VariantParser::ResourceParser rp; -	rp.ext_func = _parse_ext_resource_dummys; -	rp.sub_func = _parse_sub_resource_dummys; -	rp.userdata = &dummy_read; +	VariantParser::ResourceParser rp_new; +	rp_new.ext_func = _parse_ext_resource_dummys; +	rp_new.sub_func = _parse_sub_resource_dummys; +	rp_new.userdata = &dummy_read;  	while (next_tag.name == "ext_resource") {  		if (!next_tag.fields.has("path")) { @@ -1006,19 +1150,25 @@ Error ResourceLoaderText::save_as_binary(FileAccess *p_f, const String &p_path)  		String path = next_tag.fields["path"];  		String type = next_tag.fields["type"]; -		int index = next_tag.fields["id"]; +		String id = next_tag.fields["id"]; +		ResourceUID::ID uid = ResourceUID::INVALID_ID; +		if (next_tag.fields.has("uid")) { +			String uidt = next_tag.fields["uid"]; +			uid = ResourceUID::get_singleton()->text_to_id(uidt); +		} -		bs_save_unicode_string(wf.f, type); -		bs_save_unicode_string(wf.f, path); +		bs_save_unicode_string(wf, type); +		bs_save_unicode_string(wf, path); +		wf->store_64(uid);  		int lindex = dummy_read.external_resources.size();  		Ref<DummyResource> dr; -		dr.instance(); +		dr.instantiate();  		dr->set_path("res://dummy" + itos(lindex)); //anything is good to detect it for saving as external  		dummy_read.external_resources[dr] = lindex; -		dummy_read.rev_external_resources[index] = dr; +		dummy_read.rev_external_resources[id] = dr; -		error = VariantParser::parse_tag(&stream, lines, error_text, next_tag, &rp); +		error = VariantParser::parse_tag(&stream, lines, error_text, next_tag, &rp_new);  		if (error) {  			_printerr(); @@ -1033,82 +1183,240 @@ Error ResourceLoaderText::save_as_binary(FileAccess *p_f, const String &p_path)  	//now, save resources to a separate file, for now -	size_t sub_res_count_pos = wf->get_position(); +	uint64_t sub_res_count_pos = wf->get_position();  	wf->store_32(0); //zero sub resources, still parsing them  	String temp_file = p_path + ".temp"; -	FileAccessRef wf2 = FileAccess::open(temp_file, FileAccess::WRITE); -	if (!wf2) { -		return ERR_CANT_OPEN; -	} +	Vector<uint64_t> local_offsets; +	Vector<uint64_t> local_pointers_pos; +	{ +		Ref<FileAccess> wf2 = FileAccess::open(temp_file, FileAccess::WRITE); +		if (wf2.is_null()) { +			return ERR_CANT_OPEN; +		} -	Vector<size_t> local_offsets; -	Vector<size_t> local_pointers_pos; +		while (next_tag.name == "sub_resource" || next_tag.name == "resource") { +			String type; +			String id; +			bool main_res; -	while (next_tag.name == "sub_resource" || next_tag.name == "resource") { -		String type; -		int id = -1; -		bool main_res; +			if (next_tag.name == "sub_resource") { +				if (!next_tag.fields.has("type")) { +					error = ERR_FILE_CORRUPT; +					error_text = "Missing 'type' in external resource tag"; +					_printerr(); +					return error; +				} -		if (next_tag.name == "sub_resource") { -			if (!next_tag.fields.has("type")) { -				error = ERR_FILE_CORRUPT; -				error_text = "Missing 'type' in external resource tag"; +				if (!next_tag.fields.has("id")) { +					error = ERR_FILE_CORRUPT; +					error_text = "Missing 'id' in external resource tag"; +					_printerr(); +					return error; +				} + +				type = next_tag.fields["type"]; +				id = next_tag.fields["id"]; +				main_res = false; + +				if (!dummy_read.resource_map.has(id)) { +					Ref<DummyResource> dr; +					dr.instantiate(); +					dr->set_scene_unique_id(id); +					dummy_read.resource_map[id] = dr; +					uint32_t im_size = dummy_read.resource_index_map.size(); +					dummy_read.resource_index_map.insert(dr, im_size); +				} + +			} else { +				type = res_type; +				String uid_text = ResourceUID::get_singleton()->id_to_text(res_uid); +				id = type + "_" + uid_text.replace("uid://", "").replace("<invalid>", "0"); +				main_res = true; +			} + +			local_offsets.push_back(wf2->get_position()); + +			bs_save_unicode_string(wf, "local://" + id); +			local_pointers_pos.push_back(wf->get_position()); +			wf->store_64(0); //temp local offset + +			bs_save_unicode_string(wf2, type); +			uint64_t propcount_ofs = wf2->get_position(); +			wf2->store_32(0); + +			int prop_count = 0; + +			while (true) { +				String assign; +				Variant value; + +				error = VariantParser::parse_tag_assign_eof(&stream, lines, error_text, next_tag, assign, value, &rp_new); + +				if (error) { +					if (main_res && error == ERR_FILE_EOF) { +						next_tag.name = ""; //exit +						break; +					} + +					_printerr(); +					return error; +				} + +				if (!assign.is_empty()) { +					HashMap<StringName, int> empty_string_map; //unused +					bs_save_unicode_string(wf2, assign, true); +					ResourceFormatSaverBinaryInstance::write_variant(wf2, value, dummy_read.resource_index_map, dummy_read.external_resources, empty_string_map); +					prop_count++; + +				} else if (!next_tag.name.is_empty()) { +					error = OK; +					break; +				} else { +					error = ERR_FILE_CORRUPT; +					error_text = "Premature end of file while parsing [sub_resource]"; +					_printerr(); +					return error; +				} +			} + +			wf2->seek(propcount_ofs); +			wf2->store_32(prop_count); +			wf2->seek_end(); +		} + +		if (next_tag.name == "node") { +			// This is a node, must save one more! + +			if (!is_scene) { +				error_text += "found the 'node' tag on a resource file!";  				_printerr(); +				error = ERR_FILE_CORRUPT;  				return error;  			} -			if (!next_tag.fields.has("id")) { -				error = ERR_FILE_CORRUPT; -				error_text = "Missing 'index' in external resource tag"; -				_printerr(); +			Ref<PackedScene> packed_scene = _parse_node_tag(rp_new); + +			if (!packed_scene.is_valid()) {  				return error;  			} -			type = next_tag.fields["type"]; -			id = next_tag.fields["id"]; -			main_res = false; -		} else { -			type = res_type; -			id = 0; //used for last anyway -			main_res = true; +			error = OK; +			//get it here +			List<PropertyInfo> props; +			packed_scene->get_property_list(&props); + +			String id = "PackedScene_" + ResourceUID::get_singleton()->id_to_text(res_uid).replace("uid://", "").replace("<invalid>", "0"); +			bs_save_unicode_string(wf, "local://" + id); +			local_pointers_pos.push_back(wf->get_position()); +			wf->store_64(0); //temp local offset + +			local_offsets.push_back(wf2->get_position()); +			bs_save_unicode_string(wf2, "PackedScene"); +			uint64_t propcount_ofs = wf2->get_position(); +			wf2->store_32(0); + +			int prop_count = 0; + +			for (const PropertyInfo &E : props) { +				if (!(E.usage & PROPERTY_USAGE_STORAGE)) { +					continue; +				} + +				String name = E.name; +				Variant value = packed_scene->get(name); + +				HashMap<StringName, int> empty_string_map; //unused +				bs_save_unicode_string(wf2, name, true); +				ResourceFormatSaverBinaryInstance::write_variant(wf2, value, dummy_read.resource_index_map, dummy_read.external_resources, empty_string_map); +				prop_count++; +			} + +			wf2->seek(propcount_ofs); +			wf2->store_32(prop_count); +			wf2->seek_end();  		} +	} + +	uint64_t offset_from = wf->get_position(); +	wf->seek(sub_res_count_pos); //plus one because the saved one +	wf->store_32(local_offsets.size()); -		local_offsets.push_back(wf2->get_position()); +	for (int i = 0; i < local_offsets.size(); i++) { +		wf->seek(local_pointers_pos[i]); +		wf->store_64(local_offsets[i] + offset_from); +	} -		bs_save_unicode_string(wf, "local://" + itos(id)); -		local_pointers_pos.push_back(wf->get_position()); -		wf->store_64(0); //temp local offset +	wf->seek_end(); -		bs_save_unicode_string(wf2, type); -		size_t propcount_ofs = wf2->get_position(); -		wf2->store_32(0); +	Vector<uint8_t> data = FileAccess::get_file_as_array(temp_file); +	wf->store_buffer(data.ptr(), data.size()); +	{ +		Ref<DirAccess> dar = DirAccess::open(temp_file.get_base_dir()); +		dar->remove(temp_file); +	} -		int prop_count = 0; +	wf->store_buffer((const uint8_t *)"RSRC", 4); //magic at end + +	return OK; +} + +Error ResourceLoaderText::get_classes_used(HashSet<StringName> *r_classes) { +	if (error) { +		return error; +	} + +	ignore_resource_parsing = true; + +	DummyReadData dummy_read; +	dummy_read.no_placeholders = true; +	VariantParser::ResourceParser rp_new; +	rp_new.ext_func = _parse_ext_resource_dummys; +	rp_new.sub_func = _parse_sub_resource_dummys; +	rp_new.userdata = &dummy_read; + +	while (next_tag.name == "ext_resource") { +		error = VariantParser::parse_tag(&stream, lines, error_text, next_tag, &rp_new); + +		if (error) { +			_printerr(); +			return error; +		} +	} + +	while (next_tag.name == "sub_resource" || next_tag.name == "resource") { +		if (next_tag.name == "sub_resource") { +			if (!next_tag.fields.has("type")) { +				error = ERR_FILE_CORRUPT; +				error_text = "Missing 'type' in external resource tag"; +				_printerr(); +				return error; +			} + +			r_classes->insert(next_tag.fields["type"]); + +		} else { +			r_classes->insert(next_tag.fields["res_type"]); +		}  		while (true) {  			String assign;  			Variant value; -			error = VariantParser::parse_tag_assign_eof(&stream, lines, error_text, next_tag, assign, value, &rp); +			error = VariantParser::parse_tag_assign_eof(&stream, lines, error_text, next_tag, assign, value, &rp_new);  			if (error) { -				if (main_res && error == ERR_FILE_EOF) { -					next_tag.name = ""; //exit -					break; +				if (error == ERR_FILE_EOF) { +					return OK;  				}  				_printerr();  				return error;  			} -			if (assign != String()) { -				Map<StringName, int> empty_string_map; //unused -				bs_save_unicode_string(wf2, assign, true); -				ResourceFormatSaverBinaryInstance::write_variant(wf2, value, dummy_read.resource_set, dummy_read.external_resources, empty_string_map); -				prop_count++; - -			} else if (next_tag.name != String()) { +			if (!assign.is_empty()) { +				continue; +			} else if (!next_tag.name.is_empty()) {  				error = OK;  				break;  			} else { @@ -1118,14 +1426,10 @@ Error ResourceLoaderText::save_as_binary(FileAccess *p_f, const String &p_path)  				return error;  			}  		} - -		wf2->seek(propcount_ofs); -		wf2->store_32(prop_count); -		wf2->seek_end();  	} -	if (next_tag.name == "node") { -		//this is a node, must save one more! +	while (next_tag.name == "node") { +		// This is a node, must save one more!  		if (!is_scene) {  			error_text += "found the 'node' tag on a resource file!"; @@ -1134,75 +1438,50 @@ Error ResourceLoaderText::save_as_binary(FileAccess *p_f, const String &p_path)  			return error;  		} -		Ref<PackedScene> packed_scene = _parse_node_tag(rp); - -		if (!packed_scene.is_valid()) { +		if (!next_tag.fields.has("type")) { +			error = ERR_FILE_CORRUPT; +			error_text = "Missing 'type' in external resource tag"; +			_printerr();  			return error;  		} -		error = OK; -		//get it here -		List<PropertyInfo> props; -		packed_scene->get_property_list(&props); +		r_classes->insert(next_tag.fields["type"]); -		bs_save_unicode_string(wf, "local://0"); -		local_pointers_pos.push_back(wf->get_position()); -		wf->store_64(0); //temp local offset +		while (true) { +			String assign; +			Variant value; -		local_offsets.push_back(wf2->get_position()); -		bs_save_unicode_string(wf2, "PackedScene"); -		size_t propcount_ofs = wf2->get_position(); -		wf2->store_32(0); +			error = VariantParser::parse_tag_assign_eof(&stream, lines, error_text, next_tag, assign, value, &rp_new); -		int prop_count = 0; +			if (error) { +				if (error == ERR_FILE_MISSING_DEPENDENCIES) { +					// Resource loading error, just skip it. +				} else if (error != ERR_FILE_EOF) { +					_printerr(); +					return error; +				} else { +					return OK; +				} +			} -		for (List<PropertyInfo>::Element *E = props.front(); E; E = E->next()) { -			if (!(E->get().usage & PROPERTY_USAGE_STORAGE)) { +			if (!assign.is_empty()) {  				continue; +			} else if (!next_tag.name.is_empty()) { +				error = OK; +				break; +			} else { +				error = ERR_FILE_CORRUPT; +				error_text = "Premature end of file while parsing [sub_resource]"; +				_printerr(); +				return error;  			} - -			String name = E->get().name; -			Variant value = packed_scene->get(name); - -			Map<StringName, int> empty_string_map; //unused -			bs_save_unicode_string(wf2, name, true); -			ResourceFormatSaverBinaryInstance::write_variant(wf2, value, dummy_read.resource_set, dummy_read.external_resources, empty_string_map); -			prop_count++;  		} - -		wf2->seek(propcount_ofs); -		wf2->store_32(prop_count); -		wf2->seek_end();  	} -	wf2->close(); - -	size_t offset_from = wf->get_position(); -	wf->seek(sub_res_count_pos); //plus one because the saved one -	wf->store_32(local_offsets.size()); - -	for (int i = 0; i < local_offsets.size(); i++) { -		wf->seek(local_pointers_pos[i]); -		wf->store_64(local_offsets[i] + offset_from); -	} - -	wf->seek_end(); - -	Vector<uint8_t> data = FileAccess::get_file_as_array(temp_file); -	wf->store_buffer(data.ptr(), data.size()); -	{ -		DirAccessRef dar = DirAccess::open(temp_file.get_base_dir()); -		dar->remove(temp_file); -	} - -	wf->store_buffer((const uint8_t *)"RSRC", 4); //magic at end - -	wf->close(); -  	return OK;  } -String ResourceLoaderText::recognize(FileAccess *p_f) { +String ResourceLoaderText::recognize(Ref<FileAccess> p_f) {  	error = OK;  	lines = 1; @@ -1246,27 +1525,52 @@ String ResourceLoaderText::recognize(FileAccess *p_f) {  	return tag.fields["type"];  } +ResourceUID::ID ResourceLoaderText::get_uid(Ref<FileAccess> p_f) { +	error = OK; + +	lines = 1; +	f = p_f; + +	stream.f = f; + +	ignore_resource_parsing = true; + +	VariantParser::Tag tag; +	Error err = VariantParser::parse_tag(&stream, lines, error_text, tag); + +	if (err) { +		_printerr(); +		return ResourceUID::INVALID_ID; +	} + +	if (tag.fields.has("uid")) { //field is optional +		String uidt = tag.fields["uid"]; +		return ResourceUID::get_singleton()->text_to_id(uidt); +	} + +	return ResourceUID::INVALID_ID; +} +  ///////////////////// -RES ResourceFormatLoaderText::load(const String &p_path, const String &p_original_path, Error *r_error, bool p_use_sub_threads, float *r_progress, bool p_no_cache) { +Ref<Resource> ResourceFormatLoaderText::load(const String &p_path, const String &p_original_path, Error *r_error, bool p_use_sub_threads, float *r_progress, CacheMode p_cache_mode) {  	if (r_error) {  		*r_error = ERR_CANT_OPEN;  	}  	Error err; -	FileAccess *f = FileAccess::open(p_path, FileAccess::READ, &err); +	Ref<FileAccess> f = FileAccess::open(p_path, FileAccess::READ, &err); -	ERR_FAIL_COND_V_MSG(err != OK, RES(), "Cannot open file '" + p_path + "'."); +	ERR_FAIL_COND_V_MSG(err != OK, Ref<Resource>(), "Cannot open file '" + p_path + "'.");  	ResourceLoaderText loader; -	String path = p_original_path != "" ? p_original_path : p_path; -	loader.use_nocache = p_no_cache; +	String path = !p_original_path.is_empty() ? p_original_path : p_path; +	loader.cache_mode = p_cache_mode;  	loader.use_sub_threads = p_use_sub_threads;  	loader.local_path = ProjectSettings::get_singleton()->localize_path(path);  	loader.progress = r_progress;  	loader.res_path = loader.local_path; -	//loader.set_local_path( ProjectSettings::get_singleton()->localize_path(p_path) );  	loader.open(f);  	err = loader.load();  	if (r_error) { @@ -1275,19 +1579,22 @@ RES ResourceFormatLoaderText::load(const String &p_path, const String &p_origina  	if (err == OK) {  		return loader.get_resource();  	} else { -		return RES(); +		return Ref<Resource>();  	}  }  void ResourceFormatLoaderText::get_recognized_extensions_for_type(const String &p_type, List<String> *p_extensions) const { -	if (p_type == "") { +	if (p_type.is_empty()) {  		get_recognized_extensions(p_extensions);  		return;  	} -	if (p_type == "PackedScene") { +	if (ClassDB::is_parent_class("PackedScene", p_type)) {  		p_extensions->push_back("tscn"); -	} else { +	} + +	// Don't allow .tres for PackedScenes. +	if (p_type != "PackedScene") {  		p_extensions->push_back("tres");  	}  } @@ -1301,6 +1608,26 @@ bool ResourceFormatLoaderText::handles_type(const String &p_type) const {  	return true;  } +void ResourceFormatLoaderText::get_classes_used(const String &p_path, HashSet<StringName> *r_classes) { +	String ext = p_path.get_extension().to_lower(); +	if (ext == "tscn") { +		r_classes->insert("PackedScene"); +	} + +	// ...for anything else must test... + +	Ref<FileAccess> f = FileAccess::open(p_path, FileAccess::READ); +	if (f.is_null()) { +		return; // Could not read. +	} + +	ResourceLoaderText loader; +	loader.local_path = ProjectSettings::get_singleton()->localize_path(p_path); +	loader.res_path = loader.local_path; +	loader.open(f); +	loader.get_classes_used(r_classes); +} +  String ResourceFormatLoaderText::get_resource_type(const String &p_path) const {  	String ext = p_path.get_extension().to_lower();  	if (ext == "tscn") { @@ -1309,52 +1636,78 @@ String ResourceFormatLoaderText::get_resource_type(const String &p_path) const {  		return String();  	} -	//for anyhting else must test.. +	// ...for anything else must test... -	FileAccess *f = FileAccess::open(p_path, FileAccess::READ); -	if (!f) { -		return ""; //could not rwead +	Ref<FileAccess> f = FileAccess::open(p_path, FileAccess::READ); +	if (f.is_null()) { +		return ""; //could not read  	}  	ResourceLoaderText loader;  	loader.local_path = ProjectSettings::get_singleton()->localize_path(p_path);  	loader.res_path = loader.local_path; -	//loader.set_local_path( ProjectSettings::get_singleton()->localize_path(p_path) );  	String r = loader.recognize(f);  	return ClassDB::get_compatibility_remapped_class(r);  } -void ResourceFormatLoaderText::get_dependencies(const String &p_path, List<String> *p_dependencies, bool p_add_types) { -	FileAccess *f = FileAccess::open(p_path, FileAccess::READ); -	if (!f) { -		ERR_FAIL(); +ResourceUID::ID ResourceFormatLoaderText::get_resource_uid(const String &p_path) const { +	String ext = p_path.get_extension().to_lower(); + +	if (ext != "tscn" && ext != "tres") { +		return ResourceUID::INVALID_ID; +	} + +	Ref<FileAccess> f = FileAccess::open(p_path, FileAccess::READ); +	if (f.is_null()) { +		return ResourceUID::INVALID_ID; //could not read  	}  	ResourceLoaderText loader;  	loader.local_path = ProjectSettings::get_singleton()->localize_path(p_path);  	loader.res_path = loader.local_path; -	//loader.set_local_path( ProjectSettings::get_singleton()->localize_path(p_path) ); -	loader.get_dependencies(f, p_dependencies, p_add_types); +	return loader.get_uid(f);  } -Error ResourceFormatLoaderText::rename_dependencies(const String &p_path, const Map<String, String> &p_map) { -	FileAccess *f = FileAccess::open(p_path, FileAccess::READ); -	if (!f) { -		ERR_FAIL_V(ERR_CANT_OPEN); +void ResourceFormatLoaderText::get_dependencies(const String &p_path, List<String> *p_dependencies, bool p_add_types) { +	Ref<FileAccess> f = FileAccess::open(p_path, FileAccess::READ); +	if (f.is_null()) { +		ERR_FAIL();  	}  	ResourceLoaderText loader;  	loader.local_path = ProjectSettings::get_singleton()->localize_path(p_path);  	loader.res_path = loader.local_path; -	//loader.set_local_path( ProjectSettings::get_singleton()->localize_path(p_path) ); -	return loader.rename_dependencies(f, p_path, p_map); +	loader.get_dependencies(f, p_dependencies, p_add_types); +} + +Error ResourceFormatLoaderText::rename_dependencies(const String &p_path, const HashMap<String, String> &p_map) { +	Error err = OK; +	{ +		Ref<FileAccess> f = FileAccess::open(p_path, FileAccess::READ); +		if (f.is_null()) { +			ERR_FAIL_V(ERR_CANT_OPEN); +		} + +		ResourceLoaderText loader; +		loader.local_path = ProjectSettings::get_singleton()->localize_path(p_path); +		loader.res_path = loader.local_path; +		err = loader.rename_dependencies(f, p_path, p_map); +	} + +	if (err == OK) { +		Ref<DirAccess> da = DirAccess::create(DirAccess::ACCESS_RESOURCES); +		da->remove(p_path); +		da->rename(p_path + ".depren", p_path); +	} + +	return err;  }  ResourceFormatLoaderText *ResourceFormatLoaderText::singleton = nullptr;  Error ResourceFormatLoaderText::convert_file_to_binary(const String &p_src_path, const String &p_dst_path) {  	Error err; -	FileAccess *f = FileAccess::open(p_src_path, FileAccess::READ, &err); +	Ref<FileAccess> f = FileAccess::open(p_src_path, FileAccess::READ, &err);  	ERR_FAIL_COND_V_MSG(err != OK, ERR_CANT_OPEN, "Cannot open file '" + p_src_path + "'."); @@ -1362,9 +1715,8 @@ Error ResourceFormatLoaderText::convert_file_to_binary(const String &p_src_path,  	const String &path = p_src_path;  	loader.local_path = ProjectSettings::get_singleton()->localize_path(path);  	loader.res_path = loader.local_path; -	//loader.set_local_path( ProjectSettings::get_singleton()->localize_path(p_path) );  	loader.open(f); -	return loader.save_as_binary(f, p_dst_path); +	return loader.save_as_binary(p_dst_path);  }  /*****************************************************************************************************/ @@ -1378,24 +1730,24 @@ Error ResourceFormatLoaderText::convert_file_to_binary(const String &p_src_path,  /*****************************************************************************************************/  /*****************************************************************************************************/ -String ResourceFormatSaverTextInstance::_write_resources(void *ud, const RES &p_resource) { -	ResourceFormatSaverTextInstance *rsi = (ResourceFormatSaverTextInstance *)ud; +String ResourceFormatSaverTextInstance::_write_resources(void *ud, const Ref<Resource> &p_resource) { +	ResourceFormatSaverTextInstance *rsi = static_cast<ResourceFormatSaverTextInstance *>(ud);  	return rsi->_write_resource(p_resource);  } -String ResourceFormatSaverTextInstance::_write_resource(const RES &res) { +String ResourceFormatSaverTextInstance::_write_resource(const Ref<Resource> &res) {  	if (external_resources.has(res)) { -		return "ExtResource( " + itos(external_resources[res]) + " )"; +		return "ExtResource(\"" + external_resources[res] + "\")";  	} else {  		if (internal_resources.has(res)) { -			return "SubResource( " + itos(internal_resources[res]) + " )"; -		} else if (res->get_path().length() && res->get_path().find("::") == -1) { +			return "SubResource(\"" + internal_resources[res] + "\")"; +		} else if (!res->is_built_in()) {  			if (res->get_path() == local_path) { //circular reference attempt  				return "null";  			}  			//external resource  			String path = relative_paths ? local_path.path_to_file(res->get_path()) : res->get_path(); -			return "Resource( \"" + path + "\" )"; +			return "Resource(\"" + path + "\")";  		} else {  			ERR_FAIL_V_MSG("null", "Resource was not pre cached for the resource section, bug?");  			//internal resource @@ -1406,19 +1758,22 @@ String ResourceFormatSaverTextInstance::_write_resource(const RES &res) {  void ResourceFormatSaverTextInstance::_find_resources(const Variant &p_variant, bool p_main) {  	switch (p_variant.get_type()) {  		case Variant::OBJECT: { -			RES res = p_variant; +			Ref<Resource> res = p_variant;  			if (res.is_null() || external_resources.has(res)) {  				return;  			} -			if (!p_main && (!bundle_resources) && res->get_path().length() && res->get_path().find("::") == -1) { +			if (!p_main && (!bundle_resources) && !res->is_built_in()) {  				if (res->get_path() == local_path) {  					ERR_PRINT("Circular reference to resource being saved found: '" + local_path + "' will be null next time it's loaded.");  					return;  				} -				int index = external_resources.size(); -				external_resources[res] = index; + +				// Use a numeric ID as a base, because they are sorted in natural order before saving. +				// This increases the chances of thread loading to fetch them first. +				String id = itos(external_resources.size() + 1) + "_" + Resource::generate_scene_unique_id(); +				external_resources[res] = id;  				return;  			} @@ -1440,7 +1795,7 @@ void ResourceFormatSaverTextInstance::_find_resources(const Variant &p_variant,  					Variant v = res->get(I->get().name);  					if (pi.usage & PROPERTY_USAGE_RESOURCE_NOT_PERSISTENT) { -						RES sres = v; +						Ref<Resource> sres = v;  						if (sres.is_valid()) {  							NonPersistentKey npk;  							npk.base = res; @@ -1474,8 +1829,8 @@ void ResourceFormatSaverTextInstance::_find_resources(const Variant &p_variant,  			Dictionary d = p_variant;  			List<Variant> keys;  			d.get_key_list(&keys); -			for (List<Variant>::Element *E = keys.front(); E; E = E->next()) { -				Variant v = d[E->get()]; +			for (const Variant &E : keys) { +				Variant v = d[E];  				_find_resources(v);  			}  		} break; @@ -1484,15 +1839,24 @@ void ResourceFormatSaverTextInstance::_find_resources(const Variant &p_variant,  	}  } -Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_resource, uint32_t p_flags) { +static String _resource_get_class(Ref<Resource> p_resource) { +	Ref<MissingResource> missing_resource = p_resource; +	if (missing_resource.is_valid()) { +		return missing_resource->get_original_class(); +	} else { +		return p_resource->get_class(); +	} +} + +Error ResourceFormatSaverTextInstance::save(const String &p_path, const Ref<Resource> &p_resource, uint32_t p_flags) {  	if (p_path.ends_with(".tscn")) {  		packed_scene = p_resource;  	}  	Error err; -	f = FileAccess::open(p_path, FileAccess::WRITE, &err); +	Ref<FileAccess> f = FileAccess::open(p_path, FileAccess::WRITE, &err);  	ERR_FAIL_COND_V_MSG(err, ERR_CANT_OPEN, "Cannot save file '" + p_path + "'."); -	FileAccessRef _fref(f); +	Ref<FileAccess> _fref(f);  	local_path = ProjectSettings::get_singleton()->localize_path(p_path); @@ -1504,11 +1868,11 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  		takeover_paths = false;  	} -	// save resources +	// Save resources.  	_find_resources(p_resource, true);  	if (packed_scene.is_valid()) { -		//add instances to external resources if saving a packed scene +		// Add instances to external resources if saving a packed scene.  		for (int i = 0; i < packed_scene->get_state()->get_node_count(); i++) {  			if (packed_scene->get_state()->is_node_instance_placeholder(i)) {  				continue; @@ -1516,8 +1880,8 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  			Ref<PackedScene> instance = packed_scene->get_state()->get_node_instance(i);  			if (instance.is_valid() && !external_resources.has(instance)) { -				int index = external_resources.size(); -				external_resources[instance] = index; +				int index = external_resources.size() + 1; +				external_resources[instance] = itos(index) + "_" + Resource::generate_scene_unique_id(); // Keep the order for improved thread loading performance.  			}  		}  	} @@ -1525,67 +1889,77 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  	{  		String title = packed_scene.is_valid() ? "[gd_scene " : "[gd_resource ";  		if (packed_scene.is_null()) { -			title += "type=\"" + p_resource->get_class() + "\" "; +			title += "type=\"" + _resource_get_class(p_resource) + "\" ";  		}  		int load_steps = saved_resources.size() + external_resources.size(); -		/* -		if (packed_scene.is_valid()) { -			load_steps+=packed_scene->get_node_count(); -		} -		//no, better to not use load steps from nodes, no point to that -		*/  		if (load_steps > 1) {  			title += "load_steps=" + itos(load_steps) + " ";  		}  		title += "format=" + itos(FORMAT_VERSION) + ""; +		ResourceUID::ID uid = ResourceSaver::get_resource_id_for_path(local_path, true); + +		if (uid != ResourceUID::INVALID_ID) { +			title += " uid=\"" + ResourceUID::get_singleton()->id_to_text(uid) + "\""; +		} +  		f->store_string(title); -		f->store_line("]\n"); //one empty line +		f->store_line("]\n"); // One empty line.  	}  #ifdef TOOLS_ENABLED -	//keep order from cached ids -	Set<int> cached_ids_found; -	for (Map<RES, int>::Element *E = external_resources.front(); E; E = E->next()) { -		int cached_id = E->key()->get_id_for_path(local_path); -		if (cached_id < 0 || cached_ids_found.has(cached_id)) { -			E->get() = -1; //reset +	// Keep order from cached ids. +	HashSet<String> cached_ids_found; +	for (KeyValue<Ref<Resource>, String> &E : external_resources) { +		String cached_id = E.key->get_id_for_path(local_path); +		if (cached_id.is_empty() || cached_ids_found.has(cached_id)) { +			int sep_pos = E.value.find("_"); +			if (sep_pos != -1) { +				E.value = E.value.substr(0, sep_pos + 1); // Keep the order found, for improved thread loading performance. +			} else { +				E.value = ""; +			} +  		} else { -			E->get() = cached_id; +			E.value = cached_id;  			cached_ids_found.insert(cached_id);  		}  	} -	//create IDs for non cached resources -	for (Map<RES, int>::Element *E = external_resources.front(); E; E = E->next()) { -		if (cached_ids_found.has(E->get())) { //already cached, go on +	// Create IDs for non cached resources. +	for (KeyValue<Ref<Resource>, String> &E : external_resources) { +		if (cached_ids_found.has(E.value)) { // Already cached, go on.  			continue;  		} -		int attempt = 1; //start from one, more readable format -		while (cached_ids_found.has(attempt)) { -			attempt++; +		String attempt; +		while (true) { +			attempt = E.value + Resource::generate_scene_unique_id(); +			if (!cached_ids_found.has(attempt)) { +				break; +			}  		}  		cached_ids_found.insert(attempt); -		E->get() = attempt; -		//update also in resource -		Ref<Resource> res = E->key(); +		E.value = attempt; +		// Update also in resource. +		Ref<Resource> res = E.key;  		res->set_id_for_path(local_path, attempt);  	}  #else -	//make sure to start from one, as it makes format more readable -	for (Map<RES, int>::Element *E = external_resources.front(); E; E = E->next()) { -		E->get() = E->get() + 1; +	// Make sure to start from one, as it makes format more readable. +	int counter = 1; +	for (KeyValue<Ref<Resource>, String> &E : external_resources) { +		E.value = itos(counter++);  	}  #endif  	Vector<ResourceSort> sorted_er; -	for (Map<RES, int>::Element *E = external_resources.front(); E; E = E->next()) { +	for (const KeyValue<Ref<Resource>, String> &E : external_resources) {  		ResourceSort rs; -		rs.resource = E->key(); -		rs.index = E->get(); +		rs.resource = E.key; +		rs.id = E.value;  		sorted_er.push_back(rs);  	} @@ -1594,71 +1968,86 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  	for (int i = 0; i < sorted_er.size(); i++) {  		String p = sorted_er[i].resource->get_path(); -		f->store_string("[ext_resource path=\"" + p + "\" type=\"" + sorted_er[i].resource->get_save_class() + "\" id=" + itos(sorted_er[i].index) + "]\n"); //bundled +		String s = "[ext_resource type=\"" + sorted_er[i].resource->get_save_class() + "\""; + +		ResourceUID::ID uid = ResourceSaver::get_resource_id_for_path(p, false); +		if (uid != ResourceUID::INVALID_ID) { +			s += " uid=\"" + ResourceUID::get_singleton()->id_to_text(uid) + "\""; +		} +		s += " path=\"" + p + "\" id=\"" + sorted_er[i].id + "\"]\n"; +		f->store_string(s); // Bundled.  	}  	if (external_resources.size()) { -		f->store_line(String()); //separate +		f->store_line(String()); // Separate.  	} -	Set<int> used_indices; +	HashSet<String> used_unique_ids; -	for (List<RES>::Element *E = saved_resources.front(); E; E = E->next()) { -		RES res = E->get(); -		if (E->next() && (res->get_path() == "" || res->get_path().find("::") != -1)) { -			if (res->get_subindex() != 0) { -				if (used_indices.has(res->get_subindex())) { -					res->set_subindex(0); //repeated +	for (List<Ref<Resource>>::Element *E = saved_resources.front(); E; E = E->next()) { +		Ref<Resource> res = E->get(); +		if (E->next() && res->is_built_in()) { +			if (!res->get_scene_unique_id().is_empty()) { +				if (used_unique_ids.has(res->get_scene_unique_id())) { +					res->set_scene_unique_id(""); // Repeated.  				} else { -					used_indices.insert(res->get_subindex()); +					used_unique_ids.insert(res->get_scene_unique_id());  				}  			}  		}  	} -	for (List<RES>::Element *E = saved_resources.front(); E; E = E->next()) { -		RES res = E->get(); +	for (List<Ref<Resource>>::Element *E = saved_resources.front(); E; E = E->next()) { +		Ref<Resource> res = E->get();  		ERR_CONTINUE(!resource_set.has(res));  		bool main = (E->next() == nullptr);  		if (main && packed_scene.is_valid()) { -			break; //save as a scene +			break; // Save as a scene.  		}  		if (main) {  			f->store_line("[resource]");  		} else {  			String line = "[sub_resource "; -			if (res->get_subindex() == 0) { -				int new_subindex = 1; -				if (used_indices.size()) { -					new_subindex = used_indices.back()->get() + 1; +			if (res->get_scene_unique_id().is_empty()) { +				String new_id; +				while (true) { +					new_id = _resource_get_class(res) + "_" + Resource::generate_scene_unique_id(); + +					if (!used_unique_ids.has(new_id)) { +						break; +					}  				} -				res->set_subindex(new_subindex); -				used_indices.insert(new_subindex); +				res->set_scene_unique_id(new_id); +				used_unique_ids.insert(new_id);  			} -			int idx = res->get_subindex(); -			line += "type=\"" + res->get_class() + "\" id=" + itos(idx); -			f->store_line(line + "]"); +			String id = res->get_scene_unique_id(); +			line += "type=\"" + _resource_get_class(res) + "\" id=\"" + id; +			f->store_line(line + "\"]");  			if (takeover_paths) { -				res->set_path(p_path + "::" + itos(idx), true); +				res->set_path(p_path + "::" + id, true);  			} -			internal_resources[res] = idx; +			internal_resources[res] = id;  #ifdef TOOLS_ENABLED  			res->set_edited(false);  #endif  		} +		Dictionary missing_resource_properties = p_resource->get_meta(META_MISSING_RESOURCES, Dictionary()); +  		List<PropertyInfo> property_list;  		res->get_property_list(&property_list); -		//property_list.sort();  		for (List<PropertyInfo>::Element *PE = property_list.front(); PE; PE = PE->next()) {  			if (skip_editor && PE->get().name.begins_with("__editor")) {  				continue;  			} +			if (PE->get().name == META_PROPERTY_MISSING_RESOURCES) { +				continue; +			}  			if (PE->get().usage & PROPERTY_USAGE_STORAGE) {  				String name = PE->get().name; @@ -1673,6 +2062,15 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  				} else {  					value = res->get(name);  				} + +				if (PE->get().type == Variant::OBJECT && missing_resource_properties.has(PE->get().name)) { +					// Was this missing resource overridden? If so do not save the old value. +					Ref<Resource> ures = value; +					if (ures.is_null()) { +						value = missing_resource_properties[PE->get().name]; +					} +				} +  				Variant default_value = ClassDB::class_get_default_property_value(res->get_class(), name);  				if (default_value.get_type() != Variant::NIL && bool(Variant::evaluate(Variant::OP_EQUAL, value, default_value))) { @@ -1695,7 +2093,7 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  	}  	if (packed_scene.is_valid()) { -		//if this is a scene, save nodes and connections! +		// If this is a scene, save nodes and connections!  		Ref<SceneState> state = packed_scene->get_state();  		for (int i = 0; i < state->get_node_count(); i++) {  			StringName type = state->get_node_type(i); @@ -1706,6 +2104,7 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  			Ref<PackedScene> instance = state->get_node_instance(i);  			String instance_placeholder = state->get_node_instance_placeholder(i);  			Vector<StringName> groups = state->get_node_groups(i); +			Vector<String> deferred_node_paths = state->get_node_deferred_nodepath_properties(i);  			String header = "[node";  			header += " name=\"" + String(name).c_escape() + "\""; @@ -1722,11 +2121,21 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  				header += " index=\"" + itos(index) + "\"";  			} +			if (deferred_node_paths.size()) { +				header += " node_paths=" + Variant(deferred_node_paths).get_construct_string(); +			} +  			if (groups.size()) { +				// Write all groups on the same line as they're part of a section header. +				// This improves readability while not impacting VCS friendliness too much, +				// since it's rare to have more than 5 groups assigned to a single node.  				groups.sort_custom<StringName::AlphCompare>(); -				String sgroups = " groups=[\n"; +				String sgroups = " groups=[";  				for (int j = 0; j < groups.size(); j++) { -					sgroups += "\"" + String(groups[j]).c_escape() + "\",\n"; +					sgroups += "\"" + String(groups[j]).c_escape() + "\""; +					if (j < groups.size() - 1) { +						sgroups += ", "; +					}  				}  				sgroups += "]";  				header += sgroups; @@ -1734,7 +2143,7 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  			f->store_string(header); -			if (instance_placeholder != String()) { +			if (!instance_placeholder.is_empty()) {  				String vars;  				f->store_string(" instance_placeholder=");  				VariantWriter::write_to_string(instance_placeholder, vars, _write_resources, this); @@ -1777,6 +2186,11 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  				connstr += " flags=" + itos(flags);  			} +			int unbinds = state->get_connection_unbinds(i); +			if (unbinds > 0) { +				connstr += " unbinds=" + itos(unbinds); +			} +  			Array binds = state->get_connection_binds(i);  			f->store_string(connstr);  			if (binds.size()) { @@ -1798,18 +2212,14 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path, const RES &p_r  	}  	if (f->get_error() != OK && f->get_error() != ERR_FILE_EOF) { -		f->close();  		return ERR_CANT_CREATE;  	} -	f->close(); -	//memdelete(f); -  	return OK;  } -Error ResourceFormatSaverText::save(const String &p_path, const RES &p_resource, uint32_t p_flags) { -	if (p_path.ends_with(".sct") && p_resource->get_class() != "PackedScene") { +Error ResourceFormatSaverText::save(const Ref<Resource> &p_resource, const String &p_path, uint32_t p_flags) { +	if (p_path.ends_with(".tscn") && !Ref<PackedScene>(p_resource).is_valid()) {  		return ERR_FILE_UNRECOGNIZED;  	} @@ -1817,15 +2227,15 @@ Error ResourceFormatSaverText::save(const String &p_path, const RES &p_resource,  	return saver.save(p_path, p_resource, p_flags);  } -bool ResourceFormatSaverText::recognize(const RES &p_resource) const { -	return true; // all recognized! +bool ResourceFormatSaverText::recognize(const Ref<Resource> &p_resource) const { +	return true; // All resources recognized!  } -void ResourceFormatSaverText::get_recognized_extensions(const RES &p_resource, List<String> *p_extensions) const { -	if (p_resource->get_class() == "PackedScene") { -		p_extensions->push_back("tscn"); //text scene +void ResourceFormatSaverText::get_recognized_extensions(const Ref<Resource> &p_resource, List<String> *p_extensions) const { +	if (Ref<PackedScene>(p_resource).is_valid()) { +		p_extensions->push_back("tscn"); // Text scene.  	} else { -		p_extensions->push_back("tres"); //text resource +		p_extensions->push_back("tres"); // Text resource.  	}  } |