mirror of
				https://github.com/godotengine/godot.git
				synced 2025-10-26 03:04:31 +00:00 
			
		
		
		
	Improve sorting of Code Completion options.
Done by ordering options by their location in the code - e.g. local, parent class, global, etc.
This commit is contained in:
		
							parent
							
								
									74b07dd194
								
							
						
					
					
						commit
						4ab605d14d
					
				
					 7 changed files with 196 additions and 24 deletions
				
			
		|  | @ -490,6 +490,89 @@ struct GDScriptCompletionIdentifier { | |||
| 	const GDScriptParser::ExpressionNode *assigned_expression = nullptr; | ||||
| }; | ||||
| 
 | ||||
| // LOCATION METHODS
 | ||||
| // These methods are used to populate the `CodeCompletionOption::location` integer.
 | ||||
| // For these methods, the location is based on the depth in the inheritance chain that the property
 | ||||
| // appears. For example, if you are completing code in a class that inherits Node2D, a property found on Node2D
 | ||||
| // will have a "better" (lower) location "score" than a property that is found on CanvasItem.
 | ||||
| 
 | ||||
| static int _get_property_location(StringName p_class, StringName p_property) { | ||||
| 	if (!ClassDB::has_property(p_class, p_property)) { | ||||
| 		return ScriptLanguage::LOCATION_OTHER; | ||||
| 	} | ||||
| 
 | ||||
| 	int depth = 0; | ||||
| 	StringName class_test = p_class; | ||||
| 	while (class_test && !ClassDB::has_property(class_test, p_property, true)) { | ||||
| 		class_test = ClassDB::get_parent_class(class_test); | ||||
| 		depth++; | ||||
| 	} | ||||
| 
 | ||||
| 	return depth | ScriptLanguage::LOCATION_PARENT_MASK; | ||||
| } | ||||
| 
 | ||||
| static int _get_constant_location(StringName p_class, StringName p_constant) { | ||||
| 	if (!ClassDB::has_integer_constant(p_class, p_constant)) { | ||||
| 		return ScriptLanguage::LOCATION_OTHER; | ||||
| 	} | ||||
| 
 | ||||
| 	int depth = 0; | ||||
| 	StringName class_test = p_class; | ||||
| 	while (class_test && !ClassDB::has_integer_constant(class_test, p_constant, true)) { | ||||
| 		class_test = ClassDB::get_parent_class(class_test); | ||||
| 		depth++; | ||||
| 	} | ||||
| 
 | ||||
| 	return depth | ScriptLanguage::LOCATION_PARENT_MASK; | ||||
| } | ||||
| 
 | ||||
| static int _get_signal_location(StringName p_class, StringName p_signal) { | ||||
| 	if (!ClassDB::has_signal(p_class, p_signal)) { | ||||
| 		return ScriptLanguage::LOCATION_OTHER; | ||||
| 	} | ||||
| 
 | ||||
| 	int depth = 0; | ||||
| 	StringName class_test = p_class; | ||||
| 	while (class_test && !ClassDB::has_signal(class_test, p_signal, true)) { | ||||
| 		class_test = ClassDB::get_parent_class(class_test); | ||||
| 		depth++; | ||||
| 	} | ||||
| 
 | ||||
| 	return depth | ScriptLanguage::LOCATION_PARENT_MASK; | ||||
| } | ||||
| 
 | ||||
| static int _get_method_location(StringName p_class, StringName p_method) { | ||||
| 	if (!ClassDB::has_method(p_class, p_method)) { | ||||
| 		return ScriptLanguage::LOCATION_OTHER; | ||||
| 	} | ||||
| 
 | ||||
| 	int depth = 0; | ||||
| 	StringName class_test = p_class; | ||||
| 	while (class_test && !ClassDB::has_method(class_test, p_method, true)) { | ||||
| 		class_test = ClassDB::get_parent_class(class_test); | ||||
| 		depth++; | ||||
| 	} | ||||
| 
 | ||||
| 	return depth | ScriptLanguage::LOCATION_PARENT_MASK; | ||||
| } | ||||
| 
 | ||||
| static int _get_enum_constant_location(StringName p_class, StringName p_enum_constant) { | ||||
| 	if (!ClassDB::get_integer_constant_enum(p_class, p_enum_constant)) { | ||||
| 		return ScriptLanguage::LOCATION_OTHER; | ||||
| 	} | ||||
| 
 | ||||
| 	int depth = 0; | ||||
| 	StringName class_test = p_class; | ||||
| 	while (class_test && !ClassDB::get_integer_constant_enum(class_test, p_enum_constant, true)) { | ||||
| 		class_test = ClassDB::get_parent_class(class_test); | ||||
| 		depth++; | ||||
| 	} | ||||
| 
 | ||||
| 	return depth | ScriptLanguage::LOCATION_PARENT_MASK; | ||||
| } | ||||
| 
 | ||||
