umychart.explainer.wechat.js 52 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030
  1. /*
  2. copyright (c) 2018 jones
  3. http://www.apache.org/licenses/LICENSE-2.0
  4. 开源项目 https://github.com/jones2000/HQChart
  5. jones_2000@163.com
  6. 分析家脚本翻译器
  7. */
  8. import { JSConsole } from "./umychart.console.wechat.js"
  9. import {
  10. JSCommonComplier_ErrorHandler as ErrorHandler,
  11. JSCommonComplier_JSComplier as JSComplier,
  12. JSCommonComplier_JSParser as JSParser,
  13. JSCommonComplier_Syntax as Syntax,
  14. JS_EXECUTE_JOB_ID as JS_EXECUTE_JOB_ID,
  15. g_JSComplierResource as g_JSComplierResource,
  16. } from "./umychart.complier.wechat";
  17. import
  18. {
  19. JSCommonSplit_IFrameSplitOperator as IFrameSplitOperator,
  20. } from './umychart.framesplit.wechat.js'
  21. //脚本说明
  22. function JSExplainer(ast,option)
  23. {
  24. this.AST=ast;
  25. this.ErrorHandler=new ErrorHandler();
  26. this.ErrorCallback; //执行错误回调
  27. this.UpdateUICallback;
  28. this.CallbackParam;
  29. this.JobList=[]; //执行的任务队列
  30. this.VarTable=new Map(); //变量表
  31. this.OutVarTable=[]; //输出变量
  32. //脚本自动变量表, 只读
  33. this.ConstVarTable=new Map(
  34. [
  35. //个股数据
  36. ['CLOSE',"收盘价"],['VOL',"成交量"],['OPEN',"开盘价"],['HIGH',"最高价"],['LOW',"最低价"],['AMOUNT',"成交量"],
  37. ['C',"收盘价"],['V',"成交量"],['O',"开盘价"],['H',"最高价"],['L',"最低价"],['AMO',"成交量"],
  38. ['VOLR',"量比"], ['VOLINSTK',"持仓量"], ["OPI","持仓量"], ["ZSTJJ","均价"], ["QHJSJ","结算价"], ["SETTLE", "结算价"],
  39. //日期类
  40. ['DATE',"日期"],['YEAR',"年份"],['MONTH',"月份"],['PERIOD', "周期"],['WEEK',"星期"],["TIME","时间"],
  41. //大盘数据
  42. ['INDEXA',"大盘成交额"],['INDEXC',"大盘收盘价"],['INDEXH',"大盘最高价"],['INDEXL',"大盘最低价"],['INDEXO',"大盘开盘价"],['INDEXV',"大盘成交量"],
  43. ['INDEXADV',"大盘上涨家数"],['INDEXDEC',"´大盘下跌家数"],
  44. ["ADVANCE","上涨家数"], ['DECLINE', "下跌家数"],
  45. ['FROMOPEN',"当前离开盘分钟数"],
  46. ['TOTALFZNUM', "总分钟数"],
  47. ['CURRBARSCOUNT',"到最后交易的周期"], //到最后交易日的周期数
  48. ['TOTALBARSCOUNT',"总的周期数"],
  49. ['ISLASTBAR',"是否是最后一个周期"], //判断是否为最后一个周期
  50. ['BARSTATUS',"数据位置状态"], //BARSTATUS返回数据位置信息,1表示第一根K线,2表示最后一个数据,0表示中间位置.
  51. ['CAPITAL',"当前流通股本(手)"], ["TOTALCAPITAL","当前总股本(手)"],
  52. ['EXCHANGE',"换手率"], //换手率
  53. ['SETCODE', "市场类型"], //市场类型
  54. ['CODE',"品种代码"], //品种代码
  55. ['STKNAME',"品种名称"], //品种名称
  56. ["TQFLAG","当前复权状态"], //TQFLAG 当前的复权状态,0:无复权 1:前复权 2:后复权
  57. ['HYBLOCK',"所属行业"], //所属行业板块
  58. ['DYBLOCK',"所属地域"], //所属地域板块
  59. ['GNBLOCK',"所属概念"], //所属概念
  60. ["FGBLOCK","所属风格板块"],
  61. ["ZSBLOCK","所属指数板块"],
  62. ["ZHBLOCK",'所属组合板块'],
  63. ["ZDBLOCK",'所属自定义板块'],
  64. ["HYZSCODE","所属行业的板块指数代码"],
  65. ["GNBLOCKNUM","所属概念板块的个数"],
  66. ["FGBLOCKNUM","所属风格板块的个数"],
  67. ["ZSBLOCKNUM","所属指数板块的个数"],
  68. ["ZHBLOCKNUM","所属组合板块的个数"],
  69. ["ZDBLOCKNUM","所属自定义板块的个数"],
  70. ["HYSYL","指数市盈率或个股所属行业的市盈率"],
  71. ["HYSJL","指数市净率或个股所属行业的市净率"],
  72. ['DRAWNULL',"无效数据"]
  73. ]);
  74. if (option)
  75. {
  76. if (option.Callback) this.UpdateUICallback=option.Callback;
  77. if (option.CallbackParam) this.CallbackParam=option.CallbackParam;
  78. if (option.Arguments) this.Arguments=option.Arguments;
  79. }
  80. this.Run=function()
  81. {
  82. try
  83. {
  84. this.OutVarTable=[];
  85. this.VarTable=new Map();
  86. JSConsole.Complier.Log('[JSExecute::JSExplainer] Load Arguments', this.Arguments);
  87. for(let i in this.Arguments) //预定义的变量
  88. {
  89. let item =this.Arguments[i];
  90. this.VarTable.set(item.Name,item.Value);
  91. }
  92. let data=this.RunAST();//执行脚本
  93. JSConsole.Complier.Log('[JSExplainer.Run] explain finish', data);
  94. if (this.UpdateUICallback) //回调发送结果, 可以支持异步
  95. {
  96. JSConsole.Complier.Log('[JSExplainer.Run] invoke UpdateUICallback.');
  97. this.UpdateUICallback(data);
  98. }
  99. }
  100. catch(error)
  101. {
  102. JSConsole.Complier.Log('[JSExplainer.Run] throw error ', error);
  103. if (this.ErrorCallback)
  104. {
  105. this.ErrorCallback(error, this.OutVarTable);
  106. }
  107. }
  108. }
  109. this.RunAST=function()
  110. {
  111. if (!this.AST) this.ThrowError();
  112. if (!this.AST.Body) this.ThrowError();
  113. for(let i in this.AST.Body)
  114. {
  115. let item =this.AST.Body[i];
  116. this.VisitNode(item);
  117. //输出变量
  118. if (item.Type==Syntax.ExpressionStatement && item.Expression)
  119. {
  120. if (item.Expression.Type==Syntax.AssignmentExpression)
  121. {
  122. if (item.Expression.Operator==':' && item.Expression.Left)
  123. {
  124. let assignmentItem=item.Expression;
  125. let varName=assignmentItem.Left.Name;
  126. let outVar=`输出${varName}: ${this.VarTable.get(varName)}`;
  127. this.OutVarTable.push({ Name:varName, Data:outVar,Type:0});
  128. }
  129. else if (item.Expression.Operator==':=' && item.Expression.Left)
  130. {
  131. let assignmentItem=item.Expression;
  132. let varName=assignmentItem.Left.Name;
  133. let outVar=`赋值${varName}: ${this.VarTable.get(varName)}`;
  134. this.OutVarTable.push({ Name:varName, Data:outVar,Type:0, IsOut:false });
  135. }
  136. }
  137. else if (item.Expression.Type==Syntax.CallExpression)
  138. {
  139. let callItem=item.Expression;
  140. if (this.IsDrawFunction(callItem.Callee.Name))
  141. {
  142. let outVar=callItem.Out;
  143. var drawName=callItem.Callee.Name;
  144. this.OutVarTable.push({Name:drawName, Draw:`输出: ${outVar}`, Type:1});
  145. }
  146. else
  147. {
  148. let outVar=callItem.Out;
  149. varName=`__temp_c_${callItem.Callee.Name}_${i}__`;
  150. this.OutVarTable.push({Name:varName, Data:`输出: ${outVar}`,Type:0, NoneName:true});
  151. }
  152. }
  153. else if (item.Expression.Type==Syntax.Identifier)
  154. {
  155. let varName=item.Expression.Name;
  156. let outVar=this.ReadVariable(varName,item.Expression);
  157. varName="__temp_i_"+i+"__";
  158. this.OutVarTable.push({Name:varName, Data:`输出: ${outVar}`, Type:0, NoneName:true});
  159. }
  160. else if (item.Expression.Type==Syntax.Literal) //常量
  161. {
  162. let outVar=item.Expression.Value;
  163. if (IFrameSplitOperator.IsString(outVar) && outVar.indexOf("$")>0)
  164. outVar=this.GetOtherSymbolExplain({ Literal:outVar }, item);
  165. varName="__temp_li_"+i+"__";
  166. var type=0;
  167. this.OutVarTable.push({Name:varName, Data:`输出: ${outVar}`, Type:0, NoneName:true});
  168. }
  169. else if (item.Expression.Type==Syntax.BinaryExpression) // CLOSE+OPEN;
  170. {
  171. var varName="__temp_b_"+i+"__";
  172. let outVar=item.Expression.Out;
  173. this.OutVarTable.push({Name:varName, Data:`输出: ${outVar}`,Type:0, NoneName:true});
  174. }
  175. else if (item.Expression.Type==Syntax.LogicalExpression) //逻辑语句 如 T1 AND T2
  176. {
  177. var varName="__temp_l_"+i+"__";
  178. let outVar=item.Expression.Out;
  179. this.OutVarTable.push({Name:varName, Data:`输出: ${outVar}`,Type:0, NoneName:true});
  180. }
  181. else if (item.Expression.Type==Syntax.SequenceExpression)
  182. {
  183. let varName;
  184. let drawName;
  185. let draw;
  186. let color;
  187. let lineWidth;
  188. let colorStick=false;
  189. let pointDot=false;
  190. let circleDot=false;
  191. let lineStick=false;
  192. let stick=false;
  193. let volStick=false;
  194. let isShow=true;
  195. let isExData=false;
  196. let isDotLine=false;
  197. let isOverlayLine=false; //叠加线
  198. var isNoneName=false;
  199. //显示在位置之上,对于DRAWTEXT和DRAWNUMBER等函数有用,放在语句的最后面(不能与LINETHICK等函数共用),比如:
  200. //DRAWNUMBER(CLOSE>OPEN,HIGH,CLOSE),DRAWABOVE;
  201. var isDrawAbove=false;
  202. for(let j in item.Expression.Expression)
  203. {
  204. let itemExpression=item.Expression.Expression[j];
  205. if (itemExpression.Type==Syntax.AssignmentExpression && itemExpression.Operator==':' && itemExpression.Left)
  206. {
  207. varName=itemExpression.Left.Name;
  208. let varValue=this.VarTable.get(varName);
  209. this.VarTable.set(varName,varValue); //把常量放到变量表里
  210. }
  211. else if (itemExpression.Type==Syntax.Identifier)
  212. {
  213. let value=itemExpression.Name;
  214. if (value==='COLORSTICK') colorStick=true;
  215. else if (value==='POINTDOT') pointDot=true;
  216. else if (value==='CIRCLEDOT') circleDot=true;
  217. else if (value==='DOTLINE') isDotLine=true;
  218. else if (value==='LINESTICK') lineStick=true;
  219. else if (value==='STICK') stick=true;
  220. else if (value==='VOLSTICK') volStick=true;
  221. else if (value==="DRAWABOVE") isDrawAbove=true;
  222. else if (value.indexOf('COLOR')==0) color=value;
  223. else if (value.indexOf('LINETHICK')==0) lineWidth=value;
  224. else if (value.indexOf('NODRAW')==0) isShow=false;
  225. else if (value.indexOf('EXDATA')==0) isExData=true; //扩展数据, 不显示再图形里面
  226. else if (value.indexOf('LINEOVERLAY')==0) isOverlayLine=true;
  227. else
  228. {
  229. varName=itemExpression.Name;
  230. let varValue=this.ReadVariable(varName,itemExpression);
  231. varName="__temp_si_"+i+"__";
  232. isNoneName=true;
  233. this.VarTable.set(varName,varValue); //放到变量表里
  234. }
  235. }
  236. else if(itemExpression.Type==Syntax.Literal) //常量
  237. {
  238. let aryValue=itemExpression.Value;
  239. varName=itemExpression.Value.toString();
  240. isNoneName=true;
  241. this.VarTable.set(varName,aryValue); //把常量放到变量表里
  242. }
  243. else if (itemExpression.Type==Syntax.CallExpression)
  244. {
  245. if (this.IsDrawFunction(itemExpression.Callee.Name))
  246. {
  247. draw=itemExpression.Out;
  248. drawName=itemExpression.Callee.Name;
  249. }
  250. else
  251. {
  252. let varValue=itemExpression.Out;
  253. varName=`__temp_sc_${itemExpression.Callee.Name}_${i}__`;
  254. isNoneName=true;
  255. this.VarTable.set(varName,varValue);
  256. }
  257. }
  258. else if (itemExpression.Type==Syntax.BinaryExpression)
  259. {
  260. varName="__temp_sb_"+i+"__";
  261. let aryValue=itemExpression.Out;
  262. isNoneName=true;
  263. this.VarTable.set(varName,aryValue);
  264. }
  265. }
  266. var outValue;
  267. if (draw) outValue=`输出: ${draw}`;
  268. else if (isNoneName) outValue=`输出: ${this.VarTable.get(varName)}`;
  269. else outValue=`输出${varName}: ${this.VarTable.get(varName)}`;
  270. if (color) outValue+=`,颜色${this.GetColorExplain(color)}`;
  271. if (lineWidth) outValue+=`,线段粗细${this.GetLineWidthExplain(lineWidth)}`;
  272. if (isShow==false) outValue+=",不显示";
  273. if (isDotLine==true) outValue+=",画虚线";
  274. if (isDrawAbove==true) outValue+=',显示在位置之上';
  275. if (pointDot && varName) //圆点
  276. {
  277. outValue+=",画小圆点线";
  278. let value={Name:varName, Data:outValue, Radius:g_JSChartResource.POINTDOT.Radius, Type:3};
  279. this.OutVarTable.push(value);
  280. }
  281. else if (circleDot && varName) //圆点
  282. {
  283. outValue+=",画小圆圈线";
  284. let value={Name:varName, Data:outValue, Radius:g_JSChartResource.CIRCLEDOT.Radius, Type:3};
  285. this.OutVarTable.push(value);
  286. }
  287. else if (lineStick && varName) //LINESTICK 同时画出柱状线和指标线
  288. {
  289. outValue+=",画出柱状线和指标线";
  290. let value={Name:varName, Data:outValue, Type:4};
  291. this.OutVarTable.push(value);
  292. }
  293. else if (stick && varName) //STICK 画柱状线
  294. {
  295. outValue+=",画柱状线";
  296. let value={Name:varName, Data:outValue, Type:5};
  297. this.OutVarTable.push(value);
  298. }
  299. else if (volStick && varName) //VOLSTICK 画彩色柱状线
  300. {
  301. outValue+=",画成交量柱状线";
  302. let value={Name:varName, Data:outValue, Type:6};
  303. this.OutVarTable.push(value);
  304. }
  305. else if (varName && color)
  306. {
  307. let value={Name:varName, Data:outValue, Color:color, Type:0};
  308. this.OutVarTable.push(value);
  309. }
  310. else if (draw) //画图函数
  311. {
  312. var outVar={ Name:drawName, Data:outValue, Type:1 };
  313. this.OutVarTable.push(outVar);
  314. }
  315. else if (colorStick && varName) //CYW: SUM(VAR4,10)/10000, COLORSTICK; 画上下柱子
  316. {
  317. outValue+=",画彩色柱状线";
  318. let value={Name:varName, Data:outValue, Color:color, Type:2};
  319. this.OutVarTable.push(value);
  320. }
  321. else if (varName)
  322. {
  323. let value={Name:varName, Data:outValue,Type:0};
  324. this.OutVarTable.push(value);
  325. }
  326. }
  327. }
  328. }
  329. JSConsole.Complier.Log('[JSExplainer::Run]', this.VarTable);
  330. return this.OutVarTable;
  331. }
  332. this.VisitNode=function(node)
  333. {
  334. switch(node.Type)
  335. {
  336. case Syntax.SequenceExpression:
  337. this.VisitSequenceExpression(node);
  338. break;
  339. case Syntax.ExpressionStatement:
  340. this.VisitNode(node.Expression);
  341. break;
  342. case Syntax.AssignmentExpression:
  343. this.VisitAssignmentExpression(node);
  344. break;
  345. case Syntax.BinaryExpression:
  346. case Syntax.LogicalExpression:
  347. this.VisitBinaryExpression(node);
  348. break;
  349. case Syntax.CallExpression:
  350. this.VisitCallExpression(node);
  351. break;
  352. }
  353. }
  354. this.VisitSequenceExpression=function(node)
  355. {
  356. for(let i in node.Expression)
  357. {
  358. let item =node.Expression[i];
  359. this.VisitNode(item);
  360. }
  361. }
  362. //函数调用
  363. this.VisitCallExpression=function(node)
  364. {
  365. let funcName=node.Callee.Name;
  366. let args=[];
  367. for(let i in node.Arguments)
  368. {
  369. let item=node.Arguments[i];
  370. let value;
  371. if (item.Type==Syntax.BinaryExpression || item.Type==Syntax.LogicalExpression)
  372. value=this.VisitBinaryExpression(item);
  373. else if (item.Type==Syntax.CallExpression)
  374. value=this.VisitCallExpression(item);
  375. else
  376. value=this.GetNodeValue(item);
  377. args.push(value);
  378. }
  379. JSConsole.Complier.Log('[JSExplainer::VisitCallExpression]' , funcName, '(', args.toString() ,')');
  380. if (g_JSComplierResource.IsCustomFunction(funcName))
  381. {
  382. var data=this.Algorithm.CallCustomFunction(funcName, args, this.SymbolData, node);
  383. node.Out=[];
  384. node.Draw=null;
  385. if (data)
  386. {
  387. if (data.Out) node.Out=data.Out;
  388. if (data.Draw) node.Draw=data.Draw;
  389. }
  390. return node.Out;
  391. }
  392. node.Out=this.CallFunctionExplain(funcName, args, node);
  393. return node.Out;
  394. }
  395. this.FUNCTION_INFO_LIST=new Map(
  396. [
  397. ["REF", { Name:"REF", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日前的${args[0]}`; } } ],
  398. ["REFX", { Name:"REFX", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日后的${args[0]}`; } } ],
  399. ["REFV", { Name:"REFV", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日前的(未作平滑处理)${args[0]}`; } } ],
  400. ["REFXV", { Name:"REFXV", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日后的(未作平滑处理)${args[0]}`; } } ],
  401. ["REFDATE", { Name:"REFDATE", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日${args[0]}`; } } ],
  402. ["COUNT", { Name:"COUNT", Param:{ Count:2 }, ToString:function(args) { return `统计${args[1]}日中满足${args[0]}的天数`; } } ],
  403. ["BARSLASTCOUNT", { Name:"BARSLASTCOUNT", Param:{ Count:1 }, ToString:function(args) { return `条件${args[0]}连续成立次数`; } } ],
  404. ["BARSCOUNT", { Name:"BARSCOUNT", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}有效数据周期数`; } } ],
  405. ["BARSLAST", { Name:"BARSLAST", Param:{ Count:1 }, ToString:function(args) { return `上次${args[0]}不为0距今天数`; } } ],
  406. ["BARSNEXT", { Name:"BARSNEXT", Param:{ Count:1 }, ToString:function(args) { return `下次${args[0]}不为0距今天数`; } } ],
  407. ["BARSSINCEN", { Name:"BARSSINCEN", Param:{ Count:2 }, ToString:function(args) { return `在${args[1]}周期内首次${args[0]}距今天数`; } } ],
  408. ["BARSSINCE", { Name:"BARSSINCE", Param:{ Count:1 }, ToString:function(args) { return `首次${args[0]}距今天数`; } } ],
  409. ["HHV", { Name:"HHV", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日内${args[0]}的最高值`; } } ],
  410. ["LLV", { Name:"LLV", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日内${args[0]}的最低值`; } } ],
  411. ["HOD", { Name:"HOD", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日内${args[0]}的高值名次`; } } ],
  412. ["LOD", { Name:"LOD", Param:{ Count:2 }, ToString:function(args) { return `${args[1]}日内${args[0]}的低值名次`; } } ],
  413. ["REVERSE", { Name:"REVERSE", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的相反数`; } } ],
  414. ["FILTER", { Name:"FILTER", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日过滤`; } } ],
  415. ["FILTERX", { Name:"FILTERX", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日反向过滤`; } } ],
  416. ["SUMBARS", { Name:"SUMBARS", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}累加至${args[1]}的天数`; } } ],
  417. ["MA", { Name:"MA", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日简单移动平均`; } } ],
  418. ["SMA", { Name:"SMA", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}的${args[1]}日[${args[2]}日权重]移动平均`; } } ],
  419. ["MEMA", { Name:"MEMA", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日平滑移动平均`; } } ],
  420. ["EMA", { Name:"EMA", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日指数移动平均`; } } ],
  421. ["EXPMA", { Name:"EXPMA", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日指数移动平均`; } } ],
  422. ["WMA", { Name:"WMA", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日加权移动平均`; } } ],
  423. ["DMA", { Name:"DMA", Param:{ Count:2 }, ToString:function(args) { return `以${args[1]}为权重${args[0]}的动态移动平均`; } } ],
  424. ["XMA", { Name:"XMA", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日偏移移动平均`; } } ],
  425. ["RANGE", { Name:"RANGE", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}位于${args[1]}和${args[2]}之间`; } } ],
  426. ["CONST", { Name:"CONST", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的最后一日值`; } } ],
  427. ["TOPRANGE", { Name:"TOPRANGE", Param:{ Count:1 }, ToString:function(args) { return `当前值是近${args[0]}周期的最大值`; } } ],
  428. ["LOWRANGE", { Name:"LOWRANGE", Param:{ Count:1 }, ToString:function(args) { return `当前值是近${args[0]}周期的最小值`; } } ],
  429. ["FINDHIGH", { Name:"FINDHIGH", Param:{ Count:4 }, ToString:function(args) { return `${args[0]}在${args[1]}日前的${args[2]}天内第${args[3]}个最高价`; } } ],
  430. ["FINDHIGHBARS", { Name:"FINDHIGHBARS", Param:{ Count:4 }, ToString:function(args) { return `${args[0]}在${args[1]}日前的${args[2]}天内第${args[3]}个最高价到当前周期的周期数`; } } ],
  431. ["FINDLOW", { Name:"FINDLOW", Param:{ Count:4 }, ToString:function(args) { return `${args[0]}在${args[1]}日前的${args[2]}天内第${args[3]}个最低价`; } } ],
  432. ["FINDLOWBARS", { Name:"FINDLOWBARS", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}在${args[1]}日前的${args[2]}天内第${args[3]}个最低价到当前周期的周期数`; } } ],
  433. ["SUM", { Name:"SUM", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}${args[1]}日累加`; } } ],
  434. ["MULAR", { Name:"MULAR", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}和${args[1]}日累乘`; } } ],
  435. ["AMA", { Name:"AMA", Param:{ Count:2 }, ToString:function(args) { return `以${args[1]}为权重${args[0]}的自适应均线`; } } ],
  436. ["TMA", { Name:"TMA", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}的${args[1]}日[${args[2]}日权重]移动平均`; } } ],
  437. ["CROSS", { Name:"CROSS", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}上穿${args[1]}`; } } ],
  438. ["LONGCROSS", { Name:"LONGCROSS", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}小于${args[1]}保持${args[2]}个交易日后交叉上穿`; } } ],
  439. ["UPNDAY", { Name:"UPNDAY", Param:{ Count:2 }, ToString:function(args) { return `最近${args[1]}日${args[0]}连涨`; } } ],
  440. ["DOWNNDAY", { Name:"DOWNNDAY", Param:{ Count:2 }, ToString:function(args) { return `最近${args[1]}日${args[0]}连跌`; } } ],
  441. ["NDAY", { Name:"NDAY", Param:{ Count:3 }, ToString:function(args) { return `最近${args[2]}日${args[0]}一直大于${args[1]}`; } } ],
  442. ["EXIST", { Name:"EXIST", Param:{ Count:2 }, ToString:function(args) { return `最近${args[1]}日存在${args[0]}`; } } ],
  443. ["EXISTR", { Name:"EXISTR", Param:{ Count:3 }, ToString:function(args) { return `从前${args[1]}日到前${args[2]}日存在${args[0]}`; } } ],
  444. ["EVERY", { Name:"EVERY", Param:{ Count:2 }, ToString:function(args) { return `最近${args[1]}日一直存在${args[0]}`; } } ],
  445. ["LAST", { Name:"LAST", Param:{ Count:3 }, ToString:function(args) { return `从前${args[1]}日到前${args[2]}日持续${args[0]}`; } } ],
  446. ["NOT", { Name:"NOT", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}取反`; } } ],
  447. ["IF", { Name:"IF", Param:{ Count:3 }, ToString:function(args) { return `如果${args[0]},返回${args[1]},否则返回${args[2]}`; } } ],
  448. ["IFF", { Name:"IFF", Param:{ Count:3 }, ToString:function(args) { return `如果${args[0]},返回${args[1]},否则返回${args[2]}`; } } ],
  449. ["IFN", { Name:"IFN", Param:{ Count:3 }, ToString:function(args) { return `如果${args[0]},返回${args[1]},否则返回${args[2]}`; } } ],
  450. ["MAX", { Name:"MAX", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}和${args[1]}的较大值`; } } ],
  451. ["MIN", { Name:"MIN", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}和${args[1]}的较小值`; } } ],
  452. ["ACOS", { Name:"ACOS", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的反余弦`; } } ],
  453. ["ASIN", { Name:"ASIN", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的反正弦`; } } ],
  454. ["ATAN", { Name:"ATAN", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的反正切`; } } ],
  455. ["COS", { Name:"COS", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的余弦`; } } ],
  456. ["SIN", { Name:"SIN", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的正弦`; } } ],
  457. ["TAN", { Name:"TAN", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的正切`; } } ],
  458. ["EXP", { Name:"EXP", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的指数`; } } ],
  459. ["LN", { Name:"LN", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的自然对数`; } } ],
  460. ["LOG", { Name:"LOG", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的对数`; } } ],
  461. ["SQRT", { Name:"SQRT", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的开方`; } } ],
  462. ["ABS", { Name:"ABS", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的绝对值`; } } ],
  463. ["POW", { Name:"POW", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}乘幂`; } } ],
  464. ["CEILING", { Name:"CEILING", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的向上舍入`; } } ],
  465. ["FLOOR", { Name:"FLOOR", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的向上舍入`; } } ],
  466. ["INTPART", { Name:"INTPART", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的整数部分`; } } ],
  467. ["BETWEEN", { Name:"BETWEEN", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}位于${args[1]}和${args[2]}之间`; } } ],
  468. ["FRACPART", { Name:"FRACPART", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的小数部分`; } } ],
  469. ["ROUND", { Name:"ROUND", Param:{ Count:1 }, ToString:function(args) { return `对${args[0]}(进行)四舍五入`; } } ],
  470. ["ROUND2", { Name:"ROUND2", Param:{ Count:2 }, ToString:function(args) { return `对${args[0]}(进行)四舍五入`; } } ],
  471. ["SIGN", { Name:"SIGN", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}的符号`; } } ],
  472. ["MOD", { Name:"MOD", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}关于${args[1]}的模`; } } ],
  473. ["RAND", { Name:"RAND", Param:{ Count:1 }, ToString:function(args) { return `随机正整数`; } } ],
  474. ["AVEDEV", { Name:"AVEDEV", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日平均绝对偏差`; } } ],
  475. ["DEVSQ", { Name:"DEVSQ", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日数据偏差平方和`; } } ],
  476. ["FORCAST", { Name:"FORCAST", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日线性回归预测值`; } } ],
  477. ["TSMA", { Name:"TSMA", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}在${args[1]}个周期内的时间序列三角移动平均`; } } ],
  478. ["SLOPE", { Name:"SLOPE", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日线性回归斜率`; } } ],
  479. ["STD", { Name:"STD", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日估算标准差`; } } ],
  480. ["STDP", { Name:"STDP", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日总体标准差`; } } ],
  481. ["STDDEV", { Name:"STDDEV", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日标准偏差`; } } ],
  482. ["VAR", { Name:"VAR", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日估算样本方差`; } } ],
  483. ["VARP", { Name:"VARP", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}的${args[1]}日总体样本方差`; } } ],
  484. ["COVAR", { Name:"COVAR", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}和${args[1]}的${args[2]}周期的协方差`; } } ],
  485. ["RELATE", { Name:"RELATE", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}和${args[1]}的${args[0]}周期的相关系数`; } } ],
  486. ["BETA", { Name:"BETA", Param:{ Count:1 }, ToString:function(args) { return `β(Beta)系数`; } } ],
  487. ["BETAEX", { Name:"BETAEX", Param:{ Count:3 }, ToString:function(args) { return `${args[0]}和${args[1]}的${args[2]}周期的相关放大系数`; } } ],
  488. ["COST", { Name:"COST", Param:{ Count:1 }, ToString:function(args) { return `获利盘为${args[0]}%的成本分布`; } } ],
  489. ["WINNER", { Name:"WINNER", Param:{ Count:1 }, ToString:function(args) { return `以${args[0]}计算的获利盘比例`; } } ],
  490. ["LWINNER", { Name:"LWINNER", Param:{ Count:2 }, ToString:function(args) { return `最近${args[0]}日那部分成本以${args[1]}价格卖出的获利盘比例`; } } ],
  491. ["PWINNER", { Name:"PWINNER", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}日前那部分成本以${args[1]}价格卖出的获利盘比例`; } } ],
  492. ["COSTEX", { Name:"COSTEX", Param:{ Count:2 }, ToString:function(args) { return `位于价格${args[0]}和${args[1]}间的成本`; } } ],
  493. ["PPART", { Name:"PPART", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}日前那部分成本占总成本的比例`; } } ],
  494. ["SAR", { Name:"SAR", Param:{ Count:3 }, ToString:function(args) { return `步长为${args[1]}极限值为${args[0]}的${args[2]}日抛物转向`; } } ],
  495. ["SARTURN", { Name:"SARTURN", Param:{ Count:3 }, ToString:function(args) { return `步长为${args[1]}极限值为${args[0]}的${args[2]}日抛物转向点`; } } ],
  496. //字符串函数
  497. ["CON2STR", { Name:"CON2STR", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}转为字符串`; } } ],
  498. ["VAR2STR", { Name:"VAR2STR", Param:{ Count:2 }, ToString:function(args) { return `${args[0]}转为字符串`; } } ],
  499. ["STR2CON", { Name:"STR2CON", Param:{ Count:1 }, ToString:function(args) { return `${args[0]}转为数字`; } } ],
  500. ["STRLEN", { Name:"STRLEN", Param:{ Count:1 }, ToString:function(args) { return `得到${args[0]}字符串长度`; } } ],
  501. ["STRCAT", { Name:"STRCAT", Param:{ Count:2 }, ToString:function(args) { return `字符串相加`; } } ],
  502. ["VARCAT", { Name:"VARCAT", Param:{ Count:2 }, ToString:function(args) { return `字符串相加`; } } ],
  503. ["STRSPACE", { Name:"STRSPACE", Param:{ Count:1 }, ToString:function(args) { return `字符串${args[0]}加一空格`; } } ],
  504. ["SUBSTR", { Name:"SUBSTR", Param:{ Count:3 }, ToString:function(args) { return `字符串${args[0]}中取一部分`; } } ],
  505. ["STRCMP", { Name:"STRCMP", Param:{ Count:2 }, ToString:function(args) { return `字符串${args[0]}和字符串${args[1]}比较`; } } ],
  506. ["FINDSTR", { Name:"FINDSTR", Param:{ Count:2 }, ToString:function(args) { return `字符串${args[0]}中查找字符串${args[1]}`; } } ],
  507. ["NAMEINCLUD", { Name:"NAMEINCLUD", Param:{ Count:1 }, ToString:function(args) { return `查找品种名称中包含${args[0]}`; } } ],
  508. ["CODELIKE", { Name:"CODELIKE", Param:{ Count:1 }, ToString:function(args) { return `查找品种名称中包含${args[0]}`; } } ],
  509. ["INBLOCK", { Name:"AVEDEV", Param:{ Count:1 }, ToString:function(args) { return `属于${args[0]}板块`; } } ],
  510. [
  511. "HHVBARS",
  512. {
  513. Name:"HHVBARS", Param:{ Count:2 },
  514. ToString:function(args)
  515. {
  516. if (args[1]==0) return `历史${args[0]}新高距今天数`;
  517. return `${args[1]}日内${args[0]}新高距今天数`;
  518. }
  519. }
  520. ],
  521. [
  522. "LLVBARS",
  523. {
  524. Name:"LLVBARS", Param:{ Count:2 },
  525. ToString:function(args)
  526. {
  527. if (args[1]==0) return `历史${args[0]}新低距今天数`;
  528. return `${args[1]}日内${args[0]}新低距今天数`;
  529. }
  530. }
  531. ]
  532. ]
  533. );
  534. this.CallFunctionExplain=function(funcName, args, node)
  535. {
  536. if (this.FUNCTION_INFO_LIST.has(funcName))
  537. {
  538. var item=this.FUNCTION_INFO_LIST.get(funcName);
  539. if (item.Param.Count!=args.length)
  540. this.ThrowUnexpectedNode(node,`函数${funcName}参数个数不正确. 需要${item.Param.Count}个参数`);
  541. return item.ToString(args);
  542. }
  543. switch(funcName)
  544. {
  545. case "CALCSTOCKINDEX":
  546. return `引用${args[0]}的${args[1]}指标第${args[2]}个输出值`;
  547. case "PEAK":
  548. case "PEAKBARS":
  549. case "ZIG":
  550. case "ZIGA":
  551. case "TROUGH":
  552. case "TROUGHBARS":
  553. return this.GetZIGExplain(funcName,args);
  554. case "FINANCE":
  555. return this.GetFinanceExplain(args);
  556. case "DYNAINFO":
  557. return this.GetDynainfoExplain(args);
  558. case 'CLOSE':
  559. case 'C':
  560. case 'VOL':
  561. case 'V':
  562. case 'OPEN':
  563. case 'O':
  564. case 'HIGH':
  565. case 'H':
  566. case 'LOW':
  567. case 'L':
  568. case 'AMOUNT':
  569. case 'AMO':
  570. return this.GetOtherSymbolExplain( {FunctionName:funcName, Args:args} ,node);
  571. //绘图函数
  572. case "PLOYLINE":
  573. return `当满足条件${args[0]}时以${args[1]}位置为顶点画折线连接`;
  574. case "DRAWLINE":
  575. return `当满足条件${args[0]}时,在${args[1]}位置画直线起点,当满足条件${args[2]}时,在${args[3]}位置画直线终点,${args[4]}表示是否延长`;
  576. case "DRAWSL":
  577. return `当满足条件${args[0]}时,在${args[1]}位置画斜线线性回归,${args[2]}斜率,${args[3]}长度,${args[4]}方向`;
  578. case "DRAWKLINE":
  579. return 'K线';
  580. case "DRAWICON":
  581. return `当满足条件${args[0]}时,在${args[1]}位置画${args[2]}号图标`;
  582. case "DRAWTEXT":
  583. return `当满足条件${args[0]}时,在${args[1]}位置书写文字`;
  584. case "DRAWTEXT_FIX":
  585. return `当满足条件${args[0]}时,在横轴${args[1]}纵轴${args[2]}位置书写文字`;
  586. case "DRAWNUMBER":
  587. return `当满足条件${args[0]}时,在${args[1]}位置书写数字`;
  588. case "DRAWNUMBER_FIX":
  589. return `当满足条件${args[0]}时,在横轴${args[1]}纵轴${args[2]}位置书写数字`;
  590. case "RGB":
  591. return `自定色[${args[0]},${args[1]},${args[2]}]`;
  592. case "DRAWBAND":
  593. return '画带状线';
  594. case "DRAWRECTREL":
  595. return "相对位置上画矩形.";
  596. case "DRAWGBK":
  597. return "填充背景";
  598. case "STICKLINE":
  599. var barType="";
  600. if (args[4]==-1) barType="虚线空心柱";
  601. else if (args[4]==0) barType="实心柱";
  602. else barType="实线空心柱";
  603. return `当满足条件${args[0]}时, 在${args[1]}和${args[2]}位置之间画柱状线,宽度为${args[3]},${barType}`;
  604. default:
  605. this.ThrowUnexpectedNode(node,`函数${funcName}不存在`);
  606. }
  607. }
  608. this.GetDynainfoExplain=function(args)
  609. {
  610. const DATA_NAME_MAP=new Map(
  611. [
  612. [3,"前收盘价"], [4,"开盘价"], [5,"最高价"], [6,"最低价"], [7,"现价"], [8,'总量'], [9,"现量"],
  613. [10,"总金额"], [11,"均价"], [12,"日涨跌"], [13,"振幅"], [14,"涨幅"], [15,"开盘时的成交金额"],
  614. [16,"前5日每分钟均量"], [17,"量比"], [18,"上涨家数"], [19,"下跌家数"]
  615. ]);
  616. var id=args[0];
  617. if (DATA_NAME_MAP.has(id)) return DATA_NAME_MAP.get(id);
  618. return `即时行情[${id}]`;
  619. }
  620. this.GetFinanceExplain=function(args)
  621. {
  622. const DATA_NAME_MAP=new Map(
  623. [
  624. [1,"总股本"], [2,"市场类型"], [3,"沪深品种类型"], [4,"沪深行业代码"], [5,"B股"], [6,"H股"], [7,"流通股本[股]"], [8,"股东人数[户]"], [9,"资产负债率%"],
  625. [10,"总资产"], [11,"流动资产"], [12,"固定资产"], [13,"无形资产"], [15,"流动负债"], [16,"少数股东权益"]
  626. ]);
  627. var id=args[0];
  628. if (DATA_NAME_MAP.has(id)) return DATA_NAME_MAP.get(id);
  629. return `财务数据[${id}]`;
  630. }
  631. this.GetZIGExplain=function(funcName,args)
  632. {
  633. var value=args[0];
  634. if (value==0) value="开盘价";
  635. else if (value==1) value="最高价";
  636. else if (value==2) value="最低价";
  637. else if (value==3) value="收盘价";
  638. switch(funcName)
  639. {
  640. case "PEAK":
  641. return `${value}的${args[1]}%之字转向的前${args[2]}个波峰值`;
  642. case "PEAKBARS":
  643. return `${value}的${args[1]}5%之字转向的前${args[2]}个波峰位置`;
  644. case "ZIG":
  645. return `${value}的${args[1]}的之字转向`;
  646. case "ZIGA":
  647. return `${value}变化${args[1]}的之字转向`;
  648. case "TROUGH":
  649. return `${value}的${args[1]}%之字转向的前${args[2]}个波谷值`;
  650. case "TROUGHBARS":
  651. return `${value}的${args[1]}%之字转向的前${args[2]}个波谷位置`;
  652. }
  653. }
  654. this.GetColorExplain=function(colorName)
  655. {
  656. const COLOR_MAP=new Map(
  657. [
  658. ['COLORBLACK','黑色'],['COLORBLUE','蓝色'],['COLORGREEN','绿色'],['COLORCYAN','青色'],['COLORRED','红色'],
  659. ['COLORMAGENTA','洋红色'],['COLORBROWN','棕色'],['COLORLIGRAY','淡灰色'],['COLORGRAY','深灰色'],['COLORLIBLUE','淡蓝色'],
  660. ['COLORLIGREEN','淡绿色'],['COLORLICYAN','淡青色'],['COLORLIRED','淡红色'],['COLORLIMAGENTA','淡洋红色'],['COLORWHITE','白色'],['COLORYELLOW','黄色']
  661. ]);
  662. if (COLOR_MAP.has(colorName)) return COLOR_MAP.get(colorName);
  663. //COLOR 自定义色
  664. //格式为COLOR+“RRGGBB”:RR、GG、BB表示红色、绿色和蓝色的分量,每种颜色的取值范围是00-FF,采用了16进制。
  665. //例如:MA5:MA(CLOSE,5),COLOR00FFFF 表示纯红色与纯绿色的混合色:COLOR808000表示淡蓝色和淡绿色的混合色。
  666. if (colorName.indexOf('COLOR')==0) return '#'+colorName.substr(5);
  667. return 'rgb(30,144,255)';
  668. }
  669. this.GetLineWidthExplain=function(lineWidth)
  670. {
  671. var width=parseInt(lineWidth.replace("LINETHICK",""));
  672. if (IFrameSplitOperator.IsPlusNumber(width)) return width;
  673. return 1;
  674. }
  675. this.SymbolPeriodExplain=function(valueName,period)
  676. {
  677. const mapStockDataName=new Map(
  678. [
  679. ['CLOSE',"收盘价"],["C","收盘价"],['VOL',"成交量"],['V',"成交量"], ['OPEN',"开盘价"], ['O',"开盘价"],
  680. ['HIGH',"最高价"],['H',"最高价"], ['LOW',"最低价"],['L',"最低价"],['AMOUNT',"成交金额"],['AMO',"成交金额"],
  681. ['VOLINSTK',"持仓量"]
  682. ]);
  683. //MIN1,MIN5,MIN15,MIN30,MIN60,DAY,WEEK,MONTH,SEASON,YEAR
  684. const mapPeriodName=new Map(
  685. [
  686. ["MIN1","1分钟"], ["MIN5", "5分钟"], ["MIN15", "15分钟"], ["MIN30","30分钟"],["MIN60","60分钟"],
  687. ["DAY","日"],["WEEK","周"], ["MONTH", "月"], ['SEASON',"季"], ["YEAR", "年"],["WEEK2","双周"], ["HALFYEAR", "半年"]
  688. ]);
  689. var dataName=valueName;
  690. if (mapStockDataName.has(valueName)) dataName=mapStockDataName.get(valueName);
  691. var periodName=period;
  692. if (mapPeriodName.has(period)) periodName=mapPeriodName.get(period);
  693. return `${dataName}[取${periodName}数据]`;
  694. }
  695. this.GetOtherSymbolExplain=function(obj, node)
  696. {
  697. const mapStockDataName=new Map(
  698. [
  699. ['CLOSE',"收盘价"],["C","收盘价"],['VOL',"成交量"],['V',"成交量"], ['OPEN',"开盘价"], ['O',"开盘价"],
  700. ['HIGH',"最高价"],['H',"最高价"], ['LOW',"最低价"],['L',"最低价"],['AMOUNT',"成交金额"],['AMO',"成交金额"],
  701. ['VOLINSTK',"持仓量"]
  702. ]);
  703. if (obj.FunctionName)
  704. {
  705. var args=obj.Args;
  706. var dataName=mapStockDataName.get(obj.FunctionName);
  707. return `[${args[0]}]${dataName}`;
  708. }
  709. else if (obj.Literal)
  710. {
  711. var value=obj.Literal.toUpperCase();
  712. var args=value.split("$");
  713. if (!mapStockDataName.has(args[1])) return "";
  714. var symbol=args[0];
  715. var dataName=mapStockDataName.get(args[1]);
  716. return `[${symbol}]${dataName}`;
  717. }
  718. }
  719. this.IsDrawFunction=function(name)
  720. {
  721. let setFunctionName=new Set(
  722. [
  723. "STICKLINE","DRAWTEXT",'SUPERDRAWTEXT','DRAWLINE','DRAWBAND','DRAWKLINE','DRAWKLINE_IF','PLOYLINE',
  724. 'POLYLINE','DRAWNUMBER',"DRAWNUMBER_FIX",'DRAWICON','DRAWCHANNEL','PARTLINE','DRAWTEXT_FIX','DRAWGBK','DRAWTEXT_LINE','DRAWRECTREL',"DRAWTEXTABS",
  725. 'DRAWOVERLAYLINE',"FILLRGN", "FILLRGN2","FILLTOPRGN", "FILLBOTTOMRGN", "FILLVERTICALRGN","FLOATRGN","DRAWSL", "DRAWGBK2"
  726. ]);
  727. if (setFunctionName.has(name)) return true;
  728. return false;
  729. }
  730. //赋值
  731. this.VisitAssignmentExpression=function(node)
  732. {
  733. let left=node.Left;
  734. if (left.Type!=Syntax.Identifier) this.ThrowUnexpectedNode(node);
  735. let varName=left.Name;
  736. let right=node.Right;
  737. let value=null;
  738. if (right.Type==Syntax.BinaryExpression || right.Type==Syntax.LogicalExpression)
  739. value=this.VisitBinaryExpression(right);
  740. else if (right.Type==Syntax.CallExpression)
  741. value=this.VisitCallExpression(right);
  742. else if (right.Type==Syntax.Literal)
  743. {
  744. value=right.Value;
  745. if (IFrameSplitOperator.IsString(value) && right.Value.indexOf("$")>0)
  746. value=this.GetOtherSymbolExplain({ Literal:value }, node);
  747. }
  748. else if (right.Type==Syntax.Identifier) //右值是变量
  749. value=this.ReadVariable(right.Name,right);
  750. else if (right.Type==Syntax.MemberExpression)
  751. value=this.ReadMemberVariable(right);
  752. else if (right.Type==Syntax.UnaryExpression)
  753. {
  754. if (right.Operator=='-')
  755. {
  756. var tempValue=this.GetNodeValue(right.Argument);
  757. value='-'+tempValue;
  758. }
  759. else
  760. {
  761. value=right.Argument.Value;
  762. }
  763. }
  764. JSConsole.Complier.Log('[JSExplainer::VisitAssignmentExpression]' , varName, ' = ',value);
  765. this.VarTable.set(varName,value);
  766. }
  767. //逻辑运算
  768. this.VisitBinaryExpression=function(node)
  769. {
  770. let stack=[];
  771. stack.push(node);
  772. let temp=null;
  773. while(stack.length!=0)
  774. {
  775. temp=stack[stack.length-1];
  776. if (temp.Left && node!=temp.Left && node!=temp.Right)
  777. {
  778. stack.push(temp.Left);
  779. }
  780. else if (temp.Right && node!=temp.Right)
  781. {
  782. stack.push(temp.Right);
  783. }
  784. else
  785. {
  786. let value=stack.pop();
  787. if (value.Type==Syntax.BinaryExpression) //只遍历操作符就可以
  788. {
  789. let leftValue=this.GetNodeValue(value.Left);
  790. let rightValue=this.GetNodeValue(value.Right);
  791. JSConsole.Complier.Log('[JSExplainer::VisitBinaryExpression] BinaryExpression',value , leftValue, rightValue);
  792. value.Out=null; //保存中间值
  793. value.Out=`(${leftValue} ${value.Operator} ${rightValue})`;
  794. if (leftValue=="收盘价" && rightValue=="开盘价")
  795. {
  796. if (value.Operator==">") value.Out='(收阳线)';
  797. else if (value.Operator=="<") value.Out='(收阴线)';
  798. else if (value.Operator=="=") value.Out='(平盘)';
  799. }
  800. else if (leftValue=="开盘价" && rightValue=="收盘价")
  801. {
  802. if (value.Operator=="<") value.Out='(收阳线)';
  803. else if (value.Operator==">") value.Out='(收阴线)';
  804. else if (value.Operator=="=") value.Out='(平盘)';
  805. }
  806. JSConsole.Complier.Log('[JSExplainer::VisitBinaryExpression] BinaryExpression',value);
  807. }
  808. else if (value.Type==Syntax.LogicalExpression)
  809. {
  810. let leftValue=this.GetNodeValue(value.Left);
  811. let rightValue=this.GetNodeValue(value.Right);
  812. JSConsole.Complier.Log('[JSExecute::VisitBinaryExpression] LogicalExpression',value , leftValue, rightValue);
  813. value.Out=null; //保存中间值
  814. switch(value.Operator)
  815. {
  816. case '&&':
  817. case 'AND':
  818. value.Out=`(${leftValue} 并且 ${rightValue})`;
  819. break;
  820. case '||':
  821. case 'OR':
  822. value.Out=`(${leftValue} 或者 ${rightValue})`;
  823. break;
  824. }
  825. JSConsole.Complier.Log('[JSExplainer::VisitBinaryExpression] LogicalExpression',value);
  826. }
  827. node=temp;
  828. }
  829. }
  830. return node.Out;
  831. }
  832. this.GetNodeValue=function(node)
  833. {
  834. switch(node.Type)
  835. {
  836. case Syntax.Literal: //数字
  837. return node.Value;
  838. case Syntax.UnaryExpression:
  839. if (node.Operator=='-')
  840. {
  841. let value=this.GetNodeValue(node.Argument);
  842. return '-'+value;
  843. }
  844. return node.Argument.Value;
  845. case Syntax.Identifier:
  846. let value=this.ReadVariable(node.Name,node);
  847. return value;
  848. case Syntax.BinaryExpression:
  849. case Syntax.LogicalExpression:
  850. return node.Out;
  851. case Syntax.CallExpression:
  852. return this.VisitCallExpression(node);
  853. default:
  854. this.ThrowUnexpectedNode(node);
  855. }
  856. }
  857. //读取变量
  858. this.ReadVariable=function(name,node)
  859. {
  860. if (this.ConstVarTable.has(name))
  861. {
  862. let data=this.ConstVarTable.get(name);
  863. return data;
  864. }
  865. if (g_JSComplierResource.IsCustomVariant(name)) return this.ReadCustomVariant(name,node); //读取自定义变量
  866. if (this.VarTable.has(name)) return this.VarTable.get(name);
  867. if (name.indexOf('#')>0)
  868. {
  869. var aryPeriod=name.split('#');
  870. return this.SymbolPeriodExplain(aryPeriod[0],aryPeriod[1]);
  871. }
  872. this.ThrowUnexpectedNode(node, '变量'+name+'不存在');
  873. return name;
  874. }
  875. this.ThrowUnexpectedNode=function(node,message)
  876. {
  877. let marker=node.Marker;
  878. let msg=message || "执行异常";
  879. return this.ErrorHandler.ThrowError(marker.Index,marker.Line,marker.Column,msg);
  880. }
  881. this.ThrowError=function()
  882. {
  883. }
  884. }
  885. JSComplier.Explain=function(code,option, errorCallback)
  886. {
  887. //异步调用
  888. //var asyncExecute= async function() es5不能执行 去掉异步
  889. var asyncExplain= function()
  890. {
  891. try
  892. {
  893. JSConsole.Complier.Log('[JSComplier.Explain]',code,option);
  894. JSConsole.Complier.Log('[JSComplier.Explain] parser .....');
  895. let parser=new JSParser(code);
  896. parser.Initialize();
  897. let program=parser.ParseScript();
  898. let ast=program;
  899. JSConsole.Complier.Log('[JSComplier.Explain] parser finish.', ast);
  900. JSConsole.Complier.Log('[JSComplier.Explain] explain .....');
  901. let execute=new JSExplainer(ast,option);
  902. execute.ErrorCallback=errorCallback; //执行错误回调
  903. execute.JobList=parser.Node.GetDataJobList();
  904. execute.JobList.push({ID:JS_EXECUTE_JOB_ID.JOB_RUN_SCRIPT});
  905. let result=execute.Run();
  906. }catch(error)
  907. {
  908. JSConsole.Complier.Log(error);
  909. if (errorCallback) errorCallback(error, option.CallbackParam);
  910. }
  911. }
  912. asyncExplain();
  913. JSConsole.Complier.Log('[JSComplier.Explain] async explain.');
  914. }