umychart.explainer.wechat.js 56 KB

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