| // END LOCATION METHODS
 | ||||
| 
 | ||||
| static String _get_visual_datatype(const PropertyInfo &p_info, bool p_is_arg = true) { | ||||
| 	if (p_info.usage & PROPERTY_USAGE_CLASS_IS_ENUM) { | ||||
| 		String enum_name = p_info.class_name; | ||||
|  | @ -726,18 +809,18 @@ static void _list_available_types(bool p_inherit_only, GDScriptParser::Completio | |||
| 				const GDScriptParser::ClassNode::Member &member = current->members[i]; | ||||
| 				switch (member.type) { | ||||
| 					case GDScriptParser::ClassNode::Member::CLASS: { | ||||
| 						ScriptLanguage::CodeCompletionOption option(member.m_class->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS); | ||||
| 						ScriptLanguage::CodeCompletionOption option(member.m_class->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS, ScriptLanguage::LOCATION_LOCAL); | ||||
| 						r_result.insert(option.display, option); | ||||
| 					} break; | ||||
| 					case GDScriptParser::ClassNode::Member::ENUM: { | ||||
| 						if (!p_inherit_only) { | ||||
| 							ScriptLanguage::CodeCompletionOption option(member.m_enum->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_ENUM); | ||||
| 							ScriptLanguage::CodeCompletionOption option(member.m_enum->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_ENUM, ScriptLanguage::LOCATION_LOCAL); | ||||
| 							r_result.insert(option.display, option); | ||||
| 						} | ||||
| 					} break; | ||||
| 					case GDScriptParser::ClassNode::Member::CONSTANT: { | ||||
| 						if (member.constant->get_datatype().is_meta_type && p_context.current_class->outer != nullptr) { | ||||
| 							ScriptLanguage::CodeCompletionOption option(member.constant->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS); | ||||
| 							ScriptLanguage::CodeCompletionOption option(member.constant->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS, ScriptLanguage::LOCATION_LOCAL); | ||||
| 							r_result.insert(option.display, option); | ||||
| 						} | ||||
| 					} break; | ||||
|  | @ -753,7 +836,7 @@ static void _list_available_types(bool p_inherit_only, GDScriptParser::Completio | |||
| 	List<StringName> global_classes; | ||||
| 	ScriptServer::get_global_class_list(&global_classes); | ||||
| 	for (const StringName &E : global_classes) { | ||||
| 		ScriptLanguage::CodeCompletionOption option(E, ScriptLanguage::CODE_COMPLETION_KIND_CLASS); | ||||
| 		ScriptLanguage::CodeCompletionOption option(E, ScriptLanguage::CODE_COMPLETION_KIND_CLASS, ScriptLanguage::LOCATION_OTHER_USER_CODE); | ||||
| 		r_result.insert(option.display, option); | ||||
| 	} | ||||
| 
 | ||||
|  | @ -764,7 +847,7 @@ static void _list_available_types(bool p_inherit_only, GDScriptParser::Completio | |||
| 		if (!info.is_singleton || info.path.get_extension().to_lower() != "gd") { | ||||
| 			continue; | ||||
| 		} | ||||
| 		ScriptLanguage::CodeCompletionOption option(info.name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS); | ||||
| 		ScriptLanguage::CodeCompletionOption option(info.name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS, ScriptLanguage::LOCATION_OTHER_USER_CODE); | ||||
| 		r_result.insert(option.display, option); | ||||
| 	} | ||||
| } | ||||
|  | @ -773,10 +856,10 @@ static void _find_identifiers_in_suite(const GDScriptParser::SuiteNode *p_suite, | |||
| 	for (int i = 0; i < p_suite->locals.size(); i++) { | ||||
| 		ScriptLanguage::CodeCompletionOption option; | ||||
| 		if (p_suite->locals[i].type == GDScriptParser::SuiteNode::Local::CONSTANT) { | ||||
| 			option = ScriptLanguage::CodeCompletionOption(p_suite->locals[i].name, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT); | ||||
| 			option = ScriptLanguage::CodeCompletionOption(p_suite->locals[i].name, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT, ScriptLanguage::LOCATION_LOCAL); | ||||
| 			option.default_value = p_suite->locals[i].constant->initializer->reduced_value; | ||||
| 		} else { | ||||
| 			option = ScriptLanguage::CodeCompletionOption(p_suite->locals[i].name, ScriptLanguage::CODE_COMPLETION_KIND_VARIABLE); | ||||
| 			option = ScriptLanguage::CodeCompletionOption(p_suite->locals[i].name, ScriptLanguage::CODE_COMPLETION_KIND_VARIABLE, ScriptLanguage::LOCATION_LOCAL); | ||||
| 		} | ||||
| 		r_result.insert(option.display, option); | ||||
| 	} | ||||
|  | @ -793,8 +876,10 @@ static void _find_identifiers_in_class(const GDScriptParser::ClassNode *p_class, | |||
| 	if (!p_parent_only) { | ||||
| 		bool outer = false; | ||||
| 		const GDScriptParser::ClassNode *clss = p_class; | ||||
| 		int classes_processed = 0; | ||||
| 		while (clss) { | ||||
| 			for (int i = 0; i < clss->members.size(); i++) { | ||||
| 				const int location = (classes_processed + p_recursion_depth) | ScriptLanguage::LOCATION_PARENT_MASK; | ||||
| 				const GDScriptParser::ClassNode::Member &member = clss->members[i]; | ||||
| 				ScriptLanguage::CodeCompletionOption option; | ||||
| 				switch (member.type) { | ||||
|  | @ -802,7 +887,7 @@ static void _find_identifiers_in_class(const GDScriptParser::ClassNode *p_class, | |||
| 						if (p_only_functions || outer || (p_static)) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.variable->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_MEMBER); | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.variable->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_MEMBER, location); | ||||
| 						break; | ||||
| 					case GDScriptParser::ClassNode::Member::CONSTANT: | ||||
| 						if (p_only_functions) { | ||||
|  | @ -811,7 +896,7 @@ static void _find_identifiers_in_class(const GDScriptParser::ClassNode *p_class, | |||
| 						if (r_result.has(member.constant->identifier->name)) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.constant->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT); | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.constant->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT, location); | ||||
| 						if (member.constant->initializer) { | ||||
| 							option.default_value = member.constant->initializer->reduced_value; | ||||
| 						} | ||||
|  | @ -820,25 +905,25 @@ static void _find_identifiers_in_class(const GDScriptParser::ClassNode *p_class, | |||
| 						if (p_only_functions) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.m_class->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS); | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.m_class->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CLASS, location); | ||||
| 						break; | ||||
| 					case GDScriptParser::ClassNode::Member::ENUM_VALUE: | ||||
| 						if (p_only_functions) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.enum_value.identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT); | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.enum_value.identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT, location); | ||||
| 						break; | ||||
| 					case GDScriptParser::ClassNode::Member::ENUM: | ||||
| 						if (p_only_functions) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.m_enum->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_ENUM); | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.m_enum->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_ENUM, location); | ||||
| 						break; | ||||
| 					case GDScriptParser::ClassNode::Member::FUNCTION: | ||||
| 						if (outer || (p_static && !member.function->is_static) || member.function->identifier->name.operator String().begins_with("@")) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.function->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_FUNCTION); | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.function->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_FUNCTION, location); | ||||
| 						if (member.function->parameters.size() > 0) { | ||||
| 							option.insert_text += "("; | ||||
| 						} else { | ||||
|  | @ -849,7 +934,7 @@ static void _find_identifiers_in_class(const GDScriptParser::ClassNode *p_class, | |||
| 						if (p_only_functions || outer) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.signal->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_SIGNAL); | ||||
| 						option = ScriptLanguage::CodeCompletionOption(member.signal->identifier->name, ScriptLanguage::CODE_COMPLETION_KIND_SIGNAL, location); | ||||
| 						break; | ||||
| 					case GDScriptParser::ClassNode::Member::UNDEFINED: | ||||
| 						break; | ||||
|  | @ -858,6 +943,7 @@ static void _find_identifiers_in_class(const GDScriptParser::ClassNode *p_class, | |||
| 			} | ||||
| 			outer = true; | ||||
| 			clss = clss->outer; | ||||
| 			classes_processed++; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
|  | @ -896,21 +982,24 @@ static void _find_identifiers_in_base(const GDScriptCompletionIdentifier &p_base | |||
| 							List<PropertyInfo> members; | ||||
| 							scr->get_script_property_list(&members); | ||||
| 							for (const PropertyInfo &E : members) { | ||||
| 								ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_MEMBER); | ||||
| 								int location = p_recursion_depth + _get_property_location(scr->get_class_name(), E.class_name); | ||||
| 								ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_MEMBER, location); | ||||
| 								r_result.insert(option.display, option); | ||||
| 							} | ||||
| 						} | ||||
| 						Map<StringName, Variant> constants; | ||||
| 						scr->get_constants(&constants); | ||||
| 						for (const KeyValue<StringName, Variant> &E : constants) { | ||||
| 							ScriptLanguage::CodeCompletionOption option(E.key.operator String(), ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT); | ||||
| 							int location = p_recursion_depth + _get_constant_location(scr->get_class_name(), E.key); | ||||
| 							ScriptLanguage::CodeCompletionOption option(E.key.operator String(), ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT, location); | ||||
| 							r_result.insert(option.display, option); | ||||
| 						} | ||||
| 
 | ||||
| 						List<MethodInfo> signals; | ||||
| 						scr->get_script_signal_list(&signals); | ||||
| 						for (const MethodInfo &E : signals) { | ||||
| 							ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_SIGNAL); | ||||
| 							int location = p_recursion_depth + _get_signal_location(scr->get_class_name(), E.name); | ||||
| 							ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_SIGNAL, location); | ||||
| 							r_result.insert(option.display, option); | ||||
| 						} | ||||
| 					} | ||||
|  | @ -921,7 +1010,8 @@ static void _find_identifiers_in_base(const GDScriptCompletionIdentifier &p_base | |||
| 						if (E.name.begins_with("@")) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_FUNCTION); | ||||
| 						int location = p_recursion_depth + _get_method_location(scr->get_class_name(), E.name); | ||||
| 						ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_FUNCTION, location); | ||||
| 						if (E.arguments.size()) { | ||||
| 							option.insert_text += "("; | ||||
| 						} else { | ||||
|  | @ -951,7 +1041,8 @@ static void _find_identifiers_in_base(const GDScriptCompletionIdentifier &p_base | |||
| 					List<String> constants; | ||||
| 					ClassDB::get_integer_constant_list(type, &constants); | ||||
| 					for (const String &E : constants) { | ||||
| 						ScriptLanguage::CodeCompletionOption option(E, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT); | ||||
| 						int location = p_recursion_depth + _get_constant_location(type, StringName(E)); | ||||
| 						ScriptLanguage::CodeCompletionOption option(E, ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT, location); | ||||
| 						r_result.insert(option.display, option); | ||||
| 					} | ||||
| 
 | ||||
|  | @ -965,7 +1056,8 @@ static void _find_identifiers_in_base(const GDScriptCompletionIdentifier &p_base | |||
| 							if (E.name.contains("/")) { | ||||
| 								continue; | ||||
| 							} | ||||
| 							ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_MEMBER); | ||||
| 							int location = p_recursion_depth + _get_property_location(type, E.class_name); | ||||
| 							ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_MEMBER, location); | ||||
| 							r_result.insert(option.display, option); | ||||
| 						} | ||||
| 					} | ||||
|  | @ -978,7 +1070,8 @@ static void _find_identifiers_in_base(const GDScriptCompletionIdentifier &p_base | |||
| 						if (E.name.begins_with("_")) { | ||||
| 							continue; | ||||
| 						} | ||||
| 						ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_FUNCTION); | ||||
| 						int location = p_recursion_depth + _get_method_location(type, E.name); | ||||
| 						ScriptLanguage::CodeCompletionOption option(E.name, ScriptLanguage::CODE_COMPLETION_KIND_FUNCTION, location); | ||||
| 						if (E.arguments.size()) { | ||||
| 							option.insert_text += "("; | ||||
| 						} else { | ||||
|  | @ -987,7 +1080,6 @@ static void _find_identifiers_in_base(const GDScriptCompletionIdentifier &p_base | |||
| 						r_result.insert(option.display, option); | ||||
| 					} | ||||
| 				} | ||||
| 
 | ||||
| 				return; | ||||
| 			} break; | ||||
| 			case GDScriptParser::DataType::BUILTIN: { | ||||
|  | @ -2247,7 +2339,8 @@ static void _find_enumeration_candidates(GDScriptParser::CompletionContext &p_co | |||
| 		ClassDB::get_enum_constants(class_name, enum_name, &enum_constants); | ||||
| 		for (const StringName &E : enum_constants) { | ||||
| 			String candidate = class_name + "." + E; | ||||
| 			ScriptLanguage::CodeCompletionOption option(candidate, ScriptLanguage::CODE_COMPLETION_KIND_ENUM); | ||||
| 			int location = _get_enum_constant_location(class_name, E); | ||||
| 			ScriptLanguage::CodeCompletionOption option(candidate, ScriptLanguage::CODE_COMPLETION_KIND_ENUM, location); | ||||
| 			r_result.insert(option.display, option); | ||||
| 		} | ||||
| 	} | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Eric M
						Eric M