calc_util.js 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. /**
  2. * Created by Tony on 2017/6/21.
  3. * Modified by CSL, 2017-08-01
  4. */
  5. let executeObj = {
  6. calculation : null,
  7. ration: null,
  8. at: function(ID) {
  9. let me = executeObj,
  10. rst = 0;
  11. rst = me.calculation.compileAssistantObj[ID].unitFee;
  12. rst = parseFloat(rst);
  13. return rst;
  14. },
  15. base: function(calcBaseName) {
  16. let me = executeObj, rst = 0,
  17. base = getRationCalcBase(calcBaseName);
  18. if (base != null) {
  19. let price = 0, tmpSum = 0;
  20. for (let glj of me.ration.data.gljList) {
  21. if (base.gljTypes.indexOf(glj.type) >= 0) {
  22. if (base.calcType == baseCalc){ price = glj["basePrice"];}
  23. else if (base.calcType == adjustCalc){price = glj["adjustPrice"];}
  24. else if (base.calcType == budgetCalc){price = glj["marketPrice"];}
  25. else if (base.calcType == diffCalc){price = glj["marketPrice"] - glj["adjustPrice"];};
  26. };
  27. tmpSum = tmpSum + glj["quantity"] * price;
  28. glj = null;
  29. };
  30. rst = tmpSum;
  31. };
  32. return rst;
  33. }
  34. /* factor: function(factorID) {
  35. let me = executeObj;
  36. let rst = 7.77;
  37. return rst;
  38. }*/
  39. };
  40. class Calculation {
  41. /* init(template, feeRates){
  42. let me = this;
  43. me.template = template;
  44. me.feeRates = feeRates;
  45. me.hasCompiled = false;
  46. };*/
  47. compile(template, feeRates, labourCoes){
  48. let me = this;
  49. me.hasCompiled = false;
  50. me.errs = [];
  51. let private_extract_ID = function(str, idx){
  52. let rst = '', lBracket = 0, rBracket = 0, firstIdx = idx, lastIdx = 0;
  53. for (let i = idx; i < str.length; i++) {
  54. if (str[i] === '(') {
  55. lBracket++;
  56. if (lBracket == 1) firstIdx = i + 1;
  57. }
  58. if (str[i] === ')') {
  59. rBracket++;
  60. if (lBracket == rBracket) {
  61. lastIdx = i - 1;
  62. if (lastIdx > firstIdx) {
  63. if (str[firstIdx] === "'") firstIdx++;
  64. if (str[lastIdx] !== "'") lastIdx++;
  65. if (lastIdx > firstIdx) {
  66. rst = str.slice(firstIdx, lastIdx);
  67. }
  68. }
  69. break;
  70. }
  71. }
  72. }
  73. return rst;
  74. };
  75. let private_parse_ref = function(item, itemIdx){
  76. let idx = item.expression.indexOf('@(', 0);
  77. while (idx >= 0) {
  78. let ID = private_extract_ID(item.expression, idx);
  79. if (ID.length > 0) {
  80. let subItem = me.compileAssistantObj[ID];
  81. if (subItem) {
  82. if (subItem.ID !== item.ID) {
  83. private_parse_ref(subItem, me.compileAssistantObj[ID + "_idx"]);
  84. } else {
  85. me.errs.push("There exists the self refer ID: " + ID);
  86. }
  87. } else {
  88. me.errs.push("There exists the invalid ID by which could not find the item: " + ID);
  89. console.log('invalid ID: ' + ID);
  90. }
  91. }
  92. idx = item.expression.indexOf('@(', idx + ID.length + 3);
  93. }
  94. if (template.compiledSeq.indexOf(itemIdx) < 0) {
  95. template.compiledSeq.push(itemIdx);
  96. }
  97. };
  98. let private_setup_seq = function(item, itemIdx){
  99. if (template.compiledSeq.indexOf(itemIdx) < 0) {
  100. private_parse_ref(item, itemIdx);
  101. }
  102. };
  103. let private_compile_items = function() {
  104. for (let idx of template.compiledSeq) {
  105. let item = template.calcItems[idx];
  106. item.compiledExpr = item.expression.split('@(').join('$CE.at(');
  107. item.compiledExpr = item.compiledExpr.split('base(').join('$CE.base(');
  108. //item.compiledExpr = item.compiledExpr.split('rate(').join('$CE.rate(');
  109. //item.compiledExpr = item.compiledExpr.split('factor(').join('$CE.factor(');
  110. if (item.labourCoeID){
  111. let lc = me.compiledLabourCoes["LabourCoe_" + item.labourCoeID].coe;
  112. item.dispExpr = item.dispExpr.replace(/L/gi, lc.toString());
  113. item.compiledExpr = item.compiledExpr.replace(/L/gi, lc.toString());
  114. }
  115. }
  116. };
  117. let private_compile_feeFile = function() {
  118. if (feeRates) {
  119. me.compiledFeeRate = {};
  120. for (let rate of feeRates) {
  121. me.compiledFeeRate["feeRate_" + rate.ID] = rate;
  122. }
  123. }
  124. };
  125. let private_compile_labourCoeFile = function() {
  126. if (labourCoes) {
  127. me.compiledLabourCoes = {};
  128. for (let coe of labourCoes) {
  129. me.compiledLabourCoes["LabourCoe_" + coe.ID] = coe;
  130. }
  131. }
  132. };
  133. if (template && template.calcItems && template.calcItems.length > 0) {
  134. template.compiledSeq = [];
  135. me.compileAssistantObj = {};
  136. //1. first round -> prepare
  137. private_compile_feeFile();
  138. private_compile_labourCoeFile();
  139. for (let i = 0; i < template.calcItems.length; i++) {
  140. let item = template.calcItems[i];
  141. me.compileAssistantObj[item.ID] = item;
  142. me.compileAssistantObj[item.ID + "_idx"] = i;
  143. }
  144. //2. second round -> go!
  145. for (let i = 0; i < template.calcItems.length; i++) {
  146. private_setup_seq(template.calcItems[i], i);
  147. }
  148. if (me.errs.length == 0) {
  149. private_compile_items();
  150. if (me.errs.length == 0) me.hasCompiled = true;
  151. } else {
  152. console.log('errors: ' + me.errs.toString());
  153. }
  154. }
  155. };
  156. calculate($RATION){ // 参数$RATION也可以是清单
  157. let me = this;
  158. if ($RATION && me.hasCompiled) {
  159. let $CE = executeObj;
  160. $CE.ration = $RATION;
  161. $CE.calculation = me;
  162. if (!$RATION.fees) {
  163. $RATION.fees = [];
  164. $RATION.feesIndex = {};
  165. };
  166. //let template = getCalcTemplate($RATION.CalcTemplateID); // AAAAAA 通过定额获取计算规则
  167. let template = calcTemplates[0];
  168. for (let idx of template.compiledSeq) {
  169. let calcItem = template.calcItems[idx];
  170. let feeRate = 100; // 100%
  171. // 下面三项用于界面显示。
  172. if (calcItem.feeRateID) {
  173. feeRate = me.compiledFeeRate["feeRate_" + calcItem.feeRateID].rate;
  174. calcItem.feeRate = feeRate;
  175. };
  176. calcItem.unitFee = eval(calcItem.compiledExpr) * feeRate * 0.01; // 如果eval()对清单树有影响,就换成小麦的Expression对象再试
  177. calcItem.totalFee = calcItem.unitFee * 3; // AAAAA 测试值,因目前定额无数量(保存不上) $RATION.data.Quantity
  178. // 费用同步到定额
  179. // 引入小麦的字段检测后,快速切换定额出现计算卡顿现象,过多的循环造成。这里把她的代码拆出来,减少微循环。
  180. if (!$RATION.feesIndex[calcItem.fieldName]){
  181. let fee = {
  182. 'fieldName': calcItem.fieldName,
  183. 'unitFee': calcItem.unitFee,
  184. 'totalFee': calcItem.totalFee,
  185. 'tenderUnitFee': 0,
  186. 'tenderTotalFee': 0
  187. };
  188. $RATION.fees.push(fee);
  189. $RATION.feesIndex[calcItem.fieldName] = fee;
  190. }
  191. else{
  192. $RATION.feesIndex[calcItem.fieldName].unitFee = calcItem.unitFee;
  193. $RATION.feesIndex[calcItem.fieldName].totalFee = calcItem.totalFee;
  194. }
  195. }
  196. }
  197. }
  198. }
  199. //module.exports = new calculation();