calc_base.js 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050
  1. /**
  2. * Created by Zhong on 2017/11/28.
  3. */
  4. let cbTools = {
  5. isDef: function (v) {
  6. return v !== undefined && v !== null;
  7. },
  8. isUnDef: function (v) {
  9. return v === undefined || v === null;
  10. },
  11. isNum: function (v) {
  12. return this.isDef(v) && !isNaN(v) && v !== Infinity;
  13. },
  14. isFlag: function (v) {
  15. return this.isDef(v) && this.isDef(v.flagsIndex) && this.isDef(v.flagsIndex.fixed);
  16. },
  17. returnV: function (v, r) {
  18. if(this.isDef(v)){
  19. return v;
  20. }
  21. return r;
  22. },
  23. findBill: function (fixedFlag) {
  24. return this.isDef(calcBase.fixedBills[fixedFlag]) ? calcBase.fixedBills[fixedFlag]['bill'] : null;
  25. },
  26. findNodeByFlag: function (fixedFlag) {
  27. let bills = this.findBill(fixedFlag);
  28. if (!bills) {
  29. return null;
  30. }
  31. return this.getNodeByID(bills.ID);
  32. },
  33. //通过行获取根节点清单
  34. getBillByRow: function (items, row) {
  35. if(cbTools.isDef(items[row]) &&
  36. cbTools.isUnDef(items[row]['parent'])&&
  37. cbTools.isDef(items[row]['sourceType']) &&
  38. items[row]['sourceType'] === calcBase.project.Bills.getSourceType()){
  39. return items[row];
  40. }
  41. return null;
  42. },
  43. //通过ID获取节点行
  44. getRowByID: function (items, ID) {
  45. for(let i = 0, len = items.length; i < len; i++){
  46. if(items[i]['data']['ID'] == ID){
  47. return i + 1;
  48. }
  49. }
  50. return null;
  51. },
  52. //通过ID获取节点
  53. getNodeByID: function (ID) {
  54. return this.isDef(calcBase.project.mainTree.nodes['id_' + ID]) ? calcBase.project.mainTree.nodes['id_' + ID] : null;
  55. },
  56. //获取该节点所有父节点
  57. getParents: function (node) {
  58. let rst = [];
  59. rst.push(node);
  60. rParent(node);
  61. return rst;
  62. function rParent(node){
  63. if(cbTools.isDef(node.parent)){
  64. rst.push(node.parent);
  65. rParent(node.parent);
  66. }
  67. }
  68. },
  69. //获取所有节点的ID
  70. getNodeIDs: function (nodes) {
  71. let rst = [];
  72. for(let i = 0, len = nodes.length; i < len; i++){
  73. if(this.isDef(nodes[i]['data']['ID'])){
  74. rst.push(nodes[i]['data']['ID']);
  75. }
  76. }
  77. return rst;
  78. },
  79. //根据公式获取相关的节点
  80. getNodesByExp: function (node, formulaNodesArr) {
  81. let exp = node.data.calcBase;
  82. let rst = [], ids = [];
  83. if(this.isUnDef(exp) || exp === ''){
  84. return rst;
  85. }
  86. //获取表达式中的基数和行引用
  87. let figureF = cbParser.getFigureF(cbParser.getFigure(exp), cbParser.getUID(cbParser.getFIDArr(exp)));
  88. //首先提取出多处引用的进行排序
  89. for(let i = 0, len = figureF.length; i < len; i++){
  90. let figure = figureF[i];
  91. if(figure.type === 'base' && cbTools.isDef(calcBase.baseFigures[figure.value])){
  92. let bill = this.isDef(calcBase.baseFigures[figure.value]['fixedBill']) ? calcBase.baseFigures[figure.value]['fixedBill']['bill'] : null;
  93. let figureMultiRef = calcBase.baseFigures[figure.value]['multiRef'];
  94. if(this.isDef(figureMultiRef)){
  95. let findChildNodes = [];
  96. for(let flag of figureMultiRef){
  97. let refNode = this.findBill(flag) ? this.getNodeByID(this.findBill(flag).ID) : null;
  98. if(refNode){
  99. findChildNodes.push(refNode);
  100. }
  101. }
  102. let childrenNodes = calcTools.getChildrenFormulaNodes(node, formulaNodesArr, findChildNodes);
  103. for(let cNode of childrenNodes){
  104. ids.push(cNode.data.ID);
  105. }
  106. rst = rst.concat(childrenNodes);
  107. }
  108. else if(this.isDef(bill) && ids.indexOf(bill.ID) === -1){
  109. let node = this.getNodeByID(bill.ID);
  110. if(this.isDef(node)){
  111. ids.push(node.data.ID);
  112. rst.push(node);
  113. }
  114. }
  115. }
  116. else if(figure.type === 'id'){
  117. let node = this.getNodeByID(figure.value);
  118. if(this.isDef(node) && ids.indexOf(node.data.ID) === -1){
  119. ids.push(node.data.ID);
  120. rst.push(node);
  121. }
  122. }
  123. }
  124. return rst;
  125. },
  126. //需要用到计算基数的时候,先找出所有的固定清单,避免每个基数都要去遍历寻找清单
  127. setFixedBills: function (project, billsObj, fixedFlag) {
  128. let bills = project.Bills.datas;
  129. for(let i = 0, len = bills.length; i < len; i++){
  130. if(this.isDef(bills[i].flagsIndex.fixed)){
  131. for(let flag in fixedFlag){
  132. if(fixedFlag[flag] === bills[i].flagsIndex.fixed.flag){
  133. billsObj[fixedFlag[flag]] = Object.create(null);
  134. billsObj[fixedFlag[flag]]['base'] = Object.create(null);
  135. billsObj[fixedFlag[flag]]['bill'] = bills[i];
  136. }
  137. }
  138. }
  139. }
  140. },
  141. //清单基数设置所属固定清单属性
  142. setBaseBills: function (baseFigure, fixedBills) {
  143. for(let i in baseFigure){
  144. let calcBase = baseFigure[i];
  145. calcBase.fixedBill = null;
  146. if(cbTools.isDef(calcBase.fixedFlag) && cbTools.isDef(fixedBills[calcBase.fixedFlag])){
  147. fixedBills[calcBase.fixedFlag]['base'][i] = calcBase;
  148. calcBase.fixedBill = fixedBills[calcBase.fixedFlag];
  149. }
  150. }
  151. },
  152. //生成清单基数计算分类模板
  153. setBaseFigureClass: function (baseFigures, mapObj) {
  154. for(let figureClass in figureClassTemplate){
  155. mapObj[figureClass] = Object.create(null);
  156. }
  157. let needFixedBillsClass = ['FBFX', 'CXSM', 'QTXM', 'GF', 'SJ'];
  158. //不需要关联节点的、但是下挂在固定清单分类下的基数
  159. let noneFixedBillsFigures = ['JZMJ'];
  160. //安全文明施工专项费用只有税金和工程造价能用
  161. for(let figure in baseFigures){
  162. if(!noneFixedBillsFigures.includes(baseFigures[figure]['base'])){
  163. //过滤相关清单固定行不存在的
  164. if(needFixedBillsClass.includes(baseFigures[figure]['class']) && !baseFigures[figure]['fixedBill']){
  165. continue;
  166. }
  167. }
  168. for(let figureClass in figureClassTemplate){
  169. let figureClassFilter = figureClassTemplate[figureClass]['filter'];
  170. if(!figureClassFilter.includes(baseFigures[figure]['base'])){
  171. mapObj[figureClass][figure] = baseFigures[figure];
  172. }
  173. }
  174. }
  175. },
  176. getFigure: function (node) {
  177. return calcBase.baseFigures;
  178. },
  179. getBaseBill: function (node) {
  180. let calcBase = projectObj.project.calcBase;
  181. let parent = node.parent;
  182. if(this.isFlag(node.data) && (node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.SUB_ENGINERRING
  183. || node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.CONSTRUCTION_TECH)){
  184. return node;
  185. }
  186. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.CONSTRUCTION_ORGANIZATION){
  187. return node;
  188. }
  189. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.OTHER){
  190. return node;
  191. }
  192. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.CHARGE){
  193. return node;
  194. }
  195. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.TAX){
  196. return node;
  197. }
  198. else {
  199. if(!parent){
  200. return node;
  201. }
  202. else {
  203. return this.getBaseBill(parent);
  204. }
  205. }
  206. },
  207. //获取清单(有基数计算)引用了的其他清单,(循环引用栈中的一块)
  208. getStackBlock: function (billID) {
  209. let tempBases = [], block = [];//存引用的清单ID
  210. let node = getBill(billID);
  211. if(!node){
  212. return tempBases;
  213. }
  214. else {
  215. //获取基数和行引用
  216. getBase(node);
  217. let bases = Array.from(new Set(tempBases));
  218. //根据基数和行引用获取清单ID
  219. for(let i = 0, len = bases.length; i < len; i++){
  220. //基数是跟清单直接关联的
  221. if(bases[i]['type'] === 'base' && cbTools.isDef(calcBase.baseFigures[bases[i]['value']])){
  222. let figureMultiRef= calcBase.baseFigures[bases[i]['value']]['multiRef'];
  223. let cycleCalcRef = calcBase.baseFigures[bases[i]['value']]['cycleCalcRef'];
  224. //重构后:
  225. if(cbTools.isDef(figureMultiRef)){
  226. if(cbTools.isDef(cycleCalcRef)){
  227. figureMultiRef = cycleCalcRef;
  228. }
  229. for(let flag of figureMultiRef){
  230. let bills = cbTools.findBill(flag);
  231. if(cbTools.isDef(bills)){
  232. block.push(bills.ID);
  233. }
  234. }
  235. } else if(cbTools.isDef(calcBase.baseFigures[bases[i]['value']]['fixedBill'])){
  236. block.push(calcBase.baseFigures[bases[i]['value']]['fixedBill']['bill']['ID']);
  237. }
  238. } else if(bases[i]['type'] === 'id'){
  239. let node = cbTools.getNodeByID(bases[i]['value']);
  240. if(cbTools.isDef(node)){
  241. block.push(node.data.ID);
  242. }
  243. }
  244. }
  245. return Array.from(new Set(block));
  246. }
  247. function getBase(node){
  248. if(node && node.children.length === 0){
  249. if(cbTools.isDef(node.data.calcBase) && node.data.calcBase !== ''){
  250. let figureF = cbParser.getFigureF(cbParser.getFigure(node.data.calcBase), cbParser.getUID(cbParser.getFIDArr(node.data.calcBase)));
  251. tempBases = tempBases.concat(figureF);
  252. }
  253. } else if(node && node.children.length > 0) {
  254. for(let i = 0, len = node.children.length; i < len; i++){
  255. getBase(node.children[i]);
  256. }
  257. }
  258. }
  259. function getBill(ID){
  260. let nodes = calcBase.project.mainTree.nodes;
  261. let node = nodes['id_' + ID];
  262. if(cbTools.isDef(node) && node.sourceType === calcBase.project.Bills.getSourceType()){
  263. return node;
  264. }
  265. return null;
  266. }
  267. },
  268. // 获取全部有公式的树节点清单。 CSL, 2018-01-05
  269. getFormulaNodes: function (needOrder = false) {
  270. // 给公式结点清单换照引用计算顺序排序。
  271. function orderFormulaNodes (nodesArr) {
  272. let orderArr = [];
  273. function recursionNode(nodes) {
  274. for (let node of nodes){
  275. if (orderArr.includes(node)) continue; // 已排过序的节点则跳过
  276. if (node.data.calcBase){
  277. let subNodes = cbTools.getNodesByExp(node, nodesArr);
  278. recursionNode(subNodes);
  279. };
  280. if (nodesArr.includes(node) && !orderArr.includes(node)) orderArr.push(node);
  281. };
  282. }
  283. recursionNode(nodesArr);
  284. return orderArr;
  285. };
  286. let nodes = [];
  287. for (let node of projectObj.project.mainTree.items){
  288. if (node.sourceType == ModuleNames.bills && node.data.calcBase && node.data.calcBase != '')
  289. nodes.push(node);
  290. };
  291. if (needOrder && nodes.length >= 2) return orderFormulaNodes(nodes)
  292. else return nodes;
  293. },
  294. // 刷新全部行引用的公式清单。 CSL, 2018-01-05
  295. refreshFormulaNodes: function () {
  296. try {
  297. let nodes = this.getFormulaNodes();
  298. if (nodes.length > 0) projectObj.mainController.refreshTreeNode(nodes);
  299. } catch (err) {
  300. alert('公式引用行号显示刷新失败:' + err.message);
  301. }
  302. },
  303. // 判断结点是否被其它结点的表达式引用。
  304. isUsedByFormula: function(node){
  305. let nodes = this.getFormulaNodes();
  306. if (nodes.length == 0) return false;
  307. let sID = '@' + node.data.ID;
  308. for (let node of nodes){
  309. if (node.data.calcBase.hasSubStr(sID)) return true;
  310. };
  311. },
  312. //获取清单节点的金额
  313. //@param {Number}fixedFlag(清单固定行类别) {String}feeField(外层金额字段: common) {String}subFeeField(子金额字段: totalFee)
  314. //@return {Number}
  315. getBillsFee: function(fixedFlag, feeField, subFeeField) {
  316. //固定清单类别与清单数据、关联基数的映射
  317. let fixedBills = calcBase.fixedBills[fixedFlag];
  318. if (this.isUnDef(fixedBills)) {
  319. return 0;
  320. }
  321. let bills = fixedBills.bill;
  322. if (this.isUnDef(bills)) {
  323. return 0;
  324. }
  325. if (this.isUnDef(bills.feesIndex) || _.isEmpty(bills.feesIndex)) {
  326. return 0;
  327. }
  328. return this.isDef(bills.feesIndex[feeField]) && this.isDef(bills.feesIndex[feeField][subFeeField]) ? bills.feesIndex[feeField][subFeeField] : 0;
  329. },
  330. //获取扣除固定项后的金额,扣除其节点后重新汇总
  331. //@param {Number}fixedFlag(基数取值固定行类别) {Array}deductFlags(扣除的固定类别组) {String}fullFeeField(完整的取费字段: 'rationCommon.totalFee')
  332. //@return {Number}
  333. getFeeWithDeduction: function (fixedFlag, deductFlags, fullFeeField) {
  334. let baseNode = this.findNodeByFlag(fixedFlag);
  335. if (!baseNode) {
  336. return 0;
  337. }
  338. //要扣除的节点
  339. let deductNodes = [];
  340. for (let deFlag of deductFlags) {
  341. let node = this.findNodeByFlag(deFlag);
  342. if (node) {
  343. deductNodes.push(node);
  344. }
  345. }
  346. return projectObj.project.calcProgram.getTotalFee([baseNode], deductNodes, fullFeeField);
  347. },
  348. //获取累进办法计算的金额
  349. //@param {Number}baseFee(相关基数金额) {String}name(使用累进计算的基数名称)
  350. //@return {Number}
  351. getProgressiveFee: function (baseFee, name) {
  352. let progressiveData = projectInfoObj.projectInfo.property.progressiveInterval;
  353. if (!progressiveData) {
  354. throw '该项目不存在累进区间数据';
  355. }
  356. let matchData = _.find(progressiveData, function (data) {
  357. //根据基数名称匹配累进库数据,标准化以免(())等不同导致不匹配
  358. return cbAnalyzer.standar(data.name) === cbAnalyzer.standar(name);
  359. });
  360. if (!matchData) {
  361. return 0;//--temp
  362. //throw `计算基数{${name}}不存在累进区间数据`;
  363. }
  364. let progression = matchData.progression;
  365. //获取区间中的最小值(0, 10] = 0
  366. function getMin(intervalStr) {
  367. let str = cbAnalyzer.standar(intervalStr);
  368. let match = /\((\d+)?/g.exec(str);
  369. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null; //后台数据单位为万元,这里转为为元
  370. }
  371. //获取区间中的最大值(0, 10] = 10
  372. function getMax(intervalStr) {
  373. let str = cbAnalyzer.standar(intervalStr);
  374. let match = /[\,,,](\d+)?/g.exec(str);
  375. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null
  376. }
  377. //将累进区间进行排序
  378. progression.sort(function (a, b) {
  379. let aV = getMin(a.interval),
  380. bV = getMin(b.interval);
  381. if (aV > bV) {
  382. return 1;
  383. } else if (aV < bV) {
  384. return -1;
  385. }
  386. return 0;
  387. });
  388. //累进计算
  389. let fee = 0;
  390. //找到所在区间
  391. let within = _.find(progression, function (data) {
  392. let min = getMin(data.interval),
  393. max = getMax(data.interval);
  394. return min !== null && baseFee > min && (max !== null && baseFee <= max || max === null);
  395. });
  396. if (!within) {
  397. return 0;
  398. }
  399. //累进之前的区间
  400. for (let i = 0; i < progression.indexOf(within); i++) {
  401. let perData = progression[i],
  402. min = getMin(perData.interval),
  403. max = getMax(perData.interval);
  404. if (min !== null && max !== null) {
  405. fee += (max - min) * perData.feeRate * 0.01;
  406. }
  407. }
  408. //累进所在区间
  409. let min = getMin(within.interval);
  410. if (min !== null) {
  411. fee += (baseFee - min) * within.feeRate * 0.01;
  412. }
  413. return fee.toDecimal(decimalObj.bills.totalPrice);
  414. }
  415. };
  416. let baseFigureTemplate = {
  417. //{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  418. //取清单固定类别是“建筑安装工程”的定额建安费,但要扣除清单固定类别是“设备购置费”、及“专项费用”的定额建安费
  419. 'DEJZAZGCFBHSBZX': function (tender) {
  420. let fullFeeField = tender ? 'rationCommon.tenderTotalFee' : 'rationCommon.totalFee',
  421. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE, fixedFlag.SPECIAL_COST];
  422. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  423. },
  424. //{定额建筑安装工程(其中定额设备购置费按 40%计)} (定额建筑安装工程设备四十)
  425. //扣除设备购置费,再加上设备购置费的40%,扣除汇总算法不四舍五入,相当于汇总当中定额设备购置费就按照了40%计
  426. 'DEJZAZGCSBSS': function (tender) {
  427. let feeField = 'rationCommon',
  428. subFeeField = tender ? 'tenderTotalFee' : 'totalFee',
  429. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  430. //建安费扣除定额设备购置费
  431. let afterDeductFee = cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, `${feeField}.${subFeeField}`);
  432. //定额设备购置费
  433. let equipmentAcFee = cbTools.getBillsFee(deductFlags[0], feeField, subFeeField);
  434. return (afterDeductFee + equipmentAcFee * 0.4).toDecimal(decimalObj.bills.totalPrice);
  435. },
  436. //{建筑安装工程费(不含安全生产费)}
  437. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“安全生产费”的金额
  438. 'JZAZGCFBHSC': function (tender) {
  439. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  440. deductFlags = [fixedFlag.SAFE_COST];
  441. //建安费扣除安全生产费
  442. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  443. },
  444. //{建筑安装工程费(不含设备费)}
  445. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“设备购置费”的金额
  446. 'JZAZGCFBHSB': function (tender) {
  447. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  448. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  449. //建安费扣除设备费
  450. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  451. },
  452. //{建筑安装工程费}
  453. // 取清单固定类别是“建筑安装工程”的金额
  454. 'JZAZGCF': function (tender) {
  455. let feeField = 'common',
  456. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  457. return cbTools.getBillsFee(calcBase.fixedFlag.CONSTRUCTION_INSTALL_FEE, feeField, subFeeField);
  458. },
  459. //{土地使用及拆迁补偿费}
  460. // 取清单固定类别是“土地使用及拆迁补偿费”的金额
  461. 'TDSYJCQBCF': function (tender) {
  462. let feeField = 'common',
  463. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  464. return cbTools.getBillsFee(calcBase.fixedFlag.LAND_USED_DEMOLITION, feeField, subFeeField);
  465. },
  466. //{养护工程其他费}
  467. // 取清单固定类别是“养护工程其他费”的金额
  468. 'YHGCQTF': function (tender) {
  469. let feeField = 'common',
  470. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  471. return cbTools.getBillsFee(calcBase.fixedFlag.MAINTENANCE_EXPENSES, feeField, subFeeField);
  472. },
  473. //{预备费}
  474. // 取清单固定类别是“预备费”的金额
  475. 'YBF': function(tender) {
  476. let feeField = 'common',
  477. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  478. return cbTools.getBillsFee(calcBase.fixedFlag.BUDGET_FEE, feeField, subFeeField);
  479. },
  480. //{施工场地建设费}
  481. //使用累进办法计算,基数为{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  482. 'SGCDJSF': function (tender) {
  483. let baseFee = this['DEJZAZGCFBHSBZX'](tender);
  484. return cbTools.getProgressiveFee(baseFee, '施工场地建设费');
  485. },
  486. //{养护单位(业主)管理费}
  487. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  488. 'YHDWYZGLF': function (tender) {
  489. let baseFee = this['DEJZAZGCSBSS'](tender);
  490. return cbTools.getProgressiveFee(baseFee, '养护单位(业主)管理费');
  491. },
  492. //{信息化费}
  493. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  494. 'XXHF': function (tender) {
  495. let baseFee = this['DEJZAZGCSBSS'](tender);
  496. return cbTools.getProgressiveFee(baseFee, '信息化费');
  497. },
  498. //{路线工程监理费}
  499. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  500. 'LXGCJLF': function (tender) {
  501. let baseFee = this['DEJZAZGCSBSS'](tender),
  502. fee = cbTools.getProgressiveFee(baseFee, '路线工程监理费');
  503. return fee < 20000 ? 20000 : fee;
  504. },
  505. //{独立桥梁隧道工程监理费}
  506. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  507. 'QLSDGCJLF': function (tender) {
  508. let baseFee = this['DEJZAZGCSBSS'](tender),
  509. fee = cbTools.getProgressiveFee(baseFee, '独立桥梁隧道工程监理费');
  510. return fee < 20000 ? 20000 : fee;
  511. },
  512. //{设计文件审查费}
  513. // 使用累进办法计算,不足3千按3千,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  514. 'SJWJSCF': function (tender) {
  515. let baseFee = this['DEJZAZGCSBSS'](tender),
  516. fee = cbTools.getProgressiveFee(baseFee, '设计文件审查费');
  517. return fee < 3000 ? 3000 : fee;
  518. },
  519. //{路线勘察设计费}
  520. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  521. 'LXKCSJF': function (tender) {
  522. let baseFee = this['DEJZAZGCSBSS'](tender);
  523. return cbTools.getProgressiveFee(baseFee, '路线勘察设计费');
  524. },
  525. //{独立桥梁隧道维修加固勘察设计费}
  526. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  527. 'QLSDKCSJF': function (tender) {
  528. let baseFee = this['DEJZAZGCSBSS'](tender);
  529. return cbTools.getProgressiveFee(baseFee, '独立桥梁隧道维修加固勘察设计费');
  530. },
  531. //{招标代理及标底(最高投标限价)编制费} (招标代理及标底编制费ZBDLJBDBZF)
  532. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  533. 'ZBDLJBDBZF': function (tender) {
  534. let baseFee = this['DEJZAZGCSBSS'](tender);
  535. return cbTools.getProgressiveFee(baseFee, '招标代理及标底(最高投标限价)编制费');
  536. },
  537. //价差预备费
  538. //以建筑安装工程费为基数
  539. 'JCYBF': function (tender) {
  540. //建筑安装工程费作为基数
  541. let installFee = this['JZAZGCF'](tender);
  542. //年造价增涨
  543. let costGrowthRate = projectObj.project.property.costGrowthRate ?
  544. projectObj.project.property.costGrowthRate : 0;
  545. //增涨计费年限
  546. let growthPeriod = projectObj.project.property.growthPeriod ?
  547. projectObj.project.property.growthPeriod : 0;
  548. //= P * [(1+i)^(n-1) -1]
  549. return (installFee * (Math.pow(1 + costGrowthRate, growthPeriod - 1) - 1)).toDecimal(decimalObj.bills.totalPrice);
  550. }
  551. };
  552. let figureClassTemplate = {
  553. };
  554. //基数的值不是通过清单节点获得的,则该基数的fixedBill为空,如价差、甲供、分包; class:分类,用于基数选择界面分类显示
  555. //基数本身不与清单节点关联、但是其由与清单关联的节点四则运算得到,则拥有字段multiRef: [flags...]
  556. let baseFigureMap = {
  557. //与清单直接关联=======
  558. '定额建筑安装工程费(不含定额设备购置费及专项费用)': {base: 'DEJZAZGCFBHSBZX', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE}, //设备费及专项是其子节点,所以不需要用mulRef
  559. '定额建筑安装工程(其中定额设备购置费按40%计)': {base: 'DEJZAZGCSBSS', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  560. '建筑安装工程费(不含安全生产费)': {base: 'JZAZGCFBHSC', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  561. '建筑安装工程费(不含设备费)': {base: 'JZAZGCFBHSB', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  562. '建筑安装工程费': {base: 'JZAZGCF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  563. '土地使用及拆迁补偿费': {base: 'TDSYJCQBCF', fixedFlag: fixedFlag.LAND_USED_DEMOLITION},
  564. '养护工程其他费': {base: 'YHGCQTF', fixedFlag: fixedFlag.MAINTENANCE_EXPENSES},
  565. '预备费': {base: 'YBF', fixedFlag: fixedFlag.BUDGET_FEE},
  566. '施工场地建设费': {base: 'SGCDJSF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  567. '养护单位(业主)管理费': {base: 'YHDWYZGLF',
  568. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  569. '信息化费': {base: 'XXHF',
  570. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  571. '路线工程监理费': {base: 'LXGCJLF',
  572. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  573. '独立桥梁隧道工程监理费': {base: 'QLSDGCJLF',
  574. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  575. '设计文件审查费': {base: 'SJWJSCF',
  576. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  577. '路线勘察设计费': {base: 'LXKCSJF',
  578. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  579. '独立桥梁隧道维修加固勘察设计费': {base: 'QLSDKCSJF',
  580. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  581. '招标代理及标底(最高投标限价)编制费': {base: 'ZBDLJBDBZF',
  582. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  583. '价差预备费': {base: 'JCYBF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE}
  584. };
  585. //输入式分析器
  586. let cbAnalyzer = {
  587. standar: function (exp) {
  588. //去空格
  589. exp = exp.replace(/\s/g, '');
  590. //( to (
  591. exp = exp.replace(/(/g, '(');
  592. //)to )
  593. exp = exp.replace(/)/g, ')');
  594. //,to ,
  595. exp = exp.replace(/,/g, ',');
  596. //f to F
  597. exp = exp.replace(new RegExp('f', 'g'), 'F');
  598. return exp;
  599. },
  600. //输入合法性
  601. inputLegal: function (exp) {
  602. let ilegalRex = /[^0-9,\u4e00-\u9fa5,\+,\-,\/,\*,\(,\),.,{,},F,%]/g;
  603. return !ilegalRex.test(exp);
  604. },
  605. //基数合法性、存在性
  606. baseLegal: function (baseFigures, exp) {
  607. //保证中文表达式在{}里
  608. let cnExps = cbParser.getCN(exp);
  609. let expFigures = cbParser.getFigure(exp);
  610. if(cnExps.length !== expFigures.length){
  611. throw '清单基数必须要用花括号{}括起来'
  612. return false;
  613. }
  614. for(let i = 0, len = cnExps.length; i < len; i++){
  615. if(cnExps[i] !== expFigures[i]){
  616. throw '清单基数必须要用花括号{}括起来'
  617. return false;
  618. }
  619. }
  620. //基数存在性
  621. for(let i = 0, len = expFigures.length; i < len; i++){
  622. if(cbTools.isUnDef(baseFigures[expFigures[i]])){
  623. throw `清单基数{${expFigures[i]}}不存在`;
  624. return false;
  625. }
  626. }
  627. return true;
  628. },
  629. //行引用合法性、存在性
  630. fLegal: function (items, exp) {
  631. //提取F标记
  632. let fmArr = cbParser.getFMArr(exp);
  633. //提取行引用
  634. let fArr = cbParser.getFArr(exp);
  635. if(fmArr.length !== fArr.length){
  636. return false;
  637. }
  638. //提取行数
  639. let rArr = cbParser.getXNum(fArr);
  640. if(fArr.length !== rArr.length){
  641. return false;
  642. }
  643. //判断合法性和存在性
  644. for(let i = 0, len = rArr.length; i < len; i++){
  645. let idx = rArr[i] - 1;
  646. if(cbTools.isUnDef(cbTools.getBillByRow(items, idx))){
  647. return false;
  648. }
  649. }
  650. return true;
  651. },
  652. //循环计算
  653. cycleCalc: function (node, baseFigures, exp) {
  654. let stack = [];
  655. if(node.sourceType !== calcBase.project.Bills.getSourceType()){
  656. return false;
  657. }
  658. //用于判断的起始清单ID
  659. let sIDs = cbTools.getNodeIDs(cbTools.getParents(node));
  660. let figureF = cbParser.getFigureF(cbParser.getFigure(exp), cbParser.getUID(cbParser.getFIDArr(exp)));
  661. for(let i = 0, len = figureF.length; i < len; i++){
  662. let figure = figureF[i];
  663. let billsIDs = [];
  664. if(figure.type === 'base' && cbTools.isDef(baseFigures[figure.value])){
  665. //重构后:
  666. //多重引用基数
  667. let figureMultiRef = baseFigures[figure.value]['multiRef'];
  668. let cycleCalcRef = baseFigures[figure.value]['cycleCalcRef'];
  669. if(cbTools.isDef(figureMultiRef)){
  670. if(cbTools.isDef(cycleCalcRef)){
  671. figureMultiRef = cycleCalcRef;
  672. }
  673. for(let flag of figureMultiRef){
  674. let bills = cbTools.findBill(flag);
  675. if(bills){
  676. billsIDs.push(bills.ID);
  677. }
  678. }
  679. } else {
  680. billsIDs = cbTools.isDef(baseFigures[figure.value]['fixedBill']) ? [baseFigures[figure.value]['fixedBill']['bill']['ID']] : [];
  681. }
  682. } else if(figure.type === 'id'){
  683. let node = cbTools.getNodeByID(figure.value);
  684. billsIDs = cbTools.isDef(node) ? [node.data.ID] : [];
  685. }
  686. if(cbTools.isDef(billsIDs) && billsIDs.length > 0 && isCycle(billsIDs)){
  687. console.log('循环计算');
  688. calcBase.errMsg = '表达式出现循环计算';
  689. return true;
  690. }
  691. }
  692. return false;
  693. function checkStack(stack, sIDs){
  694. //引用栈发现了初始引用
  695. for(let i = 0, len = sIDs.length; i < len; i++){
  696. if(stack.indexOf(sIDs[i]) !== -1){
  697. return true;
  698. }
  699. }
  700. return false;
  701. }
  702. function isCycle(billIDs){
  703. stack = Array.from(new Set(stack.concat(billIDs)));
  704. if (checkStack(stack, sIDs)) {
  705. return true;
  706. }
  707. for(let i = 0, len = billIDs.length; i < len; i++){
  708. let block = cbTools.getStackBlock(billIDs[i]);
  709. if(block.length > 0){
  710. stack = Array.from(new Set(stack.concat(block)));
  711. /* if(checkStack(stack, sIDs)){
  712. return true;
  713. }*/
  714. return isCycle(block);
  715. }
  716. }
  717. }
  718. },
  719. //四则运算合法性,控制不允许重复出现运算符,这里再判断一次,控制行引用只能F
  720. arithmeticLegal: function (exp) {
  721. let ilegalRex = /[\+,\-,\*,\/]{2}/g;
  722. let rex2 = /[{]{2}/g;
  723. let rex3 = /[}]{2}/g;
  724. let rex4 = /[F]{2}/g;
  725. let rex5 = /[.]{2}/g;
  726. let rex6 = /[%]{2}/g;
  727. return !ilegalRex.test(exp) && !rex2.test(exp) && !rex3.test(exp) && !rex4.test(exp) && !rex5.test(exp) && !rex6.test(exp);
  728. },
  729. //
  730. legalExp: function (node) {
  731. let exp = this.standar(node.data.userCalcBase);
  732. if(!this.inputLegal(exp)){
  733. throw '表达式含有无效字符';
  734. }
  735. if(!this.arithmeticLegal(exp)){
  736. throw '表达式含有无效字符';
  737. }
  738. if(!this.baseLegal(cbTools.getFigure(node), exp)){
  739. throw '清单基数不合法';
  740. }
  741. if(!this.fLegal(calcBase.project.mainTree.items, exp)){
  742. throw '行引用不合法';
  743. }
  744. //转换成ID引用
  745. exp = cbParser.toIDExpr(exp);
  746. if(this.cycleCalc(node, calcBase.baseFigures, exp)){
  747. throw '出现循环计算';
  748. }
  749. return exp;
  750. }
  751. };
  752. //输入式转换器
  753. let cbParser = {
  754. //获取标记F
  755. getFMArr: function (exp) {
  756. let fmRex = /F/g;
  757. let fmArr = exp.match(fmRex);
  758. return cbTools.isDef(fmArr) ? fmArr : [];
  759. },
  760. //获取行引用 eg: F10
  761. getFArr: function (exp) {
  762. let fRex = /F\d+/g;
  763. let fArr = exp.match(fRex);
  764. return cbTools.isDef(fArr) ? fArr : [];
  765. },
  766. //获取X+num eg: F10 10 @105 105
  767. getXNum: function (arr) {
  768. let rRex = /\d+/g;
  769. let tempArr = [];
  770. for(let i = 0, len = arr.length; i < len; i++){
  771. tempArr = tempArr.concat(arr[i].match(rRex));
  772. }
  773. let rArr = Array.from(new Set(tempArr));
  774. return rArr;
  775. },
  776. //获取uuid
  777. getUID: function (arr) {
  778. let rRex = /[\d,a-z,A-Z,-]{36}/g;
  779. let tempArr = [];
  780. for(let i = 0, len = arr.length; i < len; i++){
  781. tempArr = tempArr.concat(arr[i].match(rRex));
  782. }
  783. let rArr = Array.from(new Set(tempArr));
  784. return rArr;
  785. },
  786. //获取ID引用
  787. getFIDArr: function (exp) {
  788. let fidRex = /@[\d,a-z,A-Z,-]{36}/g;
  789. let fidArr = exp.match(fidRex);
  790. return cbTools.isDef(fidArr) ? fidArr : [];
  791. },
  792. //获取表达式中的中文式
  793. getCN: function(expr){
  794. let cnRex = /[\u4e00-\u9fa5]{1,}\({0,}[\u4e00-\u9fa5]{0,}\){0,}[\u4e00-\u9fa5]{0,}/g;
  795. return _.filter(expr.match(cnRex), function (data) {
  796. return data
  797. });
  798. },
  799. //获取表达式中的基数
  800. getFigure: function(expr){
  801. let rst = [];
  802. let rex = /\{([^}]*)\}/g;
  803. let temp = expr.match(rex);
  804. if(cbTools.isDef(temp)){
  805. for(let i = 0, len = temp.length; i < len; i++){
  806. rst.push(temp[i].replace(/[{,}]/g, ''));
  807. }
  808. }
  809. return rst;
  810. },
  811. //获取表达式中的基数和ID引用
  812. getFigureF: function (figures, fidArr) {
  813. let rst = [];
  814. for(let i = 0, len = figures.length; i < len; i++){
  815. let obj = Object.create(null);
  816. obj.type = 'base';
  817. obj.value = figures[i];
  818. rst.push(obj);
  819. }
  820. for(let i = 0, len = fidArr.length; i < len; i++){
  821. let obj = Object.create(null);
  822. obj.type = 'id';
  823. obj.value = fidArr[i];
  824. rst.push(obj);
  825. }
  826. return rst;
  827. },
  828. //表达式中的百分数转换成小数
  829. percentToNum: function (exp) {
  830. let rex = /\d+(\.\d+)?%/g;
  831. let percents = exp.match(rex);
  832. let numRex = /\d+(\.\d+)?/g;
  833. if(cbTools.isDef(percents)){
  834. for(let i = 0, len = percents.length; i < len; i++){
  835. let percentNum = percents[i].match(numRex);
  836. if(cbTools.isDef(percentNum) && percentNum.length === 1){
  837. exp = exp.replace(new RegExp(percents[i], 'g'), percentNum[0]/100);
  838. }
  839. }
  840. }
  841. return exp;
  842. },
  843. //将行引用转换成ID引用
  844. toIDExpr: function (exp) {
  845. let exps = [];
  846. //获得行引用
  847. let fArr = this.getFArr(exp);
  848. for(let i = 0, len = fArr.length; i < len; i++){
  849. let r = this.getXNum([fArr[i]]);
  850. if(r.length === 1){
  851. let node = cbTools.getBillByRow(calcBase.project.mainTree.items, r[0] - 1);
  852. if(cbTools.isUnDef(node)){
  853. //continue;
  854. calcBase.errMsg = '行引用错误';
  855. throw '行引用错误';
  856. }
  857. exps.push({orgExp: fArr[i], newExp: '@' + node.data.ID});
  858. }
  859. else {
  860. calcBase.errMsg = '行引用错误';
  861. throw '行引用错误';
  862. }
  863. }
  864. for(let i = 0, len = exps.length; i < len; i++){
  865. exp = exp.replace(new RegExp(exps[i].orgExp, 'g'), exps[i].newExp);
  866. }
  867. return exp;
  868. },
  869. //将ID引用转换成行引用
  870. toFExpr: function (exp) {
  871. let exps = [];
  872. //获得ID引用
  873. let fidArr = this.getFIDArr(exp);
  874. for(let i = 0, len = fidArr.length; i < len; i++){
  875. let id = this.getUID([fidArr[i]]);
  876. if(id.length === 1){
  877. let row = cbTools.getRowByID(calcBase.project.mainTree.items, id[0]);
  878. if(cbTools.isUnDef(row)){
  879. continue;
  880. }
  881. exps.push({orgExp: fidArr[i], newExp: 'F' + row});
  882. }
  883. }
  884. for(let i = 0, len = exps.length; i < len; i++){
  885. exp = exp.replace(new RegExp(exps[i].orgExp, 'g'), exps[i].newExp);
  886. }
  887. return exp;
  888. },
  889. //将表达式转换为可编译的表达式
  890. toCompileExpr: function(v){
  891. if(v === ''){
  892. return '$CBC.base(\'NONE\')';
  893. }
  894. //基数
  895. let strs = _.uniq(this.getFigure(v));
  896. let exps = [];
  897. for(let i = 0, len = strs.length; i < len; i++){
  898. let exp = Object.create(null);
  899. exp.orgExp = `{${strs[i]}}`;
  900. exps.push(exp);
  901. }
  902. for(let i = 0, len = exps.length;i < len; i++){
  903. exps[i].compileExp = '$CBC.base(\'' + exps[i].orgExp + '\')';
  904. let regStr = exps[i].orgExp.replace(/\(/g, '\\\(');
  905. regStr = regStr.replace(/\)/g, '\\\)');
  906. v = v.replace(new RegExp(regStr, 'g'), exps[i].compileExp);
  907. }
  908. //去{}
  909. v = v.replace(/[{, },]/g, '');
  910. //行引用
  911. let fidArr = this.getFIDArr(v);
  912. let fExps = [];
  913. for(let i = 0, len = fidArr.length; i < len; i++){
  914. let fExp = Object.create(null);
  915. fExp.orgExp = fidArr[i];
  916. fExps.push(fExp);
  917. }
  918. for(let i = 0, len = fExps.length; i < len; i++){
  919. fExps[i].compileExp = '$CBC.ref(\'' + fExps[i].orgExp + '\')';
  920. v = v.replace(new RegExp(fExps[i].orgExp, 'g'), fExps[i].compileExp);
  921. }
  922. return v;
  923. }
  924. };
  925. let cbCalctor = {
  926. //计算基数
  927. base: function (figure) {
  928. if(figure === 'NONE'){
  929. return 0;
  930. }
  931. return baseFigureTemplate[calcBase.baseFigures[figure]['base']]();
  932. },
  933. //调价后计算基数
  934. tenderBase: function (figure) {
  935. if(figure === 'NONE'){
  936. return 0;
  937. }
  938. return baseFigureTemplate[calcBase.baseFigures[figure]['base']](true);
  939. },
  940. //ID引用
  941. ref: function (fExp) {
  942. let ID = cbParser.getUID([fExp]);
  943. if(ID.length === 1){
  944. let node = cbTools.getNodeByID(ID[0]);
  945. return cbTools.isDef(node) &&
  946. cbTools.isDef(node.data.feesIndex) &&
  947. cbTools.isDef(node.data.feesIndex.common) &&
  948. cbTools.isDef(node.data.feesIndex.common.totalFee) ?
  949. node.data.feesIndex.common.totalFee : 0;
  950. }
  951. return 0;
  952. },
  953. //计算
  954. exec: function () {
  955. }
  956. };
  957. let calcBase = {
  958. errMsg: '表达式不正确',
  959. success: false,
  960. //清单固定行
  961. fixedFlag: null,
  962. fixedBills: Object.create(null),
  963. //清单基数
  964. baseFigures: Object.create(null),
  965. //清单可选基数映射,分两类:组织措施项目:排除父项和计算的父项; 其他项目、规费、税金、工程造价,及新增部分:显示所有计算基数
  966. baseFigureClass: Object.create(null),
  967. //初始化
  968. init: function (project) {
  969. let me = this;
  970. me.project = project;
  971. me.fixedFlag = fixedFlag;
  972. cbTools.setFixedBills(project, me.fixedBills, me.fixedFlag);
  973. me.baseFigures = baseFigureMap;
  974. cbTools.setBaseBills(me.baseFigures, me.fixedBills);
  975. //cbTools.setBaseFigureClass(me.baseFigures, me.baseFigureClass);
  976. },
  977. getBase: function (figure) {
  978. return cbCalctor.base(figure);
  979. },
  980. getBaseByClass: function (node) {
  981. return cbTools.getFigure(node);
  982. },
  983. getBaseBill: function (node) {
  984. return cbTools.getBaseBill(node);
  985. },
  986. calculate: function (node, reCalc = null) {
  987. let me = calcBase,
  988. $CBA = cbAnalyzer,
  989. $CBP = cbParser,
  990. $CBC = cbCalctor;
  991. try {
  992. debugger;
  993. me.success = false;
  994. me.errMsg = '表达式不正确';
  995. //分析输入式合法性
  996. let exp = reCalc
  997. ? cbTools.isDef(node.data.calcBase)
  998. ? node.data.calcBase
  999. : ''
  1000. : $CBA.legalExp(node);
  1001. if(!cbTools.isDef(exp)){
  1002. throw '表达式不正确';
  1003. }
  1004. //输入式转换表达式
  1005. let compileExp = $CBP.toCompileExpr(exp);
  1006. //计算
  1007. let calcExp = $CBP.percentToNum(compileExp);
  1008. let calcBaseValue = eval(calcExp);
  1009. if(!cbTools.isNum(calcBaseValue)){
  1010. throw '基数计算结果不为数值';
  1011. }
  1012. //调价
  1013. let tenderCalcExp = calcExp.replace(new RegExp('base', 'g'), 'tenderBase');
  1014. let tenderCalcBaseValue = eval(tenderCalcExp);
  1015. if(!cbTools.isNum(tenderCalcBaseValue)){
  1016. throw '调价基数计算结果不为数值';
  1017. }
  1018. //存储
  1019. me.success = true;
  1020. node.updateData.calcBase = exp;
  1021. node.updateData.calcBaseValue = parseFloat(calcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1022. node.updateData.tenderCalcBaseValue = parseFloat(tenderCalcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1023. node.changed = true;
  1024. }
  1025. catch (err){
  1026. if(typeof err === 'object'){
  1027. err = '表达式不正确'
  1028. }
  1029. if (node) {
  1030. err = `第${node.serialNo() + 1}行${err}`;
  1031. }
  1032. alert(err);
  1033. }
  1034. }
  1035. };