Far faster and more efficient method of checking if an identifer refers an autoload.

This commit is contained in:
Jason Knight 2023-10-09 14:18:19 -06:00 committed by Relintai
parent 44453b1d58
commit 929ad6b766

View File

@ -57,6 +57,21 @@ T *GDScriptParser::alloc_node() {
return t; return t;
} }
static String _lookup_autoload_path_for_identifier(const String &p_identifier) {
String autoload_path;
String autoload_setting_path = "autoload/" + p_identifier;
if (ProjectSettings::get_singleton()->has_setting(autoload_setting_path)) {
autoload_path = ProjectSettings::get_singleton()->get(autoload_setting_path);
if (autoload_path.begins_with("*")) {
autoload_path = autoload_path.right(1);
}
if (!autoload_path.begins_with("res://")) {
autoload_path = "res://" + autoload_path;
}
}
return autoload_path;
}
#ifdef DEBUG_ENABLED #ifdef DEBUG_ENABLED
static String _find_function_name(const GDScriptParser::OperatorNode *p_call); static String _find_function_name(const GDScriptParser::OperatorNode *p_call);
#endif // DEBUG_ENABLED #endif // DEBUG_ENABLED
@ -5408,29 +5423,14 @@ void GDScriptParser::_determine_inheritance(ClassNode *p_class, bool p_recursive
} }
p = nullptr; p = nullptr;
} else { } else {
List<PropertyInfo> props; String autoload_path = _lookup_autoload_path_for_identifier(base);
ProjectSettings::get_singleton()->get_property_list(&props); if (!autoload_path.empty()) {
for (List<PropertyInfo>::Element *E = props.front(); E; E = E->next()) { base_script = ResourceLoader::load(autoload_path);
String s = E->get().name; if (!base_script.is_valid()) {
if (!s.begins_with("autoload/")) { _set_error("Class '" + base + "' could not be fully loaded (script error or cyclic inheritance).", p_class->line);
continue; return;
}
String name = s.get_slice("/", 1);
if (name == base) {
String singleton_path = ProjectSettings::get_singleton()->get(s);
if (singleton_path.begins_with("*")) {
singleton_path = singleton_path.right(1);
}
if (!singleton_path.begins_with("res://")) {
singleton_path = "res://" + singleton_path;
}
base_script = ResourceLoader::load(singleton_path);
if (!base_script.is_valid()) {
_set_error("Class '" + base + "' could not be fully loaded (script error or cyclic inheritance).", p_class->line);
return;
}
p = nullptr;
} }
p = nullptr;
} }
} }
@ -5780,28 +5780,10 @@ GDScriptParser::DataType GDScriptParser::_resolve_type(const DataType &p_source,
name_part++; name_part++;
continue; continue;
} }
List<PropertyInfo> props;
ProjectSettings::get_singleton()->get_property_list(&props); String autoload_path = _lookup_autoload_path_for_identifier(id);
String singleton_path; if (!autoload_path.empty()) {
for (List<PropertyInfo>::Element *E = props.front(); E; E = E->next()) { Ref<Script> script = ResourceLoader::load(autoload_path);
String s = E->get().name;
if (!s.begins_with("autoload/")) {
continue;
}
String name = s.get_slice("/", 1);
if (name == id) {
singleton_path = ProjectSettings::get_singleton()->get(s);
if (singleton_path.begins_with("*")) {
singleton_path = singleton_path.right(1);
}
if (!singleton_path.begins_with("res://")) {
singleton_path = "res://" + singleton_path;
}
break;
}
}
if (!singleton_path.empty()) {
Ref<Script> script = ResourceLoader::load(singleton_path);
Ref<GDScript> gds = script; Ref<GDScript> gds = script;
if (gds.is_valid()) { if (gds.is_valid()) {
if (!gds->is_valid()) { if (!gds->is_valid()) {
@ -7753,40 +7735,24 @@ GDScriptParser::DataType GDScriptParser::_reduce_identifier_type(const DataType
} }
// Non-tool singletons aren't loaded, check project settings // Non-tool singletons aren't loaded, check project settings
List<PropertyInfo> props; String autoload_path = _lookup_autoload_path_for_identifier(p_identifier);
ProjectSettings::get_singleton()->get_property_list(&props); if (!autoload_path.empty()) {
Ref<Script> singleton = ResourceLoader::load(autoload_path);
if (singleton.is_valid()) {
DataType result;
result.has_type = true;
result.is_constant = true;
result.script_type = singleton;
for (List<PropertyInfo>::Element *E = props.front(); E; E = E->next()) { Ref<GDScript> gds = singleton;
String s = E->get().name; if (gds.is_valid()) {
if (!s.begins_with("autoload/")) { if (!gds->is_valid()) {
continue; _set_error("Couldn't fully load the singleton script \"" + p_identifier + "\" (possible cyclic reference or parse error).", p_line);
} return DataType();
String name = s.get_slice("/", 1);
if (name == p_identifier) {
String script = ProjectSettings::get_singleton()->get(s);
if (script.begins_with("*")) {
script = script.right(1);
}
if (!script.begins_with("res://")) {
script = "res://" + script;
}
Ref<Script> singleton = ResourceLoader::load(script);
if (singleton.is_valid()) {
DataType result;
result.has_type = true;
result.is_constant = true;
result.script_type = singleton;
Ref<GDScript> gds = singleton;
if (gds.is_valid()) {
if (!gds->is_valid()) {
_set_error("Couldn't fully load the singleton script \"" + p_identifier + "\" (possible cyclic reference or parse error).", p_line);
return DataType();
}
result.kind = DataType::GDSCRIPT;
} else {
result.kind = DataType::SCRIPT;
} }
result.kind = DataType::GDSCRIPT;
} else {
result.kind = DataType::SCRIPT;
} }
} }
} }