ASEnumBinder.cpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. // Copyright (c) 2008-2023 the Urho3D project
  2. // License: MIT
  3. #include "ASResult.h"
  4. #include "ASUtils.h"
  5. #include "Tuning.h"
  6. #include "Utils.h"
  7. #include "XmlAnalyzer.h"
  8. #include "XmlSourceData.h"
  9. #include <cassert>
  10. using namespace std;
  11. namespace ASBindingGenerator
  12. {
  13. static void ProcessEnum(const EnumAnalyzer& analyzer)
  14. {
  15. if (analyzer.IsInternal())
  16. return;
  17. string header = analyzer.GetHeaderFile();
  18. Result::AddHeader(header);
  19. if (IsIgnoredHeader(header))
  20. return;
  21. ProcessedEnum processedEnum;
  22. processedEnum.name_ = analyzer.GetTypeName();
  23. processedEnum.insideDefine_ = InsideDefine(header);
  24. processedEnum.comment_ = analyzer.GetLocation();
  25. string comment = analyzer.GetComment();
  26. if (Contains(comment, "NO_BIND"))
  27. {
  28. processedEnum.registration_.push_back("// Not registered because have @nobind mark");
  29. Result::enums_.push_back(processedEnum);
  30. return;
  31. }
  32. if (Contains(comment, "MANUAL_BIND"))
  33. {
  34. processedEnum.registration_.push_back("// Not registered because have @manualbind mark");
  35. Result::enums_.push_back(processedEnum);
  36. return;
  37. }
  38. string enumTypeName = analyzer.GetTypeName();
  39. string cppEnumBaseType = analyzer.GetBaseType();
  40. string asEnumBaseType = CppPrimitiveTypeToAS(cppEnumBaseType);
  41. if (analyzer.IsClass()) // Scoped enumerations
  42. {
  43. processedEnum.registration_.push_back("engine->RegisterTypedef(\"" + enumTypeName + "\", \"" + asEnumBaseType + "\");");
  44. processedEnum.registration_.push_back("engine->SetDefaultNamespace(\"" + enumTypeName + "\");");
  45. for (const string& value : analyzer.GetEnumerators())
  46. {
  47. string constName = enumTypeName + "_" + value;
  48. processedEnum.glue_.push_back("static const " + cppEnumBaseType + " " + constName + " = static_cast<" + cppEnumBaseType + ">(" + enumTypeName + "::" + value + ");");
  49. processedEnum.registration_.push_back("engine->RegisterGlobalProperty(\"const " + asEnumBaseType + " " + value + "\", (void*)&" + constName + ");");
  50. }
  51. processedEnum.registration_.push_back("engine->SetDefaultNamespace(\"\");");
  52. }
  53. else // Unscoped enumerations
  54. {
  55. if (cppEnumBaseType == "int") // Enums in AngelScript can be only int
  56. {
  57. processedEnum.registration_.push_back("engine->RegisterEnum(\"" + enumTypeName + "\");");
  58. for (const string& value : analyzer.GetEnumerators())
  59. processedEnum.registration_.push_back("engine->RegisterEnumValue(\"" + enumTypeName + "\", \"" + value + "\", " + value + ");");
  60. }
  61. else // If enum is not int then register as typedef. But this type can not be used in switch
  62. {
  63. processedEnum.registration_.push_back("engine->RegisterTypedef(\"" + enumTypeName + "\", \"" + asEnumBaseType + "\");");
  64. for (const string& enumerator : analyzer.GetEnumerators())
  65. {
  66. string constName = enumTypeName + "_" + enumerator;
  67. processedEnum.glue_.push_back("static const " + cppEnumBaseType + " " + constName + " = " + enumerator + ";");
  68. processedEnum.registration_.push_back("engine->RegisterGlobalProperty(\"const " + asEnumBaseType + " " + enumerator + "\", (void*)&" + constName + ");");
  69. }
  70. }
  71. }
  72. Result::enums_.push_back(processedEnum);
  73. }
  74. static void ProcessFlagset(const GlobalFunctionAnalyzer& analyzer)
  75. {
  76. string header = analyzer.GetHeaderFile();
  77. Result::AddHeader(header);
  78. if (IsIgnoredHeader(header))
  79. return;
  80. vector<ParamAnalyzer> params = analyzer.GetParams();
  81. assert(params.size() == 2);
  82. string enumTypeName = params[0].GetType().GetName();
  83. string flagsetName = params[1].GetType().GetName();
  84. shared_ptr<EnumAnalyzer> enumAnalyzer = FindEnum(enumTypeName);
  85. assert(enumAnalyzer);
  86. string cppEnumBaseType = enumAnalyzer->GetBaseType();
  87. string asEnumBaseType = CppPrimitiveTypeToAS(cppEnumBaseType);
  88. ProcessedEnum processedEnum;
  89. processedEnum.name_ = flagsetName;
  90. processedEnum.insideDefine_ = InsideDefine(header);
  91. processedEnum.comment_ = analyzer.GetLocation();
  92. processedEnum.registration_.push_back("engine->RegisterTypedef(\"" + flagsetName + "\", \"" + asEnumBaseType + "\");");
  93. Result::enums_.push_back(processedEnum);
  94. }
  95. void ProcessAllEnums()
  96. {
  97. NamespaceAnalyzer namespaceAnalyzer(SourceData::namespaceUrho3D_);
  98. vector<EnumAnalyzer> enumAnalyzers = namespaceAnalyzer.GetEnums();
  99. for (const EnumAnalyzer& enumAnalyzer : enumAnalyzers)
  100. ProcessEnum(enumAnalyzer);
  101. vector<GlobalFunctionAnalyzer> globalFunctionAnalyzers = namespaceAnalyzer.GetFunctions();
  102. for (const GlobalFunctionAnalyzer& globalFunctionAnalyzer : globalFunctionAnalyzers)
  103. {
  104. string functionName = globalFunctionAnalyzer.GetName();
  105. if (functionName == "URHO3D_FLAGSET")
  106. ProcessFlagset(globalFunctionAnalyzer);
  107. }
  108. }
  109. } // namespace ASBindingGenerator