VariableTerm.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. using System;
  2. using NTERA.EmuEra.Game.EraEmu.GameData.Expression;
  3. using NTERA.EmuEra.Game.EraEmu.Sub;
  4. namespace NTERA.EmuEra.Game.EraEmu.GameData.Variable
  5. {
  6. internal class VariableTerm : IOperandTerm
  7. {
  8. protected VariableTerm(VariableToken token) : base(token.VariableType) { }
  9. public VariableTerm(VariableToken token, IOperandTerm[] args)
  10. : base(token.VariableType)
  11. {
  12. Identifier = token;
  13. arguments = args;
  14. transporter = new Int64[arguments.Length];
  15. allArgIsConst = false;
  16. for (int i = 0; i < arguments.Length; i++)
  17. {
  18. if (!(arguments[i] is SingleTerm))
  19. return;
  20. transporter[i] = ((SingleTerm)arguments[i]).Int;
  21. }
  22. allArgIsConst = true;
  23. }
  24. public VariableToken Identifier;
  25. private readonly IOperandTerm[] arguments;
  26. protected Int64[] transporter;
  27. protected bool allArgIsConst;
  28. public Int64 GetElementInt(int i, ExpressionMediator exm)
  29. {
  30. if (allArgIsConst)
  31. return transporter[i];
  32. return arguments[i].GetIntValue(exm);
  33. }
  34. public bool isAllConst => allArgIsConst;
  35. public int getEl1forArg => (int)transporter[0];
  36. public override Int64 GetIntValue(ExpressionMediator exm)
  37. {
  38. try
  39. {
  40. if (!allArgIsConst)
  41. for (int i = 0; i < arguments.Length; i++)
  42. transporter[i] = arguments[i].GetIntValue(exm);
  43. return Identifier.GetIntValue(exm, transporter);
  44. }
  45. catch (Exception e)
  46. {
  47. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  48. Identifier.CheckElement(transporter);
  49. throw;
  50. }
  51. }
  52. //, bool tl = false
  53. public override string GetStrValue(ExpressionMediator exm, bool translate=false)
  54. {
  55. try
  56. {
  57. if (!allArgIsConst)
  58. for (int i = 0; i < arguments.Length; i++)
  59. transporter[i] = arguments[i].GetIntValue(exm);
  60. string ret = Identifier.GetStrValue(exm, transporter, translate);
  61. if (ret == null)
  62. return "";
  63. return ret;
  64. }
  65. catch (Exception e)
  66. {
  67. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  68. Identifier.CheckElement(transporter);
  69. throw;
  70. }
  71. }
  72. public virtual void SetValue(Int64 value, ExpressionMediator exm)
  73. {
  74. try
  75. {
  76. if (!allArgIsConst)
  77. for (int i = 0; i < arguments.Length; i++)
  78. transporter[i] = arguments[i].GetIntValue(exm);
  79. Identifier.SetValue(value, transporter);
  80. }
  81. catch (Exception e)
  82. {
  83. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  84. Identifier.CheckElement(transporter);
  85. throw;
  86. }
  87. }
  88. public virtual void SetValue(string value, ExpressionMediator exm)
  89. {
  90. try
  91. {
  92. if (!allArgIsConst)
  93. for (int i = 0; i < arguments.Length; i++)
  94. transporter[i] = arguments[i].GetIntValue(exm);
  95. Identifier.SetValue(value, transporter);
  96. }
  97. catch (Exception e)
  98. {
  99. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  100. Identifier.CheckElement(transporter);
  101. throw e;
  102. }
  103. }
  104. public virtual void SetValue(Int64[] array, ExpressionMediator exm)
  105. {
  106. try
  107. {
  108. if (!allArgIsConst)
  109. for (int i = 0; i < arguments.Length; i++)
  110. transporter[i] = arguments[i].GetIntValue(exm);
  111. Identifier.SetValue(array, transporter);
  112. }
  113. catch (Exception e)
  114. {
  115. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  116. {
  117. Identifier.CheckElement(transporter);
  118. throw new CodeEE("配列変数" + Identifier.Name + "の要素数を超えて代入しようとしました");
  119. }
  120. throw;
  121. }
  122. }
  123. public virtual void SetValue(string[] array, ExpressionMediator exm)
  124. {
  125. try
  126. {
  127. if (!allArgIsConst)
  128. for (int i = 0; i < arguments.Length; i++)
  129. transporter[i] = arguments[i].GetIntValue(exm);
  130. Identifier.SetValue(array, transporter);
  131. }
  132. catch (Exception e)
  133. {
  134. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  135. {
  136. Identifier.CheckElement(transporter);
  137. throw new CodeEE("配列変数" + Identifier.Name + "の要素数を超えて代入しようとしました");
  138. }
  139. throw;
  140. }
  141. }
  142. public virtual Int64 PlusValue(Int64 value, ExpressionMediator exm)
  143. {
  144. try
  145. {
  146. if (!allArgIsConst)
  147. for (int i = 0; i < arguments.Length; i++)
  148. transporter[i] = arguments[i].GetIntValue(exm);
  149. return Identifier.PlusValue(value, transporter);
  150. }
  151. catch (Exception e)
  152. {
  153. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  154. Identifier.CheckElement(transporter);
  155. throw;
  156. }
  157. }
  158. public override SingleTerm GetValue(ExpressionMediator exm, bool tryTranslate =false)
  159. {
  160. if (Identifier.VariableType == typeof(Int64))
  161. return new SingleTerm(GetIntValue(exm));
  162. return new SingleTerm(GetStrValue(exm, tryTranslate));
  163. }
  164. public virtual void SetValue(SingleTerm value, ExpressionMediator exm)
  165. {
  166. if (Identifier.VariableType == typeof(Int64))
  167. SetValue(value.Int, exm);
  168. else
  169. SetValue(value.Str, exm);
  170. }
  171. public virtual void SetValue(IOperandTerm value, ExpressionMediator exm)
  172. {
  173. if (Identifier.VariableType == typeof(Int64))
  174. SetValue(value.GetIntValue(exm), exm);
  175. else
  176. SetValue(value.GetStrValue(exm), exm);
  177. }
  178. public Int32 GetLength()
  179. {
  180. return Identifier.GetLength();
  181. }
  182. public Int32 GetLength(int dimension)
  183. {
  184. return Identifier.GetLength(dimension);
  185. }
  186. public Int32 GetLastLength()
  187. {
  188. if (Identifier.IsArray1D)
  189. return Identifier.GetLength();
  190. if (Identifier.IsArray2D)
  191. return Identifier.GetLength(1);
  192. if (Identifier.IsArray3D)
  193. return Identifier.GetLength(2);
  194. return 0;
  195. }
  196. public virtual FixedVariableTerm GetFixedVariableTerm(ExpressionMediator exm)
  197. {
  198. if (!allArgIsConst)
  199. for (int i = 0; i < arguments.Length; i++)
  200. transporter[i] = arguments[i].GetIntValue(exm);
  201. FixedVariableTerm fp = new FixedVariableTerm(Identifier);
  202. if (transporter.Length >= 1)
  203. fp.Index1 = transporter[0];
  204. if (transporter.Length >= 2)
  205. fp.Index2 = transporter[1];
  206. if (transporter.Length >= 3)
  207. fp.Index3 = transporter[2];
  208. return fp;
  209. }
  210. public override IOperandTerm Restructure(ExpressionMediator exm, bool tryTranslate=false)
  211. {
  212. bool[] canCheck = new bool[arguments.Length];
  213. allArgIsConst = true;
  214. for (int i = 0; i < arguments.Length; i++)
  215. {
  216. arguments[i] = arguments[i].Restructure(exm);
  217. if (!(arguments[i] is SingleTerm))
  218. {
  219. allArgIsConst = false;
  220. canCheck[i] = false;
  221. }
  222. else
  223. {
  224. //キャラクターデータの第1引数はこの時点でチェックしても意味がないのと
  225. //ARG系は限界超えてても必要な数に拡張されるのでチェックしなくていい
  226. if ((i == 0 && Identifier.IsCharacterData) || Identifier.Name == "ARG" || Identifier.Name == "ARGS")
  227. canCheck[i] = false;
  228. else
  229. canCheck[i] = true;
  230. //if (allArgIsConst)
  231. //チェックのために値が必要
  232. transporter[i] = arguments[i].GetIntValue(exm);
  233. }
  234. }
  235. if (!Identifier.IsReference)
  236. Identifier.CheckElement(transporter, canCheck);
  237. if ((Identifier.CanRestructure) && (allArgIsConst))
  238. return GetValue(exm, tryTranslate);//We pass the bool by here REEEEEEEEEEEEEEEEE
  239. if (allArgIsConst)
  240. return new FixedVariableTerm(Identifier, transporter);
  241. return this;
  242. }
  243. //以下添え字解析用の追加関数
  244. public bool checkSameTerm(VariableTerm term)
  245. {
  246. //添え字が全部定数があることがこの関数の前提(そもそもそうでないと使い道がない)
  247. if (!allArgIsConst)
  248. return false;
  249. if (Identifier.Name != term.Identifier.Name)
  250. return false;
  251. for (int i = 0; i < transporter.Length; i++)
  252. {
  253. if (transporter[i] != term.transporter[i])
  254. return false;
  255. }
  256. return true;
  257. }
  258. public string GetFullString()
  259. {
  260. //添え字が全部定数があることがこの関数の前提(IOperandTermから変数名を取れないため)
  261. if (!allArgIsConst)
  262. return "";
  263. if (Identifier.IsArray1D)
  264. return Identifier.Name + ":" + transporter[0];
  265. if (Identifier.IsArray2D)
  266. return Identifier.Name + ":" + transporter[0] + ":" + transporter[1];
  267. if (Identifier.IsArray3D)
  268. return Identifier.Name + ":" + transporter[0] + ":" + transporter[1] + ":" + transporter[2];
  269. return Identifier.Name;
  270. }
  271. }
  272. internal sealed class FixedVariableTerm : VariableTerm
  273. {
  274. public FixedVariableTerm(VariableToken token)
  275. : base(token)
  276. {
  277. Identifier = token;
  278. transporter = new Int64[3];
  279. allArgIsConst = true;
  280. }
  281. public FixedVariableTerm(VariableToken token, Int64[] args)
  282. : base(token)
  283. {
  284. allArgIsConst = true;
  285. Identifier = token;
  286. transporter = new Int64[3];
  287. for(int i = 0;i< args.Length;i++)
  288. transporter[i] = args[i];
  289. }
  290. public Int64 Index1{get => transporter[0];
  291. set{transporter[0] = value;}}
  292. public Int64 Index2{get => transporter[1];
  293. set{transporter[1] = value;}}
  294. public Int64 Index3{get => transporter[2];
  295. set{transporter[2] = value;}}
  296. public override Int64 GetIntValue(ExpressionMediator exm)
  297. {
  298. try
  299. {
  300. return Identifier.GetIntValue(exm, transporter);
  301. }
  302. catch(Exception e)
  303. {
  304. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  305. Identifier.CheckElement(transporter);
  306. throw;
  307. }
  308. }
  309. //, bool tl = false
  310. public override string GetStrValue(ExpressionMediator exm, bool translate=false)
  311. {
  312. try
  313. {
  314. string ret = Identifier.GetStrValue(exm, transporter, translate);
  315. if (ret == null)
  316. return "";
  317. return ret;
  318. }
  319. catch(Exception e)
  320. {
  321. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  322. Identifier.CheckElement(transporter);
  323. throw;
  324. }
  325. }
  326. public override void SetValue(Int64 value, ExpressionMediator exm)
  327. {
  328. try
  329. {
  330. Identifier.SetValue(value, transporter);
  331. }
  332. catch(Exception e)
  333. {
  334. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  335. Identifier.CheckElement(transporter);
  336. throw;
  337. }
  338. }
  339. public override void SetValue(string value, ExpressionMediator exm)
  340. {
  341. try
  342. {
  343. Identifier.SetValue(value, transporter);
  344. }
  345. catch(Exception e)
  346. {
  347. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  348. Identifier.CheckElement(transporter);
  349. throw;
  350. }
  351. }
  352. public override Int64 PlusValue(Int64 value, ExpressionMediator exm)
  353. {
  354. try
  355. {
  356. return Identifier.PlusValue(value, transporter);
  357. }
  358. catch(Exception e)
  359. {
  360. if ((e is IndexOutOfRangeException) || (e is ArgumentOutOfRangeException) || (e is OverflowException))
  361. Identifier.CheckElement(transporter);
  362. throw;
  363. }
  364. }
  365. public override IOperandTerm Restructure(ExpressionMediator exm, bool tryTranslate=false)
  366. {
  367. if (Identifier.CanRestructure)
  368. return GetValue(exm);
  369. return this;
  370. }
  371. public void IsArrayRangeValid(Int64 index1, Int64 index2, string funcName, Int64 i1, Int64 i2)
  372. {
  373. Identifier.IsArrayRangeValid(transporter, index1, index2, funcName, i1, i2);
  374. }
  375. }
  376. /// <summary>
  377. /// 引数がない変数。値を参照、代入できない
  378. /// </summary>
  379. internal sealed class VariableNoArgTerm : VariableTerm
  380. {
  381. public VariableNoArgTerm(VariableToken token)
  382. : base(token)
  383. {
  384. Identifier = token;
  385. allArgIsConst = true;
  386. }
  387. public override Int64 GetIntValue(ExpressionMediator exm)
  388. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  389. public override string GetStrValue(ExpressionMediator exm, bool translate=false)
  390. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  391. public override void SetValue(Int64 value, ExpressionMediator exm)
  392. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  393. public override void SetValue(string value, ExpressionMediator exm)
  394. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  395. public override void SetValue(Int64[] array, ExpressionMediator exm)
  396. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  397. public override void SetValue(string[] array, ExpressionMediator exm)
  398. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  399. public override Int64 PlusValue(Int64 value, ExpressionMediator exm)
  400. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  401. public override SingleTerm GetValue(ExpressionMediator exm, bool tryTranslate =false)
  402. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  403. public override void SetValue(SingleTerm value, ExpressionMediator exm)
  404. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  405. public override void SetValue(IOperandTerm value, ExpressionMediator exm)
  406. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  407. public override FixedVariableTerm GetFixedVariableTerm(ExpressionMediator exm)
  408. { throw new CodeEE("変数" + Identifier.Name + "に必要な引数が不足しています"); }
  409. public override IOperandTerm Restructure(ExpressionMediator exm, bool tryTranslate=false)
  410. {
  411. return this;
  412. }
  413. }
  414. }