ASClassBinder.cpp 44 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270
  1. //
  2. // Copyright (c) 2008-2020 the Urho3D project.
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to deal
  6. // in the Software without restriction, including without limitation the rights
  7. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. // copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  19. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  20. // THE SOFTWARE.
  21. //
  22. #include "ASResult.h"
  23. #include "ASUtils.h"
  24. #include "Tuning.h"
  25. #include "Utils.h"
  26. #include "XmlAnalyzer.h"
  27. #include "XmlSourceData.h"
  28. #include <cassert>
  29. #include <iostream>
  30. #include <regex>
  31. #include <fstream>
  32. #include <vector>
  33. extern string _sourceDir;
  34. namespace ASBindingGenerator
  35. {
  36. static shared_ptr<ASGeneratedFile_Members> _result_Members_A;
  37. static shared_ptr<ASGeneratedFile_Members> _result_Members_B;
  38. static shared_ptr<ASGeneratedFile_Members> _result_Members_Constraint;
  39. static shared_ptr<ASGeneratedFile_Members> _result_Members_Ca_Cm;
  40. static shared_ptr<ASGeneratedFile_Members> _result_Members_Cn_Cz;
  41. static shared_ptr<ASGeneratedFile_Members> _result_Members_D;
  42. static shared_ptr<ASGeneratedFile_Members> _result_Members_E;
  43. static shared_ptr<ASGeneratedFile_Members> _result_Members_F;
  44. static shared_ptr<ASGeneratedFile_Members> _result_Members_G;
  45. static shared_ptr<ASGeneratedFile_Members> _result_Members_H;
  46. static shared_ptr<ASGeneratedFile_Members> _result_Members_I;
  47. static shared_ptr<ASGeneratedFile_Members> _result_Members_J;
  48. static shared_ptr<ASGeneratedFile_Members> _result_Members_K;
  49. static shared_ptr<ASGeneratedFile_Members> _result_Members_L;
  50. static shared_ptr<ASGeneratedFile_Members> _result_Members_M;
  51. static shared_ptr<ASGeneratedFile_Members> _result_Members_N;
  52. static shared_ptr<ASGeneratedFile_Members> _result_Members_O;
  53. static shared_ptr<ASGeneratedFile_Members> _result_Members_P;
  54. static shared_ptr<ASGeneratedFile_Members> _result_Members_Q;
  55. static shared_ptr<ASGeneratedFile_Members> _result_Members_R;
  56. static shared_ptr<ASGeneratedFile_Members> _result_Members_Sa_Sm;
  57. static shared_ptr<ASGeneratedFile_Members> _result_Members_Sn_Sz;
  58. static shared_ptr<ASGeneratedFile_Members> _result_Members_Ta_Tm;
  59. static shared_ptr<ASGeneratedFile_Members> _result_Members_Tn_Tz;
  60. static shared_ptr<ASGeneratedFile_Members> _result_Members_U;
  61. static shared_ptr<ASGeneratedFile_Members> _result_Members_V;
  62. static shared_ptr<ASGeneratedFile_Members> _result_Members_W;
  63. static shared_ptr<ASGeneratedFile_Members> _result_Members_X;
  64. static shared_ptr<ASGeneratedFile_Members> _result_Members_Y;
  65. static shared_ptr<ASGeneratedFile_Members> _result_Members_Z;
  66. static shared_ptr<ASGeneratedFile_Members> _result_Members_Other;
  67. static shared_ptr<ASGeneratedFile_Templates> _result_Templates;
  68. static shared_ptr<ASGeneratedFile_Members> GetGeneratedFile(const string& className)
  69. {
  70. if (StartsWith(className, "A"))
  71. return _result_Members_A;
  72. if (StartsWith(className, "B"))
  73. return _result_Members_B;
  74. if (StartsWith(className, "Constraint"))
  75. return _result_Members_Constraint;
  76. if (StartsWith(className, "C"))
  77. {
  78. if (className.length() < 2)
  79. return _result_Members_Ca_Cm;
  80. char secondChar = className[1];
  81. if (secondChar >= 'n')
  82. return _result_Members_Cn_Cz;
  83. return _result_Members_Ca_Cm;
  84. }
  85. if (StartsWith(className, "D"))
  86. return _result_Members_D;
  87. if (StartsWith(className, "E"))
  88. return _result_Members_E;
  89. if (StartsWith(className, "F"))
  90. return _result_Members_F;
  91. if (StartsWith(className, "G"))
  92. return _result_Members_G;
  93. if (StartsWith(className, "H"))
  94. return _result_Members_H;
  95. if (StartsWith(className, "I"))
  96. return _result_Members_I;
  97. if (StartsWith(className, "J"))
  98. return _result_Members_J;
  99. if (StartsWith(className, "K"))
  100. return _result_Members_K;
  101. if (StartsWith(className, "L"))
  102. return _result_Members_L;
  103. if (StartsWith(className, "M"))
  104. return _result_Members_M;
  105. if (StartsWith(className, "N"))
  106. return _result_Members_N;
  107. if (StartsWith(className, "O"))
  108. return _result_Members_O;
  109. if (StartsWith(className, "P"))
  110. return _result_Members_P;
  111. if (StartsWith(className, "Q"))
  112. return _result_Members_Q;
  113. if (StartsWith(className, "R"))
  114. return _result_Members_R;
  115. if (StartsWith(className, "S"))
  116. {
  117. if (className.length() < 2)
  118. return _result_Members_Sa_Sm;
  119. char secondChar = className[1];
  120. if (secondChar >= 'n')
  121. return _result_Members_Sn_Sz;
  122. return _result_Members_Sa_Sm;
  123. }
  124. if (StartsWith(className, "T"))
  125. {
  126. if (className.length() < 2)
  127. return _result_Members_Ta_Tm;
  128. char secondChar = className[1];
  129. if (secondChar >= 'n')
  130. return _result_Members_Tn_Tz;
  131. return _result_Members_Ta_Tm;
  132. }
  133. if (StartsWith(className, "U"))
  134. return _result_Members_U;
  135. if (StartsWith(className, "V"))
  136. return _result_Members_V;
  137. if (StartsWith(className, "W"))
  138. return _result_Members_W;
  139. if (StartsWith(className, "X"))
  140. return _result_Members_X;
  141. if (StartsWith(className, "Y"))
  142. return _result_Members_Y;
  143. if (StartsWith(className, "Z"))
  144. return _result_Members_Z;
  145. if (StartsWith(className, "W"))
  146. return _result_Members_W;
  147. return _result_Members_Other;
  148. }
  149. //static void RegisterStaticFunction(const ClassStaticFunctionAnalyzer& functionAnalyzer, shared_ptr<ASGeneratedFile_Members> result)
  150. /*
  151. static void RegisterStaticFunction(const ClassStaticFunctionAnalyzer& functionAnalyzer, ASGeneratedFile_Base* result)
  152. {
  153. result->reg_ << " // " << functionAnalyzer.GetLocation() << "\n";
  154. vector<ParamAnalyzer> params = functionAnalyzer.GetParams();
  155. vector<ConvertedVariable> convertedParams;
  156. string outGlue;
  157. bool needWrapper = false;
  158. for (const ParamAnalyzer& param : params)
  159. {
  160. ConvertedVariable conv;
  161. try
  162. {
  163. conv = CppVariableToAS(param.GetType(), VariableUsage::FunctionParameter, param.GetDeclname(), param.GetDefval());
  164. }
  165. catch (const Exception& e)
  166. {
  167. result->reg_ << " // " << e.what() << "\n";
  168. return;
  169. }
  170. if (conv.NeedWrapper())
  171. needWrapper = true;
  172. convertedParams.push_back(conv);
  173. }
  174. ConvertedVariable convertedReturn;
  175. try
  176. {
  177. convertedReturn = CppVariableToAS(functionAnalyzer.GetReturnType(), VariableUsage::FunctionReturn);
  178. }
  179. catch (const Exception& e)
  180. {
  181. result->reg_ << " // " << e.what() << "\n";
  182. return;
  183. }
  184. if (convertedReturn.NeedWrapper())
  185. needWrapper = true;
  186. string asFunctionName = functionAnalyzer.GetName();
  187. string className = functionAnalyzer.GetClassName();
  188. if (needWrapper)
  189. result->glue_ << GenerateWrapper(functionAnalyzer, convertedParams, convertedReturn);
  190. string decl = convertedReturn.asDeclaration_ + " " + asFunctionName + "(" + JoinASDeclarations(convertedParams) + ")";
  191. result->reg_ << " engine->SetDefaultNamespace(\"" << className << "\");\n";
  192. result->reg_ << " engine->RegisterGlobalFunction(\"" << decl << "\", ";
  193. if (needWrapper)
  194. result->reg_ << "AS_FUNCTION(" << GenerateWrapperName(functionAnalyzer) << "), AS_CALL_CDECL);\n";
  195. else
  196. result->reg_ << Generate_asFUNCTIONPR(functionAnalyzer) << ", AS_CALL_CDECL);\n";
  197. result->reg_ << " engine->SetDefaultNamespace(\"\");\n";
  198. }
  199. */
  200. static void RegisterRefCountedConstructor(const MethodAnalyzer& methodAnalyzer, bool templateVersion)
  201. {
  202. string className = methodAnalyzer.GetClassName();
  203. string wrapperName = GenerateWrapperName(methodAnalyzer, templateVersion);
  204. string header = methodAnalyzer.GetClass().GetHeaderFile();
  205. string insideDefine = InsideDefine(header);
  206. string args = ExtractCleanedFunctionArgsstring(methodAnalyzer.GetMemberdef());
  207. bool isDefaultConstructor = args.empty();
  208. if (isDefaultConstructor)
  209. return;
  210. //shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(functionAnalyzer.GetClass().GetClassName());
  211. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(methodAnalyzer.GetClass().GetClassName()).get();
  212. bool firstContext = StartsWith(args, "Context *context");
  213. if (firstContext)
  214. {
  215. args = CutStart(args, "Context *context");
  216. args = CutStart(args, ", ");
  217. }
  218. else if (Contains(args, "Context"))
  219. {
  220. result->reg_ <<
  221. " // " << methodAnalyzer.GetLocation() << "\n"
  222. " // Error: context can be only first\n";
  223. return;
  224. }
  225. string cppParamsNames = methodAnalyzer.JoinParamsNames(firstContext);
  226. if (firstContext)
  227. {
  228. if (cppParamsNames.length() > 0)
  229. cppParamsNames = ", " + cppParamsNames;
  230. cppParamsNames = "GetScriptContext()" + cppParamsNames;
  231. }
  232. string decl = "";
  233. vector<ParamAnalyzer> params = methodAnalyzer.GetParams();
  234. for (unsigned i = 0; i < params.size(); i++)
  235. {
  236. ParamAnalyzer param = params[i];
  237. if (i == 0 && firstContext)
  238. {
  239. assert(param.GetType().ToString() == "Context*");
  240. continue;
  241. }
  242. if (decl.length() > 0)
  243. decl += ", ";
  244. try
  245. {
  246. decl += CppTypeToAS(param.GetType(), TypeUsage::FunctionParameter);
  247. }
  248. catch (const Exception& e)
  249. {
  250. result->reg_ <<
  251. " // " << methodAnalyzer.GetLocation() << "\n"
  252. " // " << e.what() << "\n";
  253. return;
  254. }
  255. string defval = param.GetDefval();
  256. if (!defval.empty())
  257. {
  258. defval = CppValueToAS(defval);
  259. defval = ReplaceAll(defval, "\"", "\\\"");
  260. decl += /*" " + param.GetDeclname() +*/ " = " + defval;
  261. }
  262. }
  263. decl = className + "@+ f(" + decl + ")";
  264. if (templateVersion)
  265. {
  266. result->reg_ <<
  267. " // " << methodAnalyzer.GetLocation() << "\n"
  268. " {\n"
  269. " String declFactory(String(className) + \"@ f()\");\n"
  270. " engine->RegisterObjectBehaviour(className, "
  271. "asBEHAVE_FACTORY, "
  272. "declFactory.CString(), "
  273. "AS_FUNCTION(" << wrapperName << "), "
  274. "AS_CALL_CDECL);\n"
  275. " }\n";
  276. }
  277. else
  278. {
  279. result->reg_ <<
  280. " // " << methodAnalyzer.GetLocation() << "\n"
  281. " engine->RegisterObjectBehaviour("
  282. "\"" << className << "\", "
  283. "asBEHAVE_FACTORY, "
  284. "\"" << decl << "\", "
  285. "AS_FUNCTION(" << wrapperName << "), "
  286. "AS_CALL_CDECL);\n";
  287. }
  288. if (!insideDefine.empty())
  289. result->glue_ << "#ifdef " << insideDefine << "\n";
  290. result->glue_ <<
  291. "// " << methodAnalyzer.GetLocation() << "\n"
  292. "static " << className << "* " << wrapperName << "(" << args << ")\n"
  293. "{\n"
  294. " return new " << className << "(" << cppParamsNames << ");\n"
  295. "}\n";
  296. if (!insideDefine.empty())
  297. result->glue_ << "#endif\n";
  298. result->glue_ << "\n";
  299. }
  300. static void RegisterValueConstructor(const MethodAnalyzer& methodAnalyzer, bool templateVersion)
  301. {
  302. string header = methodAnalyzer.GetClass().GetHeaderFile();
  303. string insideDefine = InsideDefine(header);
  304. string className = methodAnalyzer.GetClassName();
  305. string wrapperName = GenerateWrapperName(methodAnalyzer, templateVersion);
  306. string args = ExtractCleanedFunctionArgsstring(methodAnalyzer.GetMemberdef());
  307. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  308. bool isDefaultConstructor = args.empty();
  309. if (isDefaultConstructor)
  310. return;
  311. stringstream& reg = result->reg_;
  312. stringstream& glue = result->glue_;
  313. string decl = "";
  314. vector<ParamAnalyzer> params = methodAnalyzer.GetParams();
  315. for (ParamAnalyzer param : params)
  316. {
  317. if (decl.length() > 0)
  318. decl += ", ";
  319. try
  320. {
  321. decl += CppTypeToAS(param.GetType(), TypeUsage::FunctionParameter);
  322. }
  323. catch (const Exception& e)
  324. {
  325. if (isDefaultConstructor && !insideDefine.empty())
  326. reg << "#ifdef " << insideDefine << "\n";
  327. reg <<
  328. " // " << methodAnalyzer.GetLocation() << "\n"
  329. " // " << e.what() << "\n";
  330. if (isDefaultConstructor && !insideDefine.empty())
  331. reg << "#endif\n";
  332. return;
  333. }
  334. string defval = param.GetDefval();
  335. if (!defval.empty())
  336. {
  337. defval = CppValueToAS(defval);
  338. defval = ReplaceAll(defval, "\"", "\\\"");
  339. decl += /*" " + param.GetDeclname() +*/ " = " + defval;
  340. }
  341. }
  342. decl = "void f(" + decl + ")";
  343. if (isDefaultConstructor && !insideDefine.empty())
  344. reg << "#ifdef " << insideDefine << "\n";
  345. reg <<
  346. " // " << methodAnalyzer.GetLocation() << "\n"
  347. " engine->RegisterObjectBehaviour("
  348. "\"" << className << "\", "
  349. "asBEHAVE_CONSTRUCT, "
  350. "\"" << decl << "\", "
  351. "AS_FUNCTION_OBJFIRST(" << wrapperName << "), "
  352. "AS_CALL_CDECL_OBJFIRST);\n";
  353. if (isDefaultConstructor && !insideDefine.empty())
  354. reg << "#endif\n";
  355. if (isDefaultConstructor && !insideDefine.empty())
  356. glue << "#ifdef " << insideDefine << "\n";
  357. glue <<
  358. "// " << methodAnalyzer.GetLocation() << "\n"
  359. "static void " << wrapperName << "(" << className << "* ptr";
  360. if (args.length() > 0)
  361. glue << ", " << args;
  362. glue << ")\n"
  363. "{\n"
  364. " new(ptr) " << className << "(" << methodAnalyzer.JoinParamsNames() << ");\n"
  365. "}\n";
  366. if (isDefaultConstructor && !insideDefine.empty())
  367. glue << "#endif\n";
  368. glue << "\n";
  369. if (isDefaultConstructor)
  370. Result::AddHeader(header);
  371. }
  372. static void TryRegisterImplicitlyDeclaredAssignOperator(const ClassAnalyzer& classAnalyzer)
  373. {
  374. string className = classAnalyzer.GetClassName();
  375. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  376. result->reg_ <<
  377. " // " << className << "& " << className << "::operator=(const " << className << "&) | Possible implicitly-declared\n"
  378. " RegisterImplicitlyDeclaredAssignOperatorIfPossible<" << className << ">(engine, \"" << className << "\");\n";
  379. }
  380. // Some required methods can not be bound automatically when processing class because implicitly-declared
  381. static void RegisterImplicitlyDeclaredMethods(const ClassAnalyzer& classAnalyzer)
  382. {
  383. if (!classAnalyzer.ContainsMethod("operator="))
  384. TryRegisterImplicitlyDeclaredAssignOperator(classAnalyzer);
  385. // No need register impicitly declared copy constructor when registered opAssign operator
  386. }
  387. static void RegisterComparisonOperator(const ClassAnalyzer& classAnalyzer)
  388. {
  389. string className = classAnalyzer.GetClassName();
  390. string wrapperName = className + "_Comparison";
  391. string operatorLessLocation = classAnalyzer.GetMethod("operator<")->GetLocation();
  392. string operatorGreaterLocation = classAnalyzer.GetMethod("operator>")->GetLocation();
  393. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  394. result->glue_ <<
  395. "// " << operatorLessLocation << "\n"
  396. "// " << operatorGreaterLocation << "\n"
  397. "static int " << wrapperName << "(const " << className << "& lhs, const " << className << "& rhs)\n"
  398. "{\n"
  399. " if (lhs < rhs)\n"
  400. " return -1;\n\n"
  401. " if (lhs > rhs)\n"
  402. " return 1;\n\n"
  403. " return 0;\n"
  404. "}\n\n";
  405. result->reg_ <<
  406. " // " << operatorLessLocation << "\n"
  407. " // " << operatorGreaterLocation << "\n"
  408. " engine->RegisterObjectMethod(\"" << className << "\", \"int opCmp(const " << className << "&in) const\", AS_FUNCTION_OBJFIRST(" << wrapperName << "), AS_CALL_CDECL_OBJFIRST);\n";
  409. }
  410. static void RegisterAddReleaseRef(const MethodAnalyzer& methodAnalyzer, bool templateVersion)
  411. {
  412. string className = methodAnalyzer.GetClassName();
  413. //shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  414. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(methodAnalyzer.GetClass().GetClassName()).get();
  415. string functionName = methodAnalyzer.GetName();
  416. string behaviour = (functionName == "AddRef") ? "asBEHAVE_ADDREF" : "asBEHAVE_RELEASE";
  417. string methodpr = Generate_asMETHODPR(methodAnalyzer, templateVersion);
  418. result->reg_ << " // " << methodAnalyzer.GetLocation() << "\n";
  419. if (templateVersion)
  420. result->reg_ << " engine->RegisterObjectBehaviour(className, " << behaviour << ", \"void f()\", " << methodpr << ", AS_CALL_THISCALL);\n";
  421. else
  422. result->reg_ << " engine->RegisterObjectBehaviour(\"" << className << "\", " << behaviour << ", \"void f()\", " << methodpr << ", AS_CALL_THISCALL);\n";
  423. }
  424. static void RegisterFakeAddReleaseRef(const ClassAnalyzer& classAnalyzer)
  425. {
  426. string className = classAnalyzer.GetClassName();
  427. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(className);
  428. result->reg_ <<
  429. " engine->RegisterObjectBehaviour(\"" << className << "\", asBEHAVE_ADDREF, \"void f()\", AS_FUNCTION_OBJLAST(FakeAddRef), AS_CALL_CDECL_OBJLAST);\n"
  430. " engine->RegisterObjectBehaviour(\"" << className << "\", asBEHAVE_RELEASE, \"void f()\", AS_FUNCTION_OBJLAST(FakeReleaseRef), AS_CALL_CDECL_OBJLAST);\n";
  431. }
  432. // https://www.angelcode.com/angelscript/sdk/docs/manual/doc_script_class_ops.html
  433. static string CppMethodNameToAS(const MethodAnalyzer& methodAnalyzer)
  434. {
  435. string name = methodAnalyzer.GetName();
  436. if (name == "operator=")
  437. return "opAssign";
  438. if (name == "operator+")
  439. return "opAdd";
  440. if (name == "operator-")
  441. {
  442. if (!methodAnalyzer.GetParams().size()) // If no params
  443. return "opNeg"; // then unary minus
  444. else
  445. return "opSub";
  446. }
  447. if (name == "operator*")
  448. return "opMul";
  449. if (name == "operator/")
  450. return "opDiv";
  451. if (name == "operator+=")
  452. return "opAddAssign";
  453. if (name == "operator-=")
  454. return "opSubAssign";
  455. if (name == "operator*=")
  456. return "opMulAssign";
  457. if (name == "operator/=")
  458. return "opDivAssign";
  459. if (name == "operator==")
  460. return "opEquals";
  461. if (name == "operator[]")
  462. return "opIndex";
  463. // Conversion to another type operator
  464. if (StartsWith(name, "operator "))
  465. {
  466. if (methodAnalyzer.IsExplicit())
  467. return "opConv";
  468. else
  469. return "opImplConv";
  470. }
  471. if (name == "operator!=")
  472. throw Exception("Only operator== is needed");
  473. if (name == "operator<")
  474. throw Exception("Registerd as opCmp separately");
  475. if (name == "operator>")
  476. throw Exception("Registerd as opCmp separately");
  477. return name;
  478. }
  479. // Iterate over overrided funcions
  480. static bool HaveMark(const MethodAnalyzer& methodAnalyzer, const string& mark)
  481. {
  482. if (Contains(methodAnalyzer.GetComment(), mark))
  483. return true;
  484. shared_ptr<MethodAnalyzer> reimplements = methodAnalyzer.Reimplements();
  485. if (!reimplements)
  486. return false;
  487. return HaveMark(*reimplements, mark);
  488. }
  489. // Can return BIND_AS_ALIAS_xxxx or BIND_AS_PROPERTY
  490. // Return "" if no this marks
  491. static string GetPropertyMark(const MethodAnalyzer& methodAnalyzer)
  492. {
  493. string comment = methodAnalyzer.GetComment();
  494. smatch match;
  495. regex_match(comment, match, regex(".*\\b(BIND_AS_ALIAS_.+?)\\b.*"));
  496. if (match.size() == 2)
  497. return match[1].str();
  498. regex_match(comment, match, regex(".*\\bBIND_AS_PROPERTY\\b.*"));
  499. if (match.size() == 1)
  500. return "BIND_AS_PROPERTY";
  501. shared_ptr<MethodAnalyzer> reimplements = methodAnalyzer.Reimplements();
  502. if (!reimplements)
  503. return "";
  504. return GetPropertyMark(*reimplements);
  505. }
  506. // https://www.angelcode.com/angelscript/sdk/docs/manual/doc_reg_objprop.html
  507. static string CppMethodNameToASProperty(const MethodAnalyzer& methodAnalyzer)
  508. {
  509. string name = methodAnalyzer.GetName();
  510. if (StartsWith(name, "Is") || StartsWith(name, "Get"))
  511. {
  512. string result = CutStart(name, "Is");
  513. result = CutStart(result, "Get");
  514. result = "get_" + FirstCharToLower(result);
  515. return result;
  516. }
  517. if (StartsWith(name, "Set"))
  518. {
  519. string result = CutStart(name, "Set");
  520. result = "set_" + FirstCharToLower(result);
  521. return result;
  522. }
  523. if (methodAnalyzer.CanBeGetProperty())
  524. {
  525. string result = name;
  526. result = "get_" + FirstCharToLower(result);
  527. return result;
  528. }
  529. if (methodAnalyzer.CanBeSetProperty())
  530. {
  531. string result = name;
  532. result = "set_" + FirstCharToLower(result);
  533. return result;
  534. }
  535. throw Exception("Can not be property");
  536. }
  537. static void RegisterMethod(const MethodAnalyzer& methodAnalyzer, bool templateVersion)
  538. {
  539. if (methodAnalyzer.IsDefine()) // URHO3D_OBJECT
  540. return;
  541. if (!methodAnalyzer.IsPublic())
  542. return;
  543. if (methodAnalyzer.IsParentDestructor())
  544. return;
  545. if (methodAnalyzer.IsParentConstructor())
  546. return;
  547. if (!templateVersion)
  548. {
  549. if (!methodAnalyzer.IsStatic())
  550. return;
  551. }
  552. //shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(functionAnalyzer.GetClass().GetClassName());
  553. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(methodAnalyzer.GetClass().GetClassName()).get();
  554. if (HaveMark(methodAnalyzer, "NO_BIND"))
  555. {
  556. result->reg_ << " // " << methodAnalyzer.GetLocation() << "\n";
  557. result->reg_ << " // Not registered because have @nobind mark\n";
  558. return;
  559. }
  560. if (HaveMark(methodAnalyzer, "MANUAL_BIND"))
  561. {
  562. result->reg_ << " // " << methodAnalyzer.GetLocation() << "\n";
  563. result->reg_ << " // Not registered because have @manualbind mark\n";
  564. return;
  565. }
  566. if (methodAnalyzer.IsTemplate())
  567. {
  568. result->reg_ << " // " << methodAnalyzer.GetLocation() << "\n";
  569. result->reg_ << " // Not registered because template\n";
  570. return;
  571. }
  572. if (methodAnalyzer.IsStatic())
  573. {
  574. ClassStaticFunctionAnalyzer classStaticFunctionAnalyzer(methodAnalyzer.GetClass(), methodAnalyzer.GetMemberdef());
  575. //RegisterStaticFunction(classStaticFunctionAnalyzer, result);
  576. return;
  577. }
  578. if (methodAnalyzer.IsDeleted())
  579. return;
  580. /*if (function.IsPureVirtual()) // Register pure virtual function but not register constructor for this class
  581. return;*/
  582. // Do not register destructor for refcounted because object is deleted by himself
  583. if (methodAnalyzer.IsThisDestructor()
  584. && (methodAnalyzer.GetClass().IsRefCounted() || Contains(methodAnalyzer.GetClass().GetComment(), "FAKE_REF")))
  585. {
  586. return;
  587. }
  588. string functionName = methodAnalyzer.GetName();
  589. if (functionName == "operator!=")
  590. return;
  591. // Already registered as sigle function opCmp
  592. if (functionName == "operator<" || functionName == "operator>")
  593. return;
  594. if (functionName == "AddRef" || functionName == "ReleaseRef")
  595. {
  596. RegisterAddReleaseRef(methodAnalyzer, templateVersion);
  597. return;
  598. }
  599. if (methodAnalyzer.IsThisConstructor())
  600. {
  601. if (methodAnalyzer.GetClass().IsRefCounted() || Contains(methodAnalyzer.GetClass().GetComment(), "FAKE_REF"))
  602. {
  603. // Do not register construnctor for abstract class
  604. if (methodAnalyzer.GetClass().IsAbstract())
  605. return;
  606. RegisterRefCountedConstructor(methodAnalyzer, templateVersion);
  607. }
  608. else
  609. {
  610. RegisterValueConstructor(methodAnalyzer, templateVersion);
  611. }
  612. return;
  613. }
  614. if (methodAnalyzer.IsThisDestructor())
  615. return;
  616. result->reg_ << " // " << methodAnalyzer.GetLocation() << "\n";
  617. vector<ParamAnalyzer> params = methodAnalyzer.GetParams();
  618. vector<ConvertedVariable> convertedParams;
  619. bool needWrapper = false;
  620. for (const ParamAnalyzer& param : params)
  621. {
  622. ConvertedVariable conv;
  623. try
  624. {
  625. conv = CppVariableToAS(param.GetType(), VariableUsage::FunctionParameter, param.GetDeclname(), param.GetDefval());
  626. }
  627. catch (const Exception& e)
  628. {
  629. result->reg_ << " // " << e.what() << "\n";
  630. return;
  631. }
  632. if (conv.NeedWrapper())
  633. needWrapper = true;
  634. convertedParams.push_back(conv);
  635. }
  636. ConvertedVariable retConv;
  637. try
  638. {
  639. retConv = CppVariableToAS(methodAnalyzer.GetReturnType(), VariableUsage::FunctionReturn);
  640. }
  641. catch (const Exception& e)
  642. {
  643. result->reg_ << " // " << e.what() << "\n";
  644. return;
  645. }
  646. if (retConv.NeedWrapper())
  647. needWrapper = true;
  648. string asReturnType = retConv.asDeclaration_;
  649. string asFunctionName = methodAnalyzer.GetName();
  650. if (methodAnalyzer.IsConsversionOperator())
  651. asReturnType = CutStart(asFunctionName, "operator ");
  652. try
  653. {
  654. asFunctionName = CppMethodNameToAS(methodAnalyzer);
  655. }
  656. catch (const Exception& e)
  657. {
  658. result->reg_ << " // " << e.what() << "\n";
  659. return;
  660. }
  661. if (needWrapper)
  662. result->glue_ << GenerateWrapper(methodAnalyzer, templateVersion, convertedParams, retConv);
  663. string decl = asReturnType + " " + asFunctionName + "(" + JoinASDeclarations(convertedParams) + ")";
  664. if (methodAnalyzer.IsConst())
  665. decl += " const";
  666. if (templateVersion)
  667. result->reg_ << " engine->RegisterObjectMethod(className, \"" << decl << "\", ";
  668. else
  669. result->reg_ << " engine->RegisterObjectMethod(\"" << methodAnalyzer.GetClassName() << "\", \"" << decl << "\", ";
  670. if (needWrapper)
  671. result->reg_ << "AS_FUNCTION_OBJFIRST(" << GenerateWrapperName(methodAnalyzer, templateVersion) << "), AS_CALL_CDECL_OBJFIRST);\n";
  672. else
  673. result->reg_ << Generate_asMETHODPR(methodAnalyzer, templateVersion) << ", AS_CALL_THISCALL);\n";
  674. // Also register as property if needed
  675. string propertyMark = GetPropertyMark(methodAnalyzer);
  676. if (!propertyMark.empty())
  677. {
  678. if (StartsWith(propertyMark, "BIND_AS_ALIAS_"))
  679. {
  680. asFunctionName = CutStart(propertyMark, "BIND_AS_ALIAS_");
  681. }
  682. else
  683. {
  684. try
  685. {
  686. asFunctionName = CppMethodNameToASProperty(methodAnalyzer);
  687. }
  688. catch (const Exception& e)
  689. {
  690. result->reg_ << " // " << e.what() << "\n";
  691. return;
  692. }
  693. }
  694. decl = asReturnType + " " + asFunctionName + "(" + JoinASDeclarations(convertedParams) + ")";
  695. if (methodAnalyzer.IsConst())
  696. decl += " const";
  697. if (templateVersion)
  698. result->reg_ << " engine->RegisterObjectMethod(className, \"" << decl << "\", ";
  699. else
  700. result->reg_ << " engine->RegisterObjectMethod(\"" << methodAnalyzer.GetClassName() << "\", \"" << decl << "\", ";
  701. if (needWrapper)
  702. result->reg_ << "AS_FUNCTION_OBJFIRST(" << GenerateWrapperName(methodAnalyzer, templateVersion) << "), AS_CALL_CDECL_OBJFIRST);\n";
  703. else
  704. result->reg_ << Generate_asMETHODPR(methodAnalyzer, templateVersion) << ", AS_CALL_THISCALL);\n";
  705. }
  706. }
  707. static void RegisterClassVarAsProperty(FieldAnalyzer& variable)
  708. {
  709. return;
  710. if (!variable.IsPublic())
  711. return;
  712. shared_ptr<ASGeneratedFile_Members> result = GetGeneratedFile(variable.GetClassName());
  713. if (Contains(variable.GetComment(), "NO_BIND"))
  714. {
  715. result->reg_ << " // " << variable.GetLocation() << "\n";
  716. result->reg_ << " // Not registered because have @nobind mark\n";
  717. return;
  718. }
  719. if (Contains(variable.GetComment(), "MANUAL_BIND"))
  720. {
  721. result->reg_ << " // " << variable.GetLocation() << "\n";
  722. result->reg_ << " // Not registered because have @manualbind mark\n";
  723. return;
  724. }
  725. if (variable.IsStatic())
  726. {
  727. result->reg_ << " // " << variable.GetLocation() << "\n";
  728. string asType;
  729. try
  730. {
  731. asType = CppTypeToAS(variable.GetType(), TypeUsage::StaticField);
  732. }
  733. catch (const Exception& e)
  734. {
  735. result->reg_ << " // " << e.what() << "\n";
  736. return;
  737. }
  738. if (variable.GetType().IsConst())
  739. asType = "const " + asType;
  740. asType = ReplaceAll(asType, "struct ", "");
  741. string className = variable.GetClassName();
  742. result->reg_ <<
  743. " engine->SetDefaultNamespace(\"" << className << "\");\n";
  744. result->reg_ <<
  745. " engine->RegisterGlobalProperty("
  746. "\"" << asType << " " << variable.GetName() << "\", "
  747. "(void*)&" << className << "::" << variable.GetName() << ");\n";
  748. result->reg_ << " engine->SetDefaultNamespace(\"\");\n";
  749. }
  750. else
  751. {
  752. result->reg_ << " // " << variable.GetLocation() << "\n";
  753. if (variable.IsArray())
  754. {
  755. result->reg_ << " // Not registered because array\n";
  756. return;
  757. }
  758. if (variable.GetType().IsPointer())
  759. {
  760. result->reg_ << " // " << variable.GetType().ToString() << " can not be registered\n";
  761. return;
  762. }
  763. string propType;
  764. try
  765. {
  766. propType = CppTypeToAS(variable.GetType(), TypeUsage::Field);
  767. }
  768. catch (const Exception& e)
  769. {
  770. result->reg_ << " // " << e.what() << "\n";
  771. return;
  772. }
  773. string varName = variable.GetName();
  774. assert(!varName.empty());
  775. string propName = CutEnd(varName, "_");
  776. string className = variable.GetClassName();
  777. result->reg_ <<
  778. " engine->RegisterObjectProperty("
  779. "\"" << className << "\", "
  780. "\"" << propType << " " << propName << "\", "
  781. "offsetof(" << className << ", " << varName << "));\n";
  782. }
  783. }
  784. static void RegisterObjectMembers(const ClassAnalyzer& classAnalyzer, bool templateVersion)
  785. {
  786. string header = classAnalyzer.GetHeaderFile();
  787. ASGeneratedFile_Base* result = templateVersion ? (ASGeneratedFile_Base*)_result_Templates.get() : (ASGeneratedFile_Base*)GetGeneratedFile(classAnalyzer.GetClassName()).get();
  788. Result::AddHeader(header);
  789. string insideDefine = InsideDefine(header);
  790. if (!insideDefine.empty())
  791. result->reg_ << "#ifdef " << insideDefine << "\n";
  792. vector<FieldAnalyzer> variables = classAnalyzer.GetAllFields();
  793. for (FieldAnalyzer variable : variables)
  794. RegisterClassVarAsProperty(variable);
  795. vector<MethodAnalyzer> methods = classAnalyzer.GetAllMethods();
  796. for (const MethodAnalyzer& method : methods)
  797. RegisterMethod(method, templateVersion);
  798. RegisterImplicitlyDeclaredMethods(classAnalyzer);
  799. // 2 operators is replaced by single function opCmp
  800. if (classAnalyzer.ContainsMethod("operator>") || classAnalyzer.ContainsMethod("operator>"))
  801. RegisterComparisonOperator(classAnalyzer);
  802. if (Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  803. RegisterFakeAddReleaseRef(classAnalyzer);
  804. vector<ClassAnalyzer> classList = classAnalyzer.GetAllBaseClasses();
  805. classList.push_back(classAnalyzer); // Also add current class to list
  806. string className = classAnalyzer.GetClassName();
  807. for (ClassAnalyzer& cl : classList)
  808. {
  809. string clName = cl.GetClassName();
  810. result->reg_ << "#ifdef REGISTER_MANUAL_PART_" << clName << "\n";
  811. if (templateVersion)
  812. result->reg_ << " REGISTER_MANUAL_PART_" << clName << "(T, className)\n";
  813. else
  814. result->reg_ << " REGISTER_MANUAL_PART_" << clName << "(" << className << ", \"" << className << "\")\n";
  815. result->reg_ << "#endif\n";
  816. }
  817. if (classAnalyzer.IsRefCounted() || Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  818. {
  819. vector<ClassAnalyzer> baseClasses = classAnalyzer.GetAllBaseClasses();
  820. for (ClassAnalyzer baseClass : baseClasses)
  821. {
  822. if (baseClass.IsRefCounted() || Contains(baseClass.GetComment(), "FAKE_REF"))
  823. {
  824. string baseClassName = baseClass.GetClassName();
  825. if (templateVersion)
  826. result->reg_ << " RegisterSubclass<" << baseClassName << ", T>(engine, \"" << baseClassName << "\", className);\n";
  827. else
  828. result->reg_ << " RegisterSubclass<" << baseClassName << ", " << className << ">(engine, \"" << baseClassName << "\", \"" << className << "\");\n";
  829. }
  830. }
  831. // In template also add current class
  832. if (templateVersion)
  833. result->reg_ << " RegisterSubclass<" << className << ", T>(engine, \"" << className << "\", className);\n";
  834. }
  835. /*if (!analyzer.IsRefCounted()) // Executed also for FAKE_REF
  836. // TODO refcounted <=> value conversion if refcounted is chield value
  837. {
  838. vector<ClassAnalyzer> baseClasses = analyzer.GetAllBaseClasses();
  839. for (ClassAnalyzer baseClass : baseClasses)
  840. {
  841. if (!baseClass.IsRefCounted()) // Executed also for FAKE_REF
  842. {
  843. ASResult::reg_ << " RegisterSubclassValue<" << baseClass.GetClassName()
  844. << ", " << analyzer.GetClassName() << ">(engine, "
  845. << "\"" << baseClass.GetClassName() << "\", "
  846. << "\"" << analyzer.GetClassName() << "\");\n";
  847. }
  848. }
  849. }
  850. */
  851. if (!insideDefine.empty())
  852. result->reg_ << "#endif\n";
  853. if (!templateVersion)
  854. result->reg_ << "\n";
  855. }
  856. // Only template version
  857. static void RegisterObjectType(const ClassAnalyzer& classAnalyzer/*, bool templateVersion*/)
  858. {
  859. string header = classAnalyzer.GetHeaderFile();
  860. ASGeneratedFile_Base* result = (ASGeneratedFile_Base*)_result_Templates.get();
  861. Result::AddHeader(header);
  862. string insideDefine = InsideDefine(header);
  863. if (!insideDefine.empty())
  864. result->reg_ << "#ifdef " << insideDefine << "\n";
  865. result->reg_ << " // " + classAnalyzer.GetLocation() + "\n";
  866. string className = classAnalyzer.GetClassName();
  867. if (classAnalyzer.IsRefCounted() || Contains(classAnalyzer.GetComment(), "FAKE_REF"))
  868. {
  869. //if (templateVersion)
  870. result->reg_ << " engine->RegisterObjectType(className, 0, asOBJ_REF);\n";
  871. //else
  872. // result->reg_ << " engine->RegisterObjectType(\"" << className << "\", 0, asOBJ_REF);\n";
  873. }
  874. else // Value type
  875. {
  876. // TODO templateversion
  877. /* string flags = "asOBJ_VALUE | asGetTypeTraits<" + className + ">()";
  878. if (classAnalyzer.IsPod())
  879. {
  880. flags += " | asOBJ_POD";
  881. if (classAnalyzer.AllFloats())
  882. flags += " | asOBJ_APP_CLASS_ALLFLOATS";
  883. else if (classAnalyzer.AllInts())
  884. flags += " | asOBJ_APP_CLASS_ALLINTS";
  885. }
  886. result->reg_ << " engine->RegisterObjectType(\"" << className << "\", sizeof(" << className << "), " << flags << ");\n";*/
  887. }
  888. if (!insideDefine.empty())
  889. result->reg_ << "#endif\n";
  890. }
  891. static void ProcessClass(const ClassAnalyzer& classAnalyzer, bool templateVersion)
  892. {
  893. if (classAnalyzer.IsInternal())
  894. return;
  895. if (classAnalyzer.IsAbstract() && !(classAnalyzer.IsRefCounted() || Contains(classAnalyzer.GetComment(), "FAKE_REF")))
  896. return;
  897. if (Contains(classAnalyzer.GetComment(), "NO_BIND"))
  898. {
  899. //_result_Classes->reg_ << " // " << classAnalyzer.GetLocation() << "\n";
  900. //_result_Classes->reg_ << " // Not registered because have @nobind mark\n";
  901. return;
  902. }
  903. if (Contains(classAnalyzer.GetComment(), "MANUAL_BIND"))
  904. {
  905. //_result_Classes->reg_ << " // " << classAnalyzer.GetLocation() << "\n";
  906. //_result_Classes->reg_ << " // Not registered because have @manualbind mark\n";
  907. return;
  908. }
  909. string header = classAnalyzer.GetHeaderFile();
  910. if (IsIgnoredHeader(header))
  911. {
  912. Result::AddHeader(header);
  913. return;
  914. }
  915. if (classAnalyzer.IsTemplate())
  916. return;
  917. if (templateVersion)
  918. _result_Templates->reg_ << "template <class T> void Register" << classAnalyzer.GetClassName() << "(asIScriptEngine* engine, const char* className)\n{\n";
  919. if (templateVersion)
  920. RegisterObjectType(classAnalyzer);
  921. RegisterObjectMembers(classAnalyzer, templateVersion);
  922. if (templateVersion)
  923. _result_Templates->reg_ << "}\n\n";
  924. }
  925. void ProcessAllClasses()
  926. {
  927. return;
  928. _result_Members_A = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_A.cpp", "ASRegisterGenerated_Members_A");
  929. _result_Members_B = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_B.cpp", "ASRegisterGenerated_Members_B");
  930. _result_Members_Constraint = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Constraint.cpp", "ASRegisterGenerated_Members_Constraint");
  931. _result_Members_Ca_Cm = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Ca_Cm.cpp", "ASRegisterGenerated_Members_Ca_Cm");
  932. _result_Members_Cn_Cz = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Cn_Cz.cpp", "ASRegisterGenerated_Members_Cn_Cz");
  933. _result_Members_D = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_D.cpp", "ASRegisterGenerated_Members_D");
  934. _result_Members_E = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_E.cpp", "ASRegisterGenerated_Members_E");
  935. _result_Members_F = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_F.cpp", "ASRegisterGenerated_Members_F");
  936. _result_Members_G = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_G.cpp", "ASRegisterGenerated_Members_G");
  937. _result_Members_H = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_H.cpp", "ASRegisterGenerated_Members_H");
  938. _result_Members_I = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_I.cpp", "ASRegisterGenerated_Members_I");
  939. _result_Members_J = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_J.cpp", "ASRegisterGenerated_Members_J");
  940. _result_Members_K = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_K.cpp", "ASRegisterGenerated_Members_K");
  941. _result_Members_L = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_L.cpp", "ASRegisterGenerated_Members_L");
  942. _result_Members_M = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_M.cpp", "ASRegisterGenerated_Members_M");
  943. _result_Members_N = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_N.cpp", "ASRegisterGenerated_Members_N");
  944. _result_Members_O = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_O.cpp", "ASRegisterGenerated_Members_O");
  945. _result_Members_P = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_P.cpp", "ASRegisterGenerated_Members_P");
  946. _result_Members_Q = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Q.cpp", "ASRegisterGenerated_Members_Q");
  947. _result_Members_R = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_R.cpp", "ASRegisterGenerated_Members_R");
  948. _result_Members_Sa_Sm = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Sa_Sm.cpp", "ASRegisterGenerated_Members_Sa_Sm");
  949. _result_Members_Sn_Sz = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Sn_Sz.cpp", "ASRegisterGenerated_Members_Sn_Sz");
  950. _result_Members_Ta_Tm = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Ta_Tm.cpp", "ASRegisterGenerated_Members_Ta_Tm");
  951. _result_Members_Tn_Tz = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Tn_Tz.cpp", "ASRegisterGenerated_Members_Tn_Tz");
  952. _result_Members_U = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_U.cpp", "ASRegisterGenerated_Members_U");
  953. _result_Members_V = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_V.cpp", "ASRegisterGenerated_Members_V");
  954. _result_Members_W = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_W.cpp", "ASRegisterGenerated_Members_W");
  955. _result_Members_X = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_X.cpp", "ASRegisterGenerated_Members_X");
  956. _result_Members_Y = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Y.cpp", "ASRegisterGenerated_Members_Y");
  957. _result_Members_Z = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Z.cpp", "ASRegisterGenerated_Members_Z");
  958. _result_Members_Other = make_shared<ASGeneratedFile_Members>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Members_Other.cpp", "ASRegisterGenerated_Members_Other");
  959. _result_Templates = make_shared<ASGeneratedFile_Templates>(_sourceDir + "/Source/Urho3D/AngelScript/Generated_Templates.h");
  960. // Sorting keys to make class order compiler independent
  961. vector<string> classIDs;
  962. classIDs.reserve(SourceData::classesByID_.size());
  963. for (pair<const string, xml_node>& it : SourceData::classesByID_)
  964. classIDs.push_back(it.first);
  965. sort(classIDs.begin(), classIDs.end());
  966. for (string classID : classIDs)
  967. {
  968. xml_node compounddef = SourceData::classesByID_[classID];
  969. ClassAnalyzer analyzer(compounddef);
  970. ProcessClass(analyzer, false);
  971. // Generate also template version of bindings
  972. if (Contains(analyzer.GetComment(), "TEMPLATE_VERSION"))
  973. ProcessClass(analyzer, true);
  974. }
  975. _result_Members_A->Save();
  976. _result_Members_B->Save();
  977. _result_Members_Constraint->Save();
  978. _result_Members_Ca_Cm->Save();
  979. _result_Members_Cn_Cz->Save();
  980. _result_Members_D->Save();
  981. _result_Members_E->Save();
  982. _result_Members_F->Save();
  983. _result_Members_G->Save();
  984. _result_Members_H->Save();
  985. _result_Members_I->Save();
  986. _result_Members_J->Save();
  987. _result_Members_K->Save();
  988. _result_Members_L->Save();
  989. _result_Members_M->Save();
  990. _result_Members_N->Save();
  991. _result_Members_O->Save();
  992. _result_Members_P->Save();
  993. _result_Members_Q->Save();
  994. _result_Members_R->Save();
  995. _result_Members_Sa_Sm->Save();
  996. _result_Members_Sn_Sz->Save();
  997. _result_Members_Ta_Tm->Save();
  998. _result_Members_Tn_Tz->Save();
  999. _result_Members_U->Save();
  1000. _result_Members_V->Save();
  1001. _result_Members_W->Save();
  1002. _result_Members_X->Save();
  1003. _result_Members_Y->Save();
  1004. _result_Members_Z->Save();
  1005. _result_Members_Other->Save();
  1006. _result_Templates->Save();
  1007. }
  1008. }