ASClassBinder.cpp 44 KB

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