utils.notest.gd 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. class_name Utils
  2. # `assert()` is not evaluated in non-debug builds. Do not use `assert()`
  3. # for anything other than testing the `assert()` itself.
  4. static func check(condition: Variant) -> void:
  5. if not condition:
  6. printerr("Check failed.")
  7. static func get_type(property: Dictionary, is_return: bool = false) -> String:
  8. match property.type:
  9. TYPE_NIL:
  10. if property.usage & PROPERTY_USAGE_NIL_IS_VARIANT:
  11. return "Variant"
  12. return "void" if is_return else "null"
  13. TYPE_INT:
  14. if property.usage & PROPERTY_USAGE_CLASS_IS_ENUM:
  15. if property.class_name == &"":
  16. return "<unknown enum>"
  17. return property.class_name
  18. TYPE_ARRAY:
  19. if property.hint == PROPERTY_HINT_ARRAY_TYPE:
  20. if str(property.hint_string).is_empty():
  21. return "Array[<unknown type>]"
  22. return "Array[%s]" % property.hint_string
  23. TYPE_DICTIONARY:
  24. if property.hint == PROPERTY_HINT_DICTIONARY_TYPE:
  25. if str(property.hint_string).is_empty():
  26. return "Dictionary[<unknown type>, <unknown type>]"
  27. return "Dictionary[%s]" % str(property.hint_string).replace(";", ", ")
  28. TYPE_OBJECT:
  29. if not str(property.class_name).is_empty():
  30. return property.class_name
  31. return type_string(property.type)
  32. static func get_property_signature(property: Dictionary, base: Object = null, is_static: bool = false) -> String:
  33. if property.usage & PROPERTY_USAGE_CATEGORY:
  34. return '@export_category("%s")' % str(property.name).c_escape()
  35. if property.usage & PROPERTY_USAGE_GROUP:
  36. return '@export_group("%s")' % str(property.name).c_escape()
  37. if property.usage & PROPERTY_USAGE_SUBGROUP:
  38. return '@export_subgroup("%s")' % str(property.name).c_escape()
  39. var result: String = ""
  40. if not (property.usage & PROPERTY_USAGE_SCRIPT_VARIABLE):
  41. printerr("Missing `PROPERTY_USAGE_SCRIPT_VARIABLE` flag.")
  42. if is_static:
  43. result += "static "
  44. result += "var " + property.name + ": " + get_type(property)
  45. if is_instance_valid(base):
  46. result += " = " + var_to_str(base.get(property.name))
  47. return result
  48. static func get_human_readable_hint_string(property: Dictionary) -> String:
  49. if property.type >= TYPE_ARRAY and property.hint == PROPERTY_HINT_TYPE_STRING:
  50. var type_hint_prefixes: String = ""
  51. var hint_string: String = property.hint_string
  52. while true:
  53. if not hint_string.contains(":"):
  54. printerr("Invalid PROPERTY_HINT_TYPE_STRING format.")
  55. var elem_type_hint: String = hint_string.get_slice(":", 0)
  56. hint_string = hint_string.substr(elem_type_hint.length() + 1)
  57. var elem_type: int
  58. var elem_hint: int
  59. if elem_type_hint.is_valid_int():
  60. elem_type = elem_type_hint.to_int()
  61. type_hint_prefixes += "<%s>:" % type_string(elem_type)
  62. else:
  63. if elem_type_hint.count("/") != 1:
  64. printerr("Invalid PROPERTY_HINT_TYPE_STRING format.")
  65. elem_type = elem_type_hint.get_slice("/", 0).to_int()
  66. elem_hint = elem_type_hint.get_slice("/", 1).to_int()
  67. type_hint_prefixes += "<%s>/<%s>:" % [
  68. type_string(elem_type),
  69. get_property_hint_name(elem_hint).trim_prefix("PROPERTY_HINT_"),
  70. ]
  71. if elem_type < TYPE_ARRAY or hint_string.is_empty():
  72. break
  73. return type_hint_prefixes + hint_string
  74. return property.hint_string
  75. static func print_property_extended_info(property: Dictionary, base: Object = null, is_static: bool = false) -> void:
  76. print(get_property_signature(property, base, is_static))
  77. print(' hint=%s hint_string="%s" usage=%s class_name=&"%s"' % [
  78. get_property_hint_name(property.hint).trim_prefix("PROPERTY_HINT_"),
  79. get_human_readable_hint_string(property).c_escape(),
  80. get_property_usage_string(property.usage).replace("PROPERTY_USAGE_", ""),
  81. property.class_name.c_escape(),
  82. ])
  83. static func get_method_signature(method: Dictionary, is_signal: bool = false) -> String:
  84. var result: String = ""
  85. if method.flags & METHOD_FLAG_VIRTUAL_REQUIRED:
  86. result += "abstract "
  87. if method.flags & METHOD_FLAG_STATIC:
  88. result += "static "
  89. result += ("signal " if is_signal else "func ") + method.name + "("
  90. var args: Array[Dictionary] = method.args
  91. var default_args: Array = method.default_args
  92. var mandatory_argc: int = args.size() - default_args.size()
  93. for i in args.size():
  94. if i > 0:
  95. result += ", "
  96. var arg: Dictionary = args[i]
  97. result += arg.name + ": " + get_type(arg)
  98. if i >= mandatory_argc:
  99. result += " = " + var_to_str(default_args[i - mandatory_argc])
  100. if method.flags & METHOD_FLAG_VARARG:
  101. # `MethodInfo` does not support the rest parameter name.
  102. result += "...args" if args.is_empty() else ", ...args"
  103. result += ")"
  104. if is_signal:
  105. if get_type(method.return, true) != "void":
  106. printerr("Signal return type must be `void`.")
  107. else:
  108. result += " -> " + get_type(method.return, true)
  109. return result
  110. static func get_property_hint_name(hint: PropertyHint) -> String:
  111. match hint:
  112. PROPERTY_HINT_NONE:
  113. return "PROPERTY_HINT_NONE"
  114. PROPERTY_HINT_RANGE:
  115. return "PROPERTY_HINT_RANGE"
  116. PROPERTY_HINT_ENUM:
  117. return "PROPERTY_HINT_ENUM"
  118. PROPERTY_HINT_ENUM_SUGGESTION:
  119. return "PROPERTY_HINT_ENUM_SUGGESTION"
  120. PROPERTY_HINT_EXP_EASING:
  121. return "PROPERTY_HINT_EXP_EASING"
  122. PROPERTY_HINT_LINK:
  123. return "PROPERTY_HINT_LINK"
  124. PROPERTY_HINT_FLAGS:
  125. return "PROPERTY_HINT_FLAGS"
  126. PROPERTY_HINT_LAYERS_2D_RENDER:
  127. return "PROPERTY_HINT_LAYERS_2D_RENDER"
  128. PROPERTY_HINT_LAYERS_2D_PHYSICS:
  129. return "PROPERTY_HINT_LAYERS_2D_PHYSICS"
  130. PROPERTY_HINT_LAYERS_2D_NAVIGATION:
  131. return "PROPERTY_HINT_LAYERS_2D_NAVIGATION"
  132. PROPERTY_HINT_LAYERS_3D_RENDER:
  133. return "PROPERTY_HINT_LAYERS_3D_RENDER"
  134. PROPERTY_HINT_LAYERS_3D_PHYSICS:
  135. return "PROPERTY_HINT_LAYERS_3D_PHYSICS"
  136. PROPERTY_HINT_LAYERS_3D_NAVIGATION:
  137. return "PROPERTY_HINT_LAYERS_3D_NAVIGATION"
  138. PROPERTY_HINT_LAYERS_AVOIDANCE:
  139. return "PROPERTY_HINT_LAYERS_AVOIDANCE"
  140. PROPERTY_HINT_FILE:
  141. return "PROPERTY_HINT_FILE"
  142. PROPERTY_HINT_DIR:
  143. return "PROPERTY_HINT_DIR"
  144. PROPERTY_HINT_GLOBAL_FILE:
  145. return "PROPERTY_HINT_GLOBAL_FILE"
  146. PROPERTY_HINT_GLOBAL_DIR:
  147. return "PROPERTY_HINT_GLOBAL_DIR"
  148. PROPERTY_HINT_RESOURCE_TYPE:
  149. return "PROPERTY_HINT_RESOURCE_TYPE"
  150. PROPERTY_HINT_MULTILINE_TEXT:
  151. return "PROPERTY_HINT_MULTILINE_TEXT"
  152. PROPERTY_HINT_EXPRESSION:
  153. return "PROPERTY_HINT_EXPRESSION"
  154. PROPERTY_HINT_PLACEHOLDER_TEXT:
  155. return "PROPERTY_HINT_PLACEHOLDER_TEXT"
  156. PROPERTY_HINT_COLOR_NO_ALPHA:
  157. return "PROPERTY_HINT_COLOR_NO_ALPHA"
  158. PROPERTY_HINT_OBJECT_ID:
  159. return "PROPERTY_HINT_OBJECT_ID"
  160. PROPERTY_HINT_TYPE_STRING:
  161. return "PROPERTY_HINT_TYPE_STRING"
  162. PROPERTY_HINT_NODE_PATH_TO_EDITED_NODE:
  163. return "PROPERTY_HINT_NODE_PATH_TO_EDITED_NODE"
  164. PROPERTY_HINT_OBJECT_TOO_BIG:
  165. return "PROPERTY_HINT_OBJECT_TOO_BIG"
  166. PROPERTY_HINT_NODE_PATH_VALID_TYPES:
  167. return "PROPERTY_HINT_NODE_PATH_VALID_TYPES"
  168. PROPERTY_HINT_SAVE_FILE:
  169. return "PROPERTY_HINT_SAVE_FILE"
  170. PROPERTY_HINT_GLOBAL_SAVE_FILE:
  171. return "PROPERTY_HINT_GLOBAL_SAVE_FILE"
  172. PROPERTY_HINT_INT_IS_OBJECTID:
  173. return "PROPERTY_HINT_INT_IS_OBJECTID"
  174. PROPERTY_HINT_INT_IS_POINTER:
  175. return "PROPERTY_HINT_INT_IS_POINTER"
  176. PROPERTY_HINT_ARRAY_TYPE:
  177. return "PROPERTY_HINT_ARRAY_TYPE"
  178. PROPERTY_HINT_DICTIONARY_TYPE:
  179. return "PROPERTY_HINT_DICTIONARY_TYPE"
  180. PROPERTY_HINT_LOCALE_ID:
  181. return "PROPERTY_HINT_LOCALE_ID"
  182. PROPERTY_HINT_LOCALIZABLE_STRING:
  183. return "PROPERTY_HINT_LOCALIZABLE_STRING"
  184. PROPERTY_HINT_NODE_TYPE:
  185. return "PROPERTY_HINT_NODE_TYPE"
  186. PROPERTY_HINT_HIDE_QUATERNION_EDIT:
  187. return "PROPERTY_HINT_HIDE_QUATERNION_EDIT"
  188. PROPERTY_HINT_PASSWORD:
  189. return "PROPERTY_HINT_PASSWORD"
  190. PROPERTY_HINT_TOOL_BUTTON:
  191. return "PROPERTY_HINT_TOOL_BUTTON"
  192. PROPERTY_HINT_INPUT_NAME:
  193. return "PROPERTY_HINT_INPUT_NAME"
  194. printerr("Argument `hint` is invalid. Use `PROPERTY_HINT_*` constants.")
  195. return "<invalid hint>"
  196. static func get_property_usage_string(usage: int) -> String:
  197. if usage == PROPERTY_USAGE_NONE:
  198. return "PROPERTY_USAGE_NONE"
  199. const FLAGS: Array[Array] = [
  200. [PROPERTY_USAGE_STORAGE, "PROPERTY_USAGE_STORAGE"],
  201. [PROPERTY_USAGE_EDITOR, "PROPERTY_USAGE_EDITOR"],
  202. [PROPERTY_USAGE_INTERNAL, "PROPERTY_USAGE_INTERNAL"],
  203. [PROPERTY_USAGE_CHECKABLE, "PROPERTY_USAGE_CHECKABLE"],
  204. [PROPERTY_USAGE_CHECKED, "PROPERTY_USAGE_CHECKED"],
  205. [PROPERTY_USAGE_GROUP, "PROPERTY_USAGE_GROUP"],
  206. [PROPERTY_USAGE_CATEGORY, "PROPERTY_USAGE_CATEGORY"],
  207. [PROPERTY_USAGE_SUBGROUP, "PROPERTY_USAGE_SUBGROUP"],
  208. [PROPERTY_USAGE_CLASS_IS_BITFIELD, "PROPERTY_USAGE_CLASS_IS_BITFIELD"],
  209. [PROPERTY_USAGE_NO_INSTANCE_STATE, "PROPERTY_USAGE_NO_INSTANCE_STATE"],
  210. [PROPERTY_USAGE_RESTART_IF_CHANGED, "PROPERTY_USAGE_RESTART_IF_CHANGED"],
  211. [PROPERTY_USAGE_SCRIPT_VARIABLE, "PROPERTY_USAGE_SCRIPT_VARIABLE"],
  212. [PROPERTY_USAGE_STORE_IF_NULL, "PROPERTY_USAGE_STORE_IF_NULL"],
  213. [PROPERTY_USAGE_UPDATE_ALL_IF_MODIFIED, "PROPERTY_USAGE_UPDATE_ALL_IF_MODIFIED"],
  214. [PROPERTY_USAGE_SCRIPT_DEFAULT_VALUE, "PROPERTY_USAGE_SCRIPT_DEFAULT_VALUE"],
  215. [PROPERTY_USAGE_CLASS_IS_ENUM, "PROPERTY_USAGE_CLASS_IS_ENUM"],
  216. [PROPERTY_USAGE_NIL_IS_VARIANT, "PROPERTY_USAGE_NIL_IS_VARIANT"],
  217. [PROPERTY_USAGE_ARRAY, "PROPERTY_USAGE_ARRAY"],
  218. [PROPERTY_USAGE_ALWAYS_DUPLICATE, "PROPERTY_USAGE_ALWAYS_DUPLICATE"],
  219. [PROPERTY_USAGE_NEVER_DUPLICATE, "PROPERTY_USAGE_NEVER_DUPLICATE"],
  220. [PROPERTY_USAGE_HIGH_END_GFX, "PROPERTY_USAGE_HIGH_END_GFX"],
  221. [PROPERTY_USAGE_NODE_PATH_FROM_SCENE_ROOT, "PROPERTY_USAGE_NODE_PATH_FROM_SCENE_ROOT"],
  222. [PROPERTY_USAGE_RESOURCE_NOT_PERSISTENT, "PROPERTY_USAGE_RESOURCE_NOT_PERSISTENT"],
  223. [PROPERTY_USAGE_KEYING_INCREMENTS, "PROPERTY_USAGE_KEYING_INCREMENTS"],
  224. [PROPERTY_USAGE_DEFERRED_SET_RESOURCE, "PROPERTY_USAGE_DEFERRED_SET_RESOURCE"],
  225. [PROPERTY_USAGE_EDITOR_INSTANTIATE_OBJECT, "PROPERTY_USAGE_EDITOR_INSTANTIATE_OBJECT"],
  226. [PROPERTY_USAGE_EDITOR_BASIC_SETTING, "PROPERTY_USAGE_EDITOR_BASIC_SETTING"],
  227. [PROPERTY_USAGE_READ_ONLY, "PROPERTY_USAGE_READ_ONLY"],
  228. [PROPERTY_USAGE_SECRET, "PROPERTY_USAGE_SECRET"],
  229. ]
  230. var result: String = ""
  231. if (usage & PROPERTY_USAGE_DEFAULT) == PROPERTY_USAGE_DEFAULT:
  232. result += "PROPERTY_USAGE_DEFAULT|"
  233. usage &= ~PROPERTY_USAGE_DEFAULT
  234. for flag in FLAGS:
  235. if usage & flag[0]:
  236. result += flag[1] + "|"
  237. usage &= ~flag[0]
  238. if usage != PROPERTY_USAGE_NONE:
  239. printerr("Argument `usage` is invalid. Use `PROPERTY_USAGE_*` constants.")
  240. return "<invalid usage flags>"
  241. return result.left(-1)