C#: Add deprecated message to generated bindings
Uses the `DocData` deprecated message in the C# `[Obsolete]` attribute added to deprecated members.
This commit is contained in:
parent
f15ced3f18
commit
40b7dd0cf3
2 changed files with 653 additions and 39 deletions
|
@ -150,8 +150,268 @@ static String fix_doc_description(const String &p_bbcode) {
|
||||||
.strip_edges();
|
.strip_edges();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String BindingsGenerator::bbcode_to_text(const String &p_bbcode, const TypeInterface *p_itype) {
|
||||||
|
// Based on the version in EditorHelp.
|
||||||
|
|
||||||
|
if (p_bbcode.is_empty()) {
|
||||||
|
return String();
|
||||||
|
}
|
||||||
|
|
||||||
|
DocTools *doc = EditorHelp::get_doc_data();
|
||||||
|
|
||||||
|
String bbcode = p_bbcode;
|
||||||
|
|
||||||
|
StringBuilder output;
|
||||||
|
|
||||||
|
List<String> tag_stack;
|
||||||
|
bool code_tag = false;
|
||||||
|
|
||||||
|
int pos = 0;
|
||||||
|
while (pos < bbcode.length()) {
|
||||||
|
int brk_pos = bbcode.find("[", pos);
|
||||||
|
|
||||||
|
if (brk_pos < 0) {
|
||||||
|
brk_pos = bbcode.length();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (brk_pos > pos) {
|
||||||
|
String text = bbcode.substr(pos, brk_pos - pos);
|
||||||
|
if (code_tag || tag_stack.size() > 0) {
|
||||||
|
output.append("'" + text + "'");
|
||||||
|
} else {
|
||||||
|
output.append(text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (brk_pos == bbcode.length()) {
|
||||||
|
// Nothing else to add.
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
int brk_end = bbcode.find("]", brk_pos + 1);
|
||||||
|
|
||||||
|
if (brk_end == -1) {
|
||||||
|
String text = bbcode.substr(brk_pos, bbcode.length() - brk_pos);
|
||||||
|
if (code_tag || tag_stack.size() > 0) {
|
||||||
|
output.append("'" + text + "'");
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
String tag = bbcode.substr(brk_pos + 1, brk_end - brk_pos - 1);
|
||||||
|
|
||||||
|
if (tag.begins_with("/")) {
|
||||||
|
bool tag_ok = tag_stack.size() && tag_stack.front()->get() == tag.substr(1, tag.length());
|
||||||
|
|
||||||
|
if (!tag_ok) {
|
||||||
|
output.append("]");
|
||||||
|
pos = brk_pos + 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
tag_stack.pop_front();
|
||||||
|
pos = brk_end + 1;
|
||||||
|
code_tag = false;
|
||||||
|
} else if (code_tag) {
|
||||||
|
output.append("[");
|
||||||
|
pos = brk_pos + 1;
|
||||||
|
} else if (tag.begins_with("method ") || tag.begins_with("constructor ") || tag.begins_with("operator ") || tag.begins_with("member ") || tag.begins_with("signal ") || tag.begins_with("enum ") || tag.begins_with("constant ") || tag.begins_with("theme_item ") || tag.begins_with("param ")) {
|
||||||
|
const int tag_end = tag.find(" ");
|
||||||
|
const String link_tag = tag.substr(0, tag_end);
|
||||||
|
const String link_target = tag.substr(tag_end + 1, tag.length()).lstrip(" ");
|
||||||
|
|
||||||
|
const Vector<String> link_target_parts = link_target.split(".");
|
||||||
|
|
||||||
|
if (link_target_parts.size() <= 0 || link_target_parts.size() > 2) {
|
||||||
|
ERR_PRINT("Invalid reference format: '" + tag + "'.");
|
||||||
|
|
||||||
|
output.append(tag);
|
||||||
|
|
||||||
|
pos = brk_end + 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const TypeInterface *target_itype;
|
||||||
|
StringName target_cname;
|
||||||
|
|
||||||
|
if (link_target_parts.size() == 2) {
|
||||||
|
target_itype = _get_type_or_null(TypeReference(link_target_parts[0]));
|
||||||
|
if (!target_itype) {
|
||||||
|
target_itype = _get_type_or_null(TypeReference("_" + link_target_parts[0]));
|
||||||
|
}
|
||||||
|
target_cname = link_target_parts[1];
|
||||||
|
} else {
|
||||||
|
target_itype = p_itype;
|
||||||
|
target_cname = link_target_parts[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (link_tag == "method") {
|
||||||
|
_append_text_method(output, target_itype, target_cname, link_target, link_target_parts);
|
||||||
|
} else if (link_tag == "constructor") {
|
||||||
|
// TODO: Support constructors?
|
||||||
|
_append_text_undeclared(output, link_target);
|
||||||
|
} else if (link_tag == "operator") {
|
||||||
|
// TODO: Support operators?
|
||||||
|
_append_text_undeclared(output, link_target);
|
||||||
|
} else if (link_tag == "member") {
|
||||||
|
_append_text_member(output, target_itype, target_cname, link_target, link_target_parts);
|
||||||
|
} else if (link_tag == "signal") {
|
||||||
|
_append_text_signal(output, target_itype, target_cname, link_target, link_target_parts);
|
||||||
|
} else if (link_tag == "enum") {
|
||||||
|
_append_text_enum(output, target_itype, target_cname, link_target, link_target_parts);
|
||||||
|
} else if (link_tag == "constant") {
|
||||||
|
_append_text_constant(output, target_itype, target_cname, link_target, link_target_parts);
|
||||||
|
} else if (link_tag == "param") {
|
||||||
|
_append_text_param(output, link_target);
|
||||||
|
} else if (link_tag == "theme_item") {
|
||||||
|
// We do not declare theme_items in any way in C#, so there is nothing to reference.
|
||||||
|
_append_text_undeclared(output, link_target);
|
||||||
|
}
|
||||||
|
|
||||||
|
pos = brk_end + 1;
|
||||||
|
} else if (doc->class_list.has(tag)) {
|
||||||
|
if (tag == "Array" || tag == "Dictionary") {
|
||||||
|
output.append("'" BINDINGS_NAMESPACE_COLLECTIONS ".");
|
||||||
|
output.append(tag);
|
||||||
|
output.append("'");
|
||||||
|
} else if (tag == "bool" || tag == "int") {
|
||||||
|
output.append(tag);
|
||||||
|
} else if (tag == "float") {
|
||||||
|
output.append(
|
||||||
|
#ifdef REAL_T_IS_DOUBLE
|
||||||
|
"double"
|
||||||
|
#else
|
||||||
|
"float"
|
||||||
|
#endif
|
||||||
|
);
|
||||||
|
} else if (tag == "Variant") {
|
||||||
|
output.append("'Godot.Variant'");
|
||||||
|
} else if (tag == "String") {
|
||||||
|
output.append("string");
|
||||||
|
} else if (tag == "Nil") {
|
||||||
|
output.append("null");
|
||||||
|
} else if (tag.begins_with("@")) {
|
||||||
|
// @GlobalScope, @GDScript, etc.
|
||||||
|
output.append("'" + tag + "'");
|
||||||
|
} else if (tag == "PackedByteArray") {
|
||||||
|
output.append("byte[]");
|
||||||
|
} else if (tag == "PackedInt32Array") {
|
||||||
|
output.append("int[]");
|
||||||
|
} else if (tag == "PackedInt64Array") {
|
||||||
|
output.append("long[]");
|
||||||
|
} else if (tag == "PackedFloat32Array") {
|
||||||
|
output.append("float[]");
|
||||||
|
} else if (tag == "PackedFloat64Array") {
|
||||||
|
output.append("double[]");
|
||||||
|
} else if (tag == "PackedStringArray") {
|
||||||
|
output.append("string[]");
|
||||||
|
} else if (tag == "PackedVector2Array") {
|
||||||
|
output.append("'" BINDINGS_NAMESPACE ".Vector2[]'");
|
||||||
|
} else if (tag == "PackedVector3Array") {
|
||||||
|
output.append("'" BINDINGS_NAMESPACE ".Vector3[]'");
|
||||||
|
} else if (tag == "PackedColorArray") {
|
||||||
|
output.append("'" BINDINGS_NAMESPACE ".Color[]'");
|
||||||
|
} else {
|
||||||
|
const TypeInterface *target_itype = _get_type_or_null(TypeReference(tag));
|
||||||
|
|
||||||
|
if (!target_itype) {
|
||||||
|
target_itype = _get_type_or_null(TypeReference("_" + tag));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (target_itype) {
|
||||||
|
output.append("'" + target_itype->proxy_name + "'");
|
||||||
|
} else {
|
||||||
|
ERR_PRINT("Cannot resolve type reference in documentation: '" + tag + "'.");
|
||||||
|
output.append("'" + tag + "'");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pos = brk_end + 1;
|
||||||
|
} else if (tag == "b") {
|
||||||
|
// Bold is not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag == "i") {
|
||||||
|
// Italic is not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag == "code" || tag.begins_with("code ")) {
|
||||||
|
code_tag = true;
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front("code");
|
||||||
|
} else if (tag == "kbd") {
|
||||||
|
// Keyboard combinations are not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag == "center") {
|
||||||
|
// Center alignment is not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag == "br") {
|
||||||
|
// Break is not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag == "u") {
|
||||||
|
// Underline is not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag == "s") {
|
||||||
|
// Strikethrough is not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag == "url") {
|
||||||
|
int end = bbcode.find("[", brk_end);
|
||||||
|
if (end == -1) {
|
||||||
|
end = bbcode.length();
|
||||||
|
}
|
||||||
|
String url = bbcode.substr(brk_end + 1, end - brk_end - 1);
|
||||||
|
// Not supported. Just append the url.
|
||||||
|
output.append(url);
|
||||||
|
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag.begins_with("url=")) {
|
||||||
|
String url = tag.substr(4, tag.length());
|
||||||
|
// Not supported. Just append the url.
|
||||||
|
output.append(url);
|
||||||
|
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front("url");
|
||||||
|
} else if (tag == "img") {
|
||||||
|
int end = bbcode.find("[", brk_end);
|
||||||
|
if (end == -1) {
|
||||||
|
end = bbcode.length();
|
||||||
|
}
|
||||||
|
String image = bbcode.substr(brk_end + 1, end - brk_end - 1);
|
||||||
|
|
||||||
|
// Not supported. Just append the bbcode.
|
||||||
|
output.append("[img]");
|
||||||
|
output.append(image);
|
||||||
|
output.append("[/img]");
|
||||||
|
|
||||||
|
pos = end;
|
||||||
|
tag_stack.push_front(tag);
|
||||||
|
} else if (tag.begins_with("color=")) {
|
||||||
|
// Not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front("color");
|
||||||
|
} else if (tag.begins_with("font=")) {
|
||||||
|
// Not supported.
|
||||||
|
pos = brk_end + 1;
|
||||||
|
tag_stack.push_front("font");
|
||||||
|
} else {
|
||||||
|
// Ignore unrecognized tag.
|
||||||
|
output.append("[");
|
||||||
|
pos = brk_pos + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output.as_string();
|
||||||
|
}
|
||||||
|
|
||||||
String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterface *p_itype, bool p_is_signal) {
|
String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterface *p_itype, bool p_is_signal) {
|
||||||
// Based on the version in EditorHelp
|
// Based on the version in EditorHelp.
|
||||||
|
|
||||||
if (p_bbcode.is_empty()) {
|
if (p_bbcode.is_empty()) {
|
||||||
return String();
|
return String();
|
||||||
|
@ -200,7 +460,8 @@ String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterf
|
||||||
}
|
}
|
||||||
|
|
||||||
if (brk_pos == bbcode.length()) {
|
if (brk_pos == bbcode.length()) {
|
||||||
break; // nothing else to add
|
// Nothing else to add.
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
int brk_end = bbcode.find("]", brk_pos + 1);
|
int brk_end = bbcode.find("]", brk_pos + 1);
|
||||||
|
@ -316,7 +577,7 @@ String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterf
|
||||||
} else if (link_tag == "param") {
|
} else if (link_tag == "param") {
|
||||||
_append_xml_param(xml_output, link_target, p_is_signal);
|
_append_xml_param(xml_output, link_target, p_is_signal);
|
||||||
} else if (link_tag == "theme_item") {
|
} else if (link_tag == "theme_item") {
|
||||||
// We do not declare theme_items in any way in C#, so there is nothing to reference
|
// We do not declare theme_items in any way in C#, so there is nothing to reference.
|
||||||
_append_xml_undeclared(xml_output, link_target);
|
_append_xml_undeclared(xml_output, link_target);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -345,7 +606,7 @@ String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterf
|
||||||
} else if (tag == "Nil") {
|
} else if (tag == "Nil") {
|
||||||
xml_output.append("<see langword=\"null\"/>");
|
xml_output.append("<see langword=\"null\"/>");
|
||||||
} else if (tag.begins_with("@")) {
|
} else if (tag.begins_with("@")) {
|
||||||
// @GlobalScope, @GDScript, etc
|
// @GlobalScope, @GDScript, etc.
|
||||||
xml_output.append("<c>");
|
xml_output.append("<c>");
|
||||||
xml_output.append(tag);
|
xml_output.append(tag);
|
||||||
xml_output.append("</c>");
|
xml_output.append("</c>");
|
||||||
|
@ -432,22 +693,22 @@ String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterf
|
||||||
pos = brk_end + 1;
|
pos = brk_end + 1;
|
||||||
tag_stack.push_front("csharp");
|
tag_stack.push_front("csharp");
|
||||||
} else if (tag == "kbd") {
|
} else if (tag == "kbd") {
|
||||||
// keyboard combinations are not supported in xml comments
|
// Keyboard combinations are not supported in xml comments.
|
||||||
pos = brk_end + 1;
|
pos = brk_end + 1;
|
||||||
tag_stack.push_front(tag);
|
tag_stack.push_front(tag);
|
||||||
} else if (tag == "center") {
|
} else if (tag == "center") {
|
||||||
// center alignment is not supported in xml comments
|
// Center alignment is not supported in xml comments.
|
||||||
pos = brk_end + 1;
|
pos = brk_end + 1;
|
||||||
tag_stack.push_front(tag);
|
tag_stack.push_front(tag);
|
||||||
} else if (tag == "br") {
|
} else if (tag == "br") {
|
||||||
xml_output.append("\n"); // FIXME: Should use <para> instead. Luckily this tag isn't used for now.
|
xml_output.append("\n"); // FIXME: Should use <para> instead. Luckily this tag isn't used for now.
|
||||||
pos = brk_end + 1;
|
pos = brk_end + 1;
|
||||||
} else if (tag == "u") {
|
} else if (tag == "u") {
|
||||||
// underline is not supported in Rider xml comments
|
// Underline is not supported in Rider xml comments.
|
||||||
pos = brk_end + 1;
|
pos = brk_end + 1;
|
||||||
tag_stack.push_front(tag);
|
tag_stack.push_front(tag);
|
||||||
} else if (tag == "s") {
|
} else if (tag == "s") {
|
||||||
// strikethrough is not supported in xml comments
|
// Strikethrough is not supported in xml comments.
|
||||||
pos = brk_end + 1;
|
pos = brk_end + 1;
|
||||||
tag_stack.push_front(tag);
|
tag_stack.push_front(tag);
|
||||||
} else if (tag == "url") {
|
} else if (tag == "url") {
|
||||||
|
@ -495,7 +756,8 @@ String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterf
|
||||||
tag_stack.push_front("font");
|
tag_stack.push_front("font");
|
||||||
} else {
|
} else {
|
||||||
if (!line_del) {
|
if (!line_del) {
|
||||||
xml_output.append("["); // ignore
|
// Ignore unrecognized tag.
|
||||||
|
xml_output.append("[");
|
||||||
}
|
}
|
||||||
pos = brk_pos + 1;
|
pos = brk_pos + 1;
|
||||||
}
|
}
|
||||||
|
@ -506,6 +768,285 @@ String BindingsGenerator::bbcode_to_xml(const String &p_bbcode, const TypeInterf
|
||||||
return xml_output.as_string();
|
return xml_output.as_string();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_method(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts) {
|
||||||
|
if (p_link_target_parts[0] == name_cache.type_at_GlobalScope) {
|
||||||
|
if (OS::get_singleton()->is_stdout_verbose()) {
|
||||||
|
OS::get_singleton()->print("Cannot resolve @GlobalScope method reference in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Map what we can
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
} else if (!p_target_itype || !p_target_itype->is_object_type) {
|
||||||
|
if (OS::get_singleton()->is_stdout_verbose()) {
|
||||||
|
if (p_target_itype) {
|
||||||
|
OS::get_singleton()->print("Cannot resolve method reference for non-GodotObject type in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
} else {
|
||||||
|
OS::get_singleton()->print("Cannot resolve type from method reference in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Map what we can
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
} else {
|
||||||
|
if (p_target_cname == "_init") {
|
||||||
|
// The _init method is not declared in C#, reference the constructor instead
|
||||||
|
p_output.append("'new " BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(p_target_itype->proxy_name);
|
||||||
|
p_output.append("()'");
|
||||||
|
} else {
|
||||||
|
const MethodInterface *target_imethod = p_target_itype->find_method_by_name(p_target_cname);
|
||||||
|
|
||||||
|
if (target_imethod) {
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(p_target_itype->proxy_name);
|
||||||
|
p_output.append(".");
|
||||||
|
p_output.append(target_imethod->proxy_name);
|
||||||
|
p_output.append("(");
|
||||||
|
bool first_key = true;
|
||||||
|
for (const ArgumentInterface &iarg : target_imethod->arguments) {
|
||||||
|
const TypeInterface *arg_type = _get_type_or_null(iarg.type);
|
||||||
|
|
||||||
|
if (first_key) {
|
||||||
|
first_key = false;
|
||||||
|
} else {
|
||||||
|
p_output.append(", ");
|
||||||
|
}
|
||||||
|
if (!arg_type) {
|
||||||
|
ERR_PRINT("Cannot resolve argument type in documentation: '" + p_link_target + "'.");
|
||||||
|
p_output.append(iarg.type.cname);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (iarg.def_param_mode == ArgumentInterface::NULLABLE_VAL) {
|
||||||
|
p_output.append("Nullable<");
|
||||||
|
}
|
||||||
|
String arg_cs_type = arg_type->cs_type + _get_generic_type_parameters(*arg_type, iarg.type.generic_type_parameters);
|
||||||
|
p_output.append(arg_cs_type.replacen("params ", ""));
|
||||||
|
if (iarg.def_param_mode == ArgumentInterface::NULLABLE_VAL) {
|
||||||
|
p_output.append(">");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
p_output.append(")'");
|
||||||
|
} else {
|
||||||
|
if (!p_target_itype->is_intentionally_ignored(p_link_target)) {
|
||||||
|
ERR_PRINT("Cannot resolve method reference in documentation: '" + p_link_target + "'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_member(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts) {
|
||||||
|
if (p_link_target.find("/") >= 0) {
|
||||||
|
// Properties with '/' (slash) in the name are not declared in C#, so there is nothing to reference.
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
} else if (!p_target_itype || !p_target_itype->is_object_type) {
|
||||||
|
if (OS::get_singleton()->is_stdout_verbose()) {
|
||||||
|
if (p_target_itype) {
|
||||||
|
OS::get_singleton()->print("Cannot resolve member reference for non-GodotObject type in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
} else {
|
||||||
|
OS::get_singleton()->print("Cannot resolve type from member reference in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Map what we can
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
} else {
|
||||||
|
const TypeInterface *current_itype = p_target_itype;
|
||||||
|
const PropertyInterface *target_iprop = nullptr;
|
||||||
|
|
||||||
|
while (target_iprop == nullptr && current_itype != nullptr) {
|
||||||
|
target_iprop = current_itype->find_property_by_name(p_target_cname);
|
||||||
|
if (target_iprop == nullptr) {
|
||||||
|
current_itype = _get_type_or_null(TypeReference(current_itype->base_name));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (target_iprop) {
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(current_itype->proxy_name);
|
||||||
|
p_output.append(".");
|
||||||
|
p_output.append(target_iprop->proxy_name);
|
||||||
|
p_output.append("'");
|
||||||
|
} else {
|
||||||
|
if (!p_target_itype->is_intentionally_ignored(p_link_target)) {
|
||||||
|
ERR_PRINT("Cannot resolve member reference in documentation: '" + p_link_target + "'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_signal(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts) {
|
||||||
|
if (!p_target_itype || !p_target_itype->is_object_type) {
|
||||||
|
if (OS::get_singleton()->is_stdout_verbose()) {
|
||||||
|
if (p_target_itype) {
|
||||||
|
OS::get_singleton()->print("Cannot resolve signal reference for non-GodotObject type in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
} else {
|
||||||
|
OS::get_singleton()->print("Cannot resolve type from signal reference in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Map what we can
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
} else {
|
||||||
|
const SignalInterface *target_isignal = p_target_itype->find_signal_by_name(p_target_cname);
|
||||||
|
|
||||||
|
if (target_isignal) {
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(p_target_itype->proxy_name);
|
||||||
|
p_output.append(".");
|
||||||
|
p_output.append(target_isignal->proxy_name);
|
||||||
|
p_output.append("'");
|
||||||
|
} else {
|
||||||
|
if (!p_target_itype->is_intentionally_ignored(p_link_target)) {
|
||||||
|
ERR_PRINT("Cannot resolve signal reference in documentation: '" + p_link_target + "'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_enum(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts) {
|
||||||
|
const StringName search_cname = !p_target_itype ? p_target_cname : StringName(p_target_itype->name + "." + (String)p_target_cname);
|
||||||
|
|
||||||
|
HashMap<StringName, TypeInterface>::ConstIterator enum_match = enum_types.find(search_cname);
|
||||||
|
|
||||||
|
if (!enum_match && search_cname != p_target_cname) {
|
||||||
|
enum_match = enum_types.find(p_target_cname);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (enum_match) {
|
||||||
|
const TypeInterface &target_enum_itype = enum_match->value;
|
||||||
|
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(target_enum_itype.proxy_name); // Includes nesting class if any
|
||||||
|
p_output.append("'");
|
||||||
|
} else {
|
||||||
|
if (!p_target_itype->is_intentionally_ignored(p_link_target)) {
|
||||||
|
ERR_PRINT("Cannot resolve enum reference in documentation: '" + p_link_target + "'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_constant(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts) {
|
||||||
|
if (p_link_target_parts[0] == name_cache.type_at_GlobalScope) {
|
||||||
|
_append_text_constant_in_global_scope(p_output, p_target_cname, p_link_target);
|
||||||
|
} else if (!p_target_itype || !p_target_itype->is_object_type) {
|
||||||
|
// Search in @GlobalScope as a last resort if no class was specified
|
||||||
|
if (p_link_target_parts.size() == 1) {
|
||||||
|
_append_text_constant_in_global_scope(p_output, p_target_cname, p_link_target);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (OS::get_singleton()->is_stdout_verbose()) {
|
||||||
|
if (p_target_itype) {
|
||||||
|
OS::get_singleton()->print("Cannot resolve constant reference for non-GodotObject type in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
} else {
|
||||||
|
OS::get_singleton()->print("Cannot resolve type from constant reference in documentation: %s\n", p_link_target.utf8().get_data());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO Map what we can
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
} else {
|
||||||
|
// Try to find the constant in the current class
|
||||||
|
if (p_target_itype->is_singleton_instance) {
|
||||||
|
// Constants and enums are declared in the static singleton class.
|
||||||
|
p_target_itype = &obj_types[p_target_itype->cname];
|
||||||
|
}
|
||||||
|
|
||||||
|
const ConstantInterface *target_iconst = find_constant_by_name(p_target_cname, p_target_itype->constants);
|
||||||
|
|
||||||
|
if (target_iconst) {
|
||||||
|
// Found constant in current class
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(p_target_itype->proxy_name);
|
||||||
|
p_output.append(".");
|
||||||
|
p_output.append(target_iconst->proxy_name);
|
||||||
|
p_output.append("'");
|
||||||
|
} else {
|
||||||
|
// Try to find as enum constant in the current class
|
||||||
|
const EnumInterface *target_ienum = nullptr;
|
||||||
|
|
||||||
|
for (const EnumInterface &ienum : p_target_itype->enums) {
|
||||||
|
target_ienum = &ienum;
|
||||||
|
target_iconst = find_constant_by_name(p_target_cname, target_ienum->constants);
|
||||||
|
if (target_iconst) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (target_iconst) {
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(p_target_itype->proxy_name);
|
||||||
|
p_output.append(".");
|
||||||
|
p_output.append(target_ienum->proxy_name);
|
||||||
|
p_output.append(".");
|
||||||
|
p_output.append(target_iconst->proxy_name);
|
||||||
|
p_output.append("'");
|
||||||
|
} else if (p_link_target_parts.size() == 1) {
|
||||||
|
// Also search in @GlobalScope as a last resort if no class was specified
|
||||||
|
_append_text_constant_in_global_scope(p_output, p_target_cname, p_link_target);
|
||||||
|
} else {
|
||||||
|
if (!p_target_itype->is_intentionally_ignored(p_link_target)) {
|
||||||
|
ERR_PRINT("Cannot resolve constant reference in documentation: '" + p_link_target + "'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
_append_xml_undeclared(p_output, p_link_target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_constant_in_global_scope(StringBuilder &p_output, const String &p_target_cname, const String &p_link_target) {
|
||||||
|
// Try to find as a global constant
|
||||||
|
const ConstantInterface *target_iconst = find_constant_by_name(p_target_cname, global_constants);
|
||||||
|
|
||||||
|
if (target_iconst) {
|
||||||
|
// Found global constant
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE "." BINDINGS_GLOBAL_SCOPE_CLASS ".");
|
||||||
|
p_output.append(target_iconst->proxy_name);
|
||||||
|
p_output.append("'");
|
||||||
|
} else {
|
||||||
|
// Try to find as global enum constant
|
||||||
|
const EnumInterface *target_ienum = nullptr;
|
||||||
|
|
||||||
|
for (const EnumInterface &ienum : global_enums) {
|
||||||
|
target_ienum = &ienum;
|
||||||
|
target_iconst = find_constant_by_name(p_target_cname, target_ienum->constants);
|
||||||
|
if (target_iconst) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (target_iconst) {
|
||||||
|
p_output.append("'" BINDINGS_NAMESPACE ".");
|
||||||
|
p_output.append(target_ienum->proxy_name);
|
||||||
|
p_output.append(".");
|
||||||
|
p_output.append(target_iconst->proxy_name);
|
||||||
|
p_output.append("'");
|
||||||
|
} else {
|
||||||
|
ERR_PRINT("Cannot resolve global constant reference in documentation: '" + p_link_target + "'.");
|
||||||
|
_append_text_undeclared(p_output, p_link_target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_param(StringBuilder &p_output, const String &p_link_target) {
|
||||||
|
const String link_target = snake_to_camel_case(p_link_target);
|
||||||
|
p_output.append("'" + link_target + "'");
|
||||||
|
}
|
||||||
|
|
||||||
|
void BindingsGenerator::_append_text_undeclared(StringBuilder &p_output, const String &p_link_target) {
|
||||||
|
p_output.append("'" + p_link_target + "'");
|
||||||
|
}
|
||||||
|
|
||||||
void BindingsGenerator::_append_xml_method(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts) {
|
void BindingsGenerator::_append_xml_method(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts) {
|
||||||
if (p_link_target_parts[0] == name_cache.type_at_GlobalScope) {
|
if (p_link_target_parts[0] == name_cache.type_at_GlobalScope) {
|
||||||
if (OS::get_singleton()->is_stdout_verbose()) {
|
if (OS::get_singleton()->is_stdout_verbose()) {
|
||||||
|
@ -1429,10 +1970,12 @@ Error BindingsGenerator::_generate_cs_type(const TypeInterface &itype, const Str
|
||||||
|
|
||||||
output.append("/// </summary>\n");
|
output.append("/// </summary>\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (class_doc->is_deprecated) {
|
|
||||||
output.append("[Obsolete(\"This class is deprecated.\")]\n");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (itype.is_deprecated) {
|
||||||
|
output.append("[Obsolete(\"");
|
||||||
|
output.append(bbcode_to_text(itype.deprecation_message, &itype));
|
||||||
|
output.append("\")]\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
// We generate a `GodotClassName` attribute if the engine class name is not the same as the
|
// We generate a `GodotClassName` attribute if the engine class name is not the same as the
|
||||||
|
@ -1489,10 +2032,12 @@ Error BindingsGenerator::_generate_cs_type(const TypeInterface &itype, const Str
|
||||||
|
|
||||||
output.append(INDENT1 "/// </summary>");
|
output.append(INDENT1 "/// </summary>");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (iconstant.const_doc->is_deprecated) {
|
|
||||||
output.append(MEMBER_BEGIN "[Obsolete(\"This constant is deprecated.\")]");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (iconstant.is_deprecated) {
|
||||||
|
output.append(MEMBER_BEGIN "[Obsolete(\"");
|
||||||
|
output.append(bbcode_to_text(iconstant.deprecation_message, &itype));
|
||||||
|
output.append("\")]");
|
||||||
}
|
}
|
||||||
|
|
||||||
output.append(MEMBER_BEGIN "public const long ");
|
output.append(MEMBER_BEGIN "public const long ");
|
||||||
|
@ -1537,10 +2082,12 @@ Error BindingsGenerator::_generate_cs_type(const TypeInterface &itype, const Str
|
||||||
|
|
||||||
output.append(INDENT2 "/// </summary>\n");
|
output.append(INDENT2 "/// </summary>\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (iconstant.const_doc->is_deprecated) {
|
|
||||||
output.append(INDENT2 "[Obsolete(\"This enum member is deprecated.\")]\n");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (iconstant.is_deprecated) {
|
||||||
|
output.append(INDENT2 "[Obsolete(\"");
|
||||||
|
output.append(bbcode_to_text(iconstant.deprecation_message, &itype));
|
||||||
|
output.append("\")]\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
output.append(INDENT2);
|
output.append(INDENT2);
|
||||||
|
@ -1992,10 +2539,12 @@ Error BindingsGenerator::_generate_cs_property(const BindingsGenerator::TypeInte
|
||||||
|
|
||||||
p_output.append(INDENT1 "/// </summary>");
|
p_output.append(INDENT1 "/// </summary>");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (p_iprop.prop_doc->is_deprecated) {
|
|
||||||
p_output.append(MEMBER_BEGIN "[Obsolete(\"This property is deprecated.\")]");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (p_iprop.is_deprecated) {
|
||||||
|
p_output.append(MEMBER_BEGIN "[Obsolete(\"");
|
||||||
|
p_output.append(bbcode_to_text(p_iprop.deprecation_message, &p_itype));
|
||||||
|
p_output.append("\")]");
|
||||||
}
|
}
|
||||||
|
|
||||||
p_output.append(MEMBER_BEGIN "public ");
|
p_output.append(MEMBER_BEGIN "public ");
|
||||||
|
@ -2248,15 +2797,9 @@ Error BindingsGenerator::_generate_cs_method(const BindingsGenerator::TypeInterf
|
||||||
}
|
}
|
||||||
|
|
||||||
if (p_imethod.is_deprecated) {
|
if (p_imethod.is_deprecated) {
|
||||||
if (p_imethod.deprecation_message.is_empty()) {
|
|
||||||
WARN_PRINT("An empty deprecation message is discouraged. Method: '" + p_imethod.proxy_name + "'.");
|
|
||||||
}
|
|
||||||
|
|
||||||
p_output.append(MEMBER_BEGIN "[Obsolete(\"");
|
p_output.append(MEMBER_BEGIN "[Obsolete(\"");
|
||||||
p_output.append(p_imethod.deprecation_message);
|
p_output.append(bbcode_to_text(p_imethod.deprecation_message, &p_itype));
|
||||||
p_output.append("\")]");
|
p_output.append("\")]");
|
||||||
} else if (p_imethod.method_doc && p_imethod.method_doc->is_deprecated) {
|
|
||||||
p_output.append(MEMBER_BEGIN "[Obsolete(\"This method is deprecated.\")]");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (p_imethod.is_compat) {
|
if (p_imethod.is_compat) {
|
||||||
|
@ -2401,12 +2944,8 @@ Error BindingsGenerator::_generate_cs_signal(const BindingsGenerator::TypeInterf
|
||||||
p_output.append(INDENT1 "/// </summary>");
|
p_output.append(INDENT1 "/// </summary>");
|
||||||
|
|
||||||
if (p_isignal.is_deprecated) {
|
if (p_isignal.is_deprecated) {
|
||||||
if (p_isignal.deprecation_message.is_empty()) {
|
|
||||||
WARN_PRINT("An empty deprecation message is discouraged. Signal: '" + p_isignal.proxy_name + "'.");
|
|
||||||
}
|
|
||||||
|
|
||||||
p_output.append(MEMBER_BEGIN "[Obsolete(\"");
|
p_output.append(MEMBER_BEGIN "[Obsolete(\"");
|
||||||
p_output.append(p_isignal.deprecation_message);
|
p_output.append(bbcode_to_text(p_isignal.deprecation_message, &p_itype));
|
||||||
p_output.append("\")]");
|
p_output.append("\")]");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2459,15 +2998,11 @@ Error BindingsGenerator::_generate_cs_signal(const BindingsGenerator::TypeInterf
|
||||||
|
|
||||||
p_output.append(INDENT1 "/// </summary>");
|
p_output.append(INDENT1 "/// </summary>");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (p_isignal.method_doc->is_deprecated) {
|
|
||||||
p_output.append(MEMBER_BEGIN "[Obsolete(\"This signal is deprecated.\")]");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (p_isignal.is_deprecated) {
|
if (p_isignal.is_deprecated) {
|
||||||
p_output.append(MEMBER_BEGIN "[Obsolete(\"");
|
p_output.append(MEMBER_BEGIN "[Obsolete(\"");
|
||||||
p_output.append(p_isignal.deprecation_message);
|
p_output.append(bbcode_to_text(p_isignal.deprecation_message, &p_itype));
|
||||||
p_output.append("\")]");
|
p_output.append("\")]");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3029,6 +3564,16 @@ bool BindingsGenerator::_populate_object_type_interfaces() {
|
||||||
itype.is_ref_counted = ClassDB::is_parent_class(type_cname, name_cache.type_RefCounted);
|
itype.is_ref_counted = ClassDB::is_parent_class(type_cname, name_cache.type_RefCounted);
|
||||||
itype.memory_own = itype.is_ref_counted;
|
itype.memory_own = itype.is_ref_counted;
|
||||||
|
|
||||||
|
if (itype.class_doc) {
|
||||||
|
itype.is_deprecated = itype.class_doc->is_deprecated;
|
||||||
|
itype.deprecation_message = itype.class_doc->deprecated_message;
|
||||||
|
|
||||||
|
if (itype.deprecation_message.is_empty()) {
|
||||||
|
WARN_PRINT("An empty deprecation message is discouraged. Type: '" + itype.proxy_name + "'.");
|
||||||
|
itype.deprecation_message = "This class is deprecated.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (itype.is_singleton && compat_singletons.has(itype.cname)) {
|
if (itype.is_singleton && compat_singletons.has(itype.cname)) {
|
||||||
itype.is_singleton = false;
|
itype.is_singleton = false;
|
||||||
itype.is_compat_singleton = true;
|
itype.is_compat_singleton = true;
|
||||||
|
@ -3103,6 +3648,16 @@ bool BindingsGenerator::_populate_object_type_interfaces() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (iprop.prop_doc) {
|
||||||
|
iprop.is_deprecated = iprop.prop_doc->is_deprecated;
|
||||||
|
iprop.deprecation_message = iprop.prop_doc->deprecated_message;
|
||||||
|
|
||||||
|
if (iprop.deprecation_message.is_empty()) {
|
||||||
|
WARN_PRINT("An empty deprecation message is discouraged. Property: '" + itype.proxy_name + "." + iprop.proxy_name + "'.");
|
||||||
|
iprop.deprecation_message = "This property is deprecated.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
itype.properties.push_back(iprop);
|
itype.properties.push_back(iprop);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3282,6 +3837,16 @@ bool BindingsGenerator::_populate_object_type_interfaces() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (imethod.method_doc) {
|
||||||
|
imethod.is_deprecated = imethod.method_doc->is_deprecated;
|
||||||
|
imethod.deprecation_message = imethod.method_doc->deprecated_message;
|
||||||
|
|
||||||
|
if (imethod.deprecation_message.is_empty()) {
|
||||||
|
WARN_PRINT("An empty deprecation message is discouraged. Method: '" + itype.proxy_name + "." + imethod.proxy_name + "'.");
|
||||||
|
imethod.deprecation_message = "This method is deprecated.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ERR_FAIL_COND_V_MSG(itype.find_property_by_name(imethod.cname), false,
|
ERR_FAIL_COND_V_MSG(itype.find_property_by_name(imethod.cname), false,
|
||||||
"Method name conflicts with property: '" + itype.name + "." + imethod.name + "'.");
|
"Method name conflicts with property: '" + itype.name + "." + imethod.name + "'.");
|
||||||
|
|
||||||
|
@ -3388,6 +3953,16 @@ bool BindingsGenerator::_populate_object_type_interfaces() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isignal.method_doc) {
|
||||||
|
isignal.is_deprecated = isignal.method_doc->is_deprecated;
|
||||||
|
isignal.deprecation_message = isignal.method_doc->deprecated_message;
|
||||||
|
|
||||||
|
if (isignal.deprecation_message.is_empty()) {
|
||||||
|
WARN_PRINT("An empty deprecation message is discouraged. Signal: '" + itype.proxy_name + "." + isignal.proxy_name + "'.");
|
||||||
|
isignal.deprecation_message = "This signal is deprecated.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
itype.signals_.push_back(isignal);
|
itype.signals_.push_back(isignal);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3428,6 +4003,16 @@ bool BindingsGenerator::_populate_object_type_interfaces() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (iconstant.const_doc) {
|
||||||
|
iconstant.is_deprecated = iconstant.const_doc->is_deprecated;
|
||||||
|
iconstant.deprecation_message = iconstant.const_doc->deprecated_message;
|
||||||
|
|
||||||
|
if (iconstant.deprecation_message.is_empty()) {
|
||||||
|
WARN_PRINT("An empty deprecation message is discouraged. Enum member: '" + itype.proxy_name + "." + ienum.proxy_name + "." + iconstant.proxy_name + "'.");
|
||||||
|
iconstant.deprecation_message = "This enum member is deprecated.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ienum.constants.push_back(iconstant);
|
ienum.constants.push_back(iconstant);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3470,6 +4055,16 @@ bool BindingsGenerator::_populate_object_type_interfaces() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (iconstant.const_doc) {
|
||||||
|
iconstant.is_deprecated = iconstant.const_doc->is_deprecated;
|
||||||
|
iconstant.deprecation_message = iconstant.const_doc->deprecated_message;
|
||||||
|
|
||||||
|
if (iconstant.deprecation_message.is_empty()) {
|
||||||
|
WARN_PRINT("An empty deprecation message is discouraged. Constant: '" + itype.proxy_name + "." + iconstant.proxy_name + "'.");
|
||||||
|
iconstant.deprecation_message = "This constant is deprecated.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
itype.constants.push_back(iconstant);
|
itype.constants.push_back(iconstant);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -47,7 +47,10 @@ class BindingsGenerator {
|
||||||
String name;
|
String name;
|
||||||
String proxy_name;
|
String proxy_name;
|
||||||
int64_t value = 0;
|
int64_t value = 0;
|
||||||
const DocData::ConstantDoc *const_doc;
|
const DocData::ConstantDoc *const_doc = nullptr;
|
||||||
|
|
||||||
|
bool is_deprecated = false;
|
||||||
|
String deprecation_message;
|
||||||
|
|
||||||
ConstantInterface() {}
|
ConstantInterface() {}
|
||||||
|
|
||||||
|
@ -86,6 +89,9 @@ class BindingsGenerator {
|
||||||
StringName getter;
|
StringName getter;
|
||||||
|
|
||||||
const DocData::PropertyDoc *prop_doc;
|
const DocData::PropertyDoc *prop_doc;
|
||||||
|
|
||||||
|
bool is_deprecated = false;
|
||||||
|
String deprecation_message;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct TypeReference {
|
struct TypeReference {
|
||||||
|
@ -427,6 +433,9 @@ class BindingsGenerator {
|
||||||
|
|
||||||
const DocData::ClassDoc *class_doc = nullptr;
|
const DocData::ClassDoc *class_doc = nullptr;
|
||||||
|
|
||||||
|
bool is_deprecated = false;
|
||||||
|
String deprecation_message;
|
||||||
|
|
||||||
List<ConstantInterface> constants;
|
List<ConstantInterface> constants;
|
||||||
List<EnumInterface> enums;
|
List<EnumInterface> enums;
|
||||||
List<PropertyInterface> properties;
|
List<PropertyInterface> properties;
|
||||||
|
@ -765,8 +774,18 @@ class BindingsGenerator {
|
||||||
return p_type->name;
|
return p_type->name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String bbcode_to_text(const String &p_bbcode, const TypeInterface *p_itype);
|
||||||
String bbcode_to_xml(const String &p_bbcode, const TypeInterface *p_itype, bool p_is_signal = false);
|
String bbcode_to_xml(const String &p_bbcode, const TypeInterface *p_itype, bool p_is_signal = false);
|
||||||
|
|
||||||
|
void _append_text_method(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
|
void _append_text_member(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
|
void _append_text_signal(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
|
void _append_text_enum(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
|
void _append_text_constant(StringBuilder &p_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
|
void _append_text_constant_in_global_scope(StringBuilder &p_output, const String &p_target_cname, const String &p_link_target);
|
||||||
|
void _append_text_param(StringBuilder &p_output, const String &p_link_target);
|
||||||
|
void _append_text_undeclared(StringBuilder &p_output, const String &p_link_target);
|
||||||
|
|
||||||
void _append_xml_method(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
void _append_xml_method(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
void _append_xml_member(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
void _append_xml_member(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
void _append_xml_signal(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
void _append_xml_signal(StringBuilder &p_xml_output, const TypeInterface *p_target_itype, const StringName &p_target_cname, const String &p_link_target, const Vector<String> &p_link_target_parts);
|
||||||
|
|
Loading…
Reference in a new issue