calc_base.js 43 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048
  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. throw `计算基数{${name}}不存在累进区间数据`;
  362. }
  363. let progression = matchData.progression;
  364. //获取区间中的最小值(0, 10] = 0
  365. function getMin(intervalStr) {
  366. let str = cbAnalyzer.standar(intervalStr);
  367. let match = /\((\d+)?/g.exec(str);
  368. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null; //后台数据单位为万元,这里转为为元
  369. }
  370. //获取区间中的最大值(0, 10] = 10
  371. function getMax(intervalStr) {
  372. let str = cbAnalyzer.standar(intervalStr);
  373. let match = /[\,,,](\d+)?/g.exec(str);
  374. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null
  375. }
  376. //将累进区间进行排序
  377. progression.sort(function (a, b) {
  378. let aV = getMin(a.interval),
  379. bV = getMin(b.interval);
  380. if (aV > bV) {
  381. return 1;
  382. } else if (aV < bV) {
  383. return -1;
  384. }
  385. return 0;
  386. });
  387. //累进计算
  388. let fee = 0;
  389. //找到所在区间
  390. let within = _.find(progression, function (data) {
  391. let min = getMin(data.interval),
  392. max = getMax(data.interval);
  393. return min !== null && baseFee > min && (max !== null && baseFee <= max || max === null);
  394. });
  395. if (!within) {
  396. return 0;
  397. }
  398. //累进之前的区间
  399. for (let i = 0; i < progression.indexOf(within); i++) {
  400. let perData = progression[i],
  401. min = getMin(perData.interval),
  402. max = getMax(perData.interval);
  403. if (min !== null && max !== null) {
  404. fee += (max - min) * perData.feeRate * 0.01;
  405. }
  406. }
  407. //累进所在区间
  408. let min = getMin(within.interval);
  409. if (min !== null) {
  410. fee += (baseFee - min) * within.feeRate * 0.01;
  411. }
  412. return fee.toDecimal(decimalObj.bills.totalPrice);
  413. }
  414. };
  415. let baseFigureTemplate = {
  416. //{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  417. //取清单固定类别是“建筑安装工程”的定额建安费,但要扣除清单固定类别是“设备购置费”、及“专项费用”的定额建安费
  418. 'DEJZAZGCFBHSBZX': function (tender) {
  419. let fullFeeField = tender ? 'rationCommon.tenderTotalFee' : 'rationCommon.totalFee',
  420. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE, fixedFlag.SPECIAL_COST];
  421. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  422. },
  423. //{定额建筑安装工程(其中定额设备购置费按 40%计)} (定额建筑安装工程设备四十)
  424. //扣除设备购置费,再加上设备购置费的40%,扣除汇总算法不四舍五入,相当于汇总当中定额设备购置费就按照了40%计
  425. 'DEJZAZGCSBSS': function (tender) {
  426. let feeField = 'rationCommon',
  427. subFeeField = tender ? 'tenderTotalFee' : 'totalFee',
  428. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  429. //建安费扣除定额设备购置费
  430. let afterDeductFee = cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, `${feeField}.${subFeeField}`);
  431. //定额设备购置费
  432. let equipmentAcFee = cbTools.getBillsFee(deductFlags[0], feeField, subFeeField);
  433. return (afterDeductFee + equipmentAcFee * 0.4).toDecimal(decimalObj.bills.totalPrice);
  434. },
  435. //{建筑安装工程费(不含安全生产费)}
  436. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“安全生产费”的金额
  437. 'JZAZGCFBHSC': function (tender) {
  438. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  439. deductFlags = [fixedFlag.SAFE_COST];
  440. //建安费扣除安全生产费
  441. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  442. },
  443. //{建筑安装工程费(不含设备费)}
  444. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“设备购置费”的金额
  445. 'JZAZGCFBHSB': function (tender) {
  446. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  447. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  448. //建安费扣除设备费
  449. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  450. },
  451. //{建筑安装工程费}
  452. // 取清单固定类别是“建筑安装工程”的金额
  453. 'JZAZGCF': function (tender) {
  454. let feeField = 'common',
  455. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  456. return cbTools.getBillsFee(calcBase.fixedFlag.CONSTRUCTION_INSTALL_FEE, feeField, subFeeField);
  457. },
  458. //{土地使用及拆迁补偿费}
  459. // 取清单固定类别是“土地使用及拆迁补偿费”的金额
  460. 'TDSYJCQBCF': function (tender) {
  461. let feeField = 'common',
  462. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  463. return cbTools.getBillsFee(calcBase.fixedFlag.LAND_USED_DEMOLITION, feeField, subFeeField);
  464. },
  465. //{养护工程其他费}
  466. // 取清单固定类别是“养护工程其他费”的金额
  467. 'YHGCQTF': function (tender) {
  468. let feeField = 'common',
  469. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  470. return cbTools.getBillsFee(calcBase.fixedFlag.MAINTENANCE_EXPENSES, feeField, subFeeField);
  471. },
  472. //{预备费}
  473. // 取清单固定类别是“预备费”的金额
  474. 'YBF': function(tender) {
  475. let feeField = 'common',
  476. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  477. return cbTools.getBillsFee(calcBase.fixedFlag.BUDGET_FEE, feeField, subFeeField);
  478. },
  479. //{施工场地建设费}
  480. //使用累进办法计算,基数为{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  481. 'SGCDJSF': function (tender) {
  482. let baseFee = this['DEJZAZGCFBHSBZX'](tender);
  483. return cbTools.getProgressiveFee(baseFee, '施工场地建设费');
  484. },
  485. //{养护单位(业主)管理费}
  486. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  487. 'YHDWYZGLF': function (tender) {
  488. let baseFee = this['DEJZAZGCSBSS'](tender);
  489. return cbTools.getProgressiveFee(baseFee, '养护单位(业主)管理费');
  490. },
  491. //{信息化费}
  492. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  493. 'XXHF': function (tender) {
  494. let baseFee = this['DEJZAZGCSBSS'](tender);
  495. return cbTools.getProgressiveFee(baseFee, '信息化费');
  496. },
  497. //{路线工程监理费}
  498. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  499. 'LXGCJLF': function (tender) {
  500. let baseFee = this['DEJZAZGCSBSS'](tender),
  501. fee = cbTools.getProgressiveFee(baseFee, '路线工程监理费');
  502. return fee < 20000 ? 20000 : fee;
  503. },
  504. //{独立桥梁隧道工程监理费}
  505. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  506. 'QLSDGCJLF': function (tender) {
  507. let baseFee = this['DEJZAZGCSBSS'](tender),
  508. fee = cbTools.getProgressiveFee(baseFee, '独立桥梁隧道工程监理费');
  509. return fee < 20000 ? 20000 : fee;
  510. },
  511. //{设计文件审查费}
  512. // 使用累进办法计算,不足3千按3千,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  513. 'SJWJSCF': function (tender) {
  514. let baseFee = this['DEJZAZGCSBSS'](tender),
  515. fee = cbTools.getProgressiveFee(baseFee, '设计文件审查费');
  516. return fee < 3000 ? 3000 : fee;
  517. },
  518. //{路线勘察设计费}
  519. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  520. 'LXKCSJF': function (tender) {
  521. let baseFee = this['DEJZAZGCSBSS'](tender);
  522. return cbTools.getProgressiveFee(baseFee, '路线勘察设计费');
  523. },
  524. //{独立桥梁隧道维修加固勘察设计费}
  525. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  526. 'QLSDKCSJF': function (tender) {
  527. let baseFee = this['DEJZAZGCSBSS'](tender);
  528. return cbTools.getProgressiveFee(baseFee, '独立桥梁隧道维修加固勘察设计费');
  529. },
  530. //{招标代理及标底(最高投标限价)编制费} (招标代理及标底编制费ZBDLJBDBZF)
  531. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  532. 'ZBDLJBDBZF': function (tender) {
  533. let baseFee = this['DEJZAZGCSBSS'](tender);
  534. return cbTools.getProgressiveFee(baseFee, '招标代理及标底(最高投标限价)编制费');
  535. },
  536. //价差预备费
  537. //以建筑安装工程费为基数
  538. 'JCYBF': function (tender) {
  539. //建筑安装工程费作为基数
  540. let installFee = this['JZAZGCF'](tender);
  541. //年造价增涨
  542. let costGrowthRate = projectObj.project.property.costGrowthRate ?
  543. projectObj.project.property.costGrowthRate : 0;
  544. //增涨计费年限
  545. let growthPeriod = projectObj.project.property.growthPeriod ?
  546. projectObj.project.property.growthPeriod : 0;
  547. //= P * [(1+i)^(n-1) -1]
  548. return (installFee * (Math.pow(1 + costGrowthRate, growthPeriod - 1) - 1)).toDecimal(decimalObj.bills.totalPrice);
  549. }
  550. };
  551. let figureClassTemplate = {
  552. };
  553. //基数的值不是通过清单节点获得的,则该基数的fixedBill为空,如价差、甲供、分包; class:分类,用于基数选择界面分类显示
  554. //基数本身不与清单节点关联、但是其由与清单关联的节点四则运算得到,则拥有字段multiRef: [flags...]
  555. let baseFigureMap = {
  556. //与清单直接关联=======
  557. '定额建筑安装工程费(不含定额设备购置费及专项费用)': {base: 'DEJZAZGCFBHSBZX', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE}, //设备费及专项是其子节点,所以不需要用mulRef
  558. '定额建筑安装工程(其中定额设备购置费按40%计)': {base: 'DEJZAZGCSBSS', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  559. '建筑安装工程费(不含安全生产费)': {base: 'JZAZGCFBHSC', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  560. '建筑安装工程费(不含设备费)': {base: 'JZAZGCFBHSB', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  561. '建筑安装工程费': {base: 'JZAZGCF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  562. '土地使用及拆迁补偿费': {base: 'TDSYJCQBCF', fixedFlag: fixedFlag.LAND_USED_DEMOLITION},
  563. '养护工程其他费': {base: 'YHGCQTF', fixedFlag: fixedFlag.MAINTENANCE_EXPENSES},
  564. '预备费': {base: 'YBF', fixedFlag: fixedFlag.BUDGET_FEE},
  565. '施工场地建设费': {base: 'SGCDJSF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  566. '养护单位(业主)管理费': {base: 'YHDWYZGLF',
  567. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  568. '信息化费': {base: 'XXHF',
  569. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  570. '路线工程监理费': {base: 'LXGCJLF',
  571. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  572. '独立桥梁隧道工程监理费': {base: 'QLSDGCJLF',
  573. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  574. '设计文件审查费': {base: 'SJWJSCF',
  575. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  576. '路线勘察设计费': {base: 'LXKCSJF',
  577. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  578. '独立桥梁隧道维修加固勘察设计费': {base: 'QLSDKCSJF',
  579. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  580. '招标代理及标底(最高投标限价)编制费': {base: 'ZBDLJBDBZF',
  581. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  582. '价差预备费': {base: 'JCYBF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE}
  583. };
  584. //输入式分析器
  585. let cbAnalyzer = {
  586. standar: function (exp) {
  587. //去空格
  588. exp = exp.replace(/\s/g, '');
  589. //( to (
  590. exp = exp.replace(/(/g, '(');
  591. //)to )
  592. exp = exp.replace(/)/g, ')');
  593. //,to ,
  594. exp = exp.replace(/,/g, ',');
  595. //f to F
  596. exp = exp.replace(new RegExp('f', 'g'), 'F');
  597. return exp;
  598. },
  599. //输入合法性
  600. inputLegal: function (exp) {
  601. let ilegalRex = /[^0-9,\u4e00-\u9fa5,\+,\-,\/,\*,\(,\),.,{,},F,%]/g;
  602. return !ilegalRex.test(exp);
  603. },
  604. //基数合法性、存在性
  605. baseLegal: function (baseFigures, exp) {
  606. //保证中文表达式在{}里
  607. let cnExps = cbParser.getCN(exp);
  608. let expFigures = cbParser.getFigure(exp);
  609. if(cnExps.length !== expFigures.length){
  610. throw '清单基数必须要用花括号{}括起来'
  611. return false;
  612. }
  613. for(let i = 0, len = cnExps.length; i < len; i++){
  614. if(cnExps[i] !== expFigures[i]){
  615. throw '清单基数必须要用花括号{}括起来'
  616. return false;
  617. }
  618. }
  619. //基数存在性
  620. for(let i = 0, len = expFigures.length; i < len; i++){
  621. if(cbTools.isUnDef(baseFigures[expFigures[i]])){
  622. throw `清单基数{${expFigures[i]}}不存在`;
  623. return false;
  624. }
  625. }
  626. return true;
  627. },
  628. //行引用合法性、存在性
  629. fLegal: function (items, exp) {
  630. //提取F标记
  631. let fmArr = cbParser.getFMArr(exp);
  632. //提取行引用
  633. let fArr = cbParser.getFArr(exp);
  634. if(fmArr.length !== fArr.length){
  635. return false;
  636. }
  637. //提取行数
  638. let rArr = cbParser.getXNum(fArr);
  639. if(fArr.length !== rArr.length){
  640. return false;
  641. }
  642. //判断合法性和存在性
  643. for(let i = 0, len = rArr.length; i < len; i++){
  644. let idx = rArr[i] - 1;
  645. if(cbTools.isUnDef(cbTools.getBillByRow(items, idx))){
  646. return false;
  647. }
  648. }
  649. return true;
  650. },
  651. //循环计算
  652. cycleCalc: function (node, baseFigures, exp) {
  653. let stack = [];
  654. if(node.sourceType !== calcBase.project.Bills.getSourceType()){
  655. return false;
  656. }
  657. //用于判断的起始清单ID
  658. let sIDs = cbTools.getNodeIDs(cbTools.getParents(node));
  659. let figureF = cbParser.getFigureF(cbParser.getFigure(exp), cbParser.getUID(cbParser.getFIDArr(exp)));
  660. for(let i = 0, len = figureF.length; i < len; i++){
  661. let figure = figureF[i];
  662. let billsIDs = [];
  663. if(figure.type === 'base' && cbTools.isDef(baseFigures[figure.value])){
  664. //重构后:
  665. //多重引用基数
  666. let figureMultiRef = baseFigures[figure.value]['multiRef'];
  667. let cycleCalcRef = baseFigures[figure.value]['cycleCalcRef'];
  668. if(cbTools.isDef(figureMultiRef)){
  669. if(cbTools.isDef(cycleCalcRef)){
  670. figureMultiRef = cycleCalcRef;
  671. }
  672. for(let flag of figureMultiRef){
  673. let bills = cbTools.findBill(flag);
  674. if(bills){
  675. billsIDs.push(bills.ID);
  676. }
  677. }
  678. } else {
  679. billsIDs = cbTools.isDef(baseFigures[figure.value]['fixedBill']) ? [baseFigures[figure.value]['fixedBill']['bill']['ID']] : [];
  680. }
  681. } else if(figure.type === 'id'){
  682. let node = cbTools.getNodeByID(figure.value);
  683. billsIDs = cbTools.isDef(node) ? [node.data.ID] : [];
  684. }
  685. if(cbTools.isDef(billsIDs) && billsIDs.length > 0 && isCycle(billsIDs)){
  686. console.log('循环计算');
  687. calcBase.errMsg = '表达式出现循环计算';
  688. return true;
  689. }
  690. }
  691. return false;
  692. function checkStack(stack, sIDs){
  693. //引用栈发现了初始引用
  694. for(let i = 0, len = sIDs.length; i < len; i++){
  695. if(stack.indexOf(sIDs[i]) !== -1){
  696. return true;
  697. }
  698. }
  699. return false;
  700. }
  701. function isCycle(billIDs){
  702. stack = Array.from(new Set(stack.concat(billIDs)));
  703. if (checkStack(stack, sIDs)) {
  704. return true;
  705. }
  706. for(let i = 0, len = billIDs.length; i < len; i++){
  707. let block = cbTools.getStackBlock(billIDs[i]);
  708. if(block.length > 0){
  709. stack = Array.from(new Set(stack.concat(block)));
  710. /* if(checkStack(stack, sIDs)){
  711. return true;
  712. }*/
  713. return isCycle(block);
  714. }
  715. }
  716. }
  717. },
  718. //四则运算合法性,控制不允许重复出现运算符,这里再判断一次,控制行引用只能F
  719. arithmeticLegal: function (exp) {
  720. let ilegalRex = /[\+,\-,\*,\/]{2}/g;
  721. let rex2 = /[{]{2}/g;
  722. let rex3 = /[}]{2}/g;
  723. let rex4 = /[F]{2}/g;
  724. let rex5 = /[.]{2}/g;
  725. let rex6 = /[%]{2}/g;
  726. return !ilegalRex.test(exp) && !rex2.test(exp) && !rex3.test(exp) && !rex4.test(exp) && !rex5.test(exp) && !rex6.test(exp);
  727. },
  728. //
  729. legalExp: function (node) {
  730. let exp = this.standar(node.data.userCalcBase);
  731. if(!this.inputLegal(exp)){
  732. throw '表达式含有无效字符';
  733. }
  734. if(!this.arithmeticLegal(exp)){
  735. throw '表达式含有无效字符';
  736. }
  737. if(!this.baseLegal(cbTools.getFigure(node), exp)){
  738. throw '清单基数不合法';
  739. }
  740. if(!this.fLegal(calcBase.project.mainTree.items, exp)){
  741. throw '行引用不合法';
  742. }
  743. //转换成ID引用
  744. exp = cbParser.toIDExpr(exp);
  745. if(this.cycleCalc(node, calcBase.baseFigures, exp)){
  746. throw '出现循环计算';
  747. }
  748. return exp;
  749. }
  750. };
  751. //输入式转换器
  752. let cbParser = {
  753. //获取标记F
  754. getFMArr: function (exp) {
  755. let fmRex = /F/g;
  756. let fmArr = exp.match(fmRex);
  757. return cbTools.isDef(fmArr) ? fmArr : [];
  758. },
  759. //获取行引用 eg: F10
  760. getFArr: function (exp) {
  761. let fRex = /F\d+/g;
  762. let fArr = exp.match(fRex);
  763. return cbTools.isDef(fArr) ? fArr : [];
  764. },
  765. //获取X+num eg: F10 10 @105 105
  766. getXNum: function (arr) {
  767. let rRex = /\d+/g;
  768. let tempArr = [];
  769. for(let i = 0, len = arr.length; i < len; i++){
  770. tempArr = tempArr.concat(arr[i].match(rRex));
  771. }
  772. let rArr = Array.from(new Set(tempArr));
  773. return rArr;
  774. },
  775. //获取uuid
  776. getUID: function (arr) {
  777. let rRex = /[\d,a-z,A-Z,-]{36}/g;
  778. let tempArr = [];
  779. for(let i = 0, len = arr.length; i < len; i++){
  780. tempArr = tempArr.concat(arr[i].match(rRex));
  781. }
  782. let rArr = Array.from(new Set(tempArr));
  783. return rArr;
  784. },
  785. //获取ID引用
  786. getFIDArr: function (exp) {
  787. let fidRex = /@[\d,a-z,A-Z,-]{36}/g;
  788. let fidArr = exp.match(fidRex);
  789. return cbTools.isDef(fidArr) ? fidArr : [];
  790. },
  791. //获取表达式中的中文式
  792. getCN: function(expr){
  793. let cnRex = /[\u4e00-\u9fa5]{1,}\({0,}[\u4e00-\u9fa5]{0,}\){0,}[\u4e00-\u9fa5]{0,}/g;
  794. return _.filter(expr.match(cnRex), function (data) {
  795. return data
  796. });
  797. },
  798. //获取表达式中的基数
  799. getFigure: function(expr){
  800. let rst = [];
  801. let rex = /\{([^}]*)\}/g;
  802. let temp = expr.match(rex);
  803. if(cbTools.isDef(temp)){
  804. for(let i = 0, len = temp.length; i < len; i++){
  805. rst.push(temp[i].replace(/[{,}]/g, ''));
  806. }
  807. }
  808. return rst;
  809. },
  810. //获取表达式中的基数和ID引用
  811. getFigureF: function (figures, fidArr) {
  812. let rst = [];
  813. for(let i = 0, len = figures.length; i < len; i++){
  814. let obj = Object.create(null);
  815. obj.type = 'base';
  816. obj.value = figures[i];
  817. rst.push(obj);
  818. }
  819. for(let i = 0, len = fidArr.length; i < len; i++){
  820. let obj = Object.create(null);
  821. obj.type = 'id';
  822. obj.value = fidArr[i];
  823. rst.push(obj);
  824. }
  825. return rst;
  826. },
  827. //表达式中的百分数转换成小数
  828. percentToNum: function (exp) {
  829. let rex = /\d+(\.\d+)?%/g;
  830. let percents = exp.match(rex);
  831. let numRex = /\d+(\.\d+)?/g;
  832. if(cbTools.isDef(percents)){
  833. for(let i = 0, len = percents.length; i < len; i++){
  834. let percentNum = percents[i].match(numRex);
  835. if(cbTools.isDef(percentNum) && percentNum.length === 1){
  836. exp = exp.replace(new RegExp(percents[i], 'g'), percentNum[0]/100);
  837. }
  838. }
  839. }
  840. return exp;
  841. },
  842. //将行引用转换成ID引用
  843. toIDExpr: function (exp) {
  844. let exps = [];
  845. //获得行引用
  846. let fArr = this.getFArr(exp);
  847. for(let i = 0, len = fArr.length; i < len; i++){
  848. let r = this.getXNum([fArr[i]]);
  849. if(r.length === 1){
  850. let node = cbTools.getBillByRow(calcBase.project.mainTree.items, r[0] - 1);
  851. if(cbTools.isUnDef(node)){
  852. //continue;
  853. calcBase.errMsg = '行引用错误';
  854. throw '行引用错误';
  855. }
  856. exps.push({orgExp: fArr[i], newExp: '@' + node.data.ID});
  857. }
  858. else {
  859. calcBase.errMsg = '行引用错误';
  860. throw '行引用错误';
  861. }
  862. }
  863. for(let i = 0, len = exps.length; i < len; i++){
  864. exp = exp.replace(new RegExp(exps[i].orgExp, 'g'), exps[i].newExp);
  865. }
  866. return exp;
  867. },
  868. //将ID引用转换成行引用
  869. toFExpr: function (exp) {
  870. let exps = [];
  871. //获得ID引用
  872. let fidArr = this.getFIDArr(exp);
  873. for(let i = 0, len = fidArr.length; i < len; i++){
  874. let id = this.getUID([fidArr[i]]);
  875. if(id.length === 1){
  876. let row = cbTools.getRowByID(calcBase.project.mainTree.items, id[0]);
  877. if(cbTools.isUnDef(row)){
  878. continue;
  879. }
  880. exps.push({orgExp: fidArr[i], newExp: 'F' + row});
  881. }
  882. }
  883. for(let i = 0, len = exps.length; i < len; i++){
  884. exp = exp.replace(new RegExp(exps[i].orgExp, 'g'), exps[i].newExp);
  885. }
  886. return exp;
  887. },
  888. //将表达式转换为可编译的表达式
  889. toCompileExpr: function(v){
  890. if(v === ''){
  891. return '$CBC.base(\'NONE\')';
  892. }
  893. //基数
  894. let strs = _.uniq(this.getFigure(v));
  895. let exps = [];
  896. for(let i = 0, len = strs.length; i < len; i++){
  897. let exp = Object.create(null);
  898. exp.orgExp = `{${strs[i]}}`;
  899. exps.push(exp);
  900. }
  901. for(let i = 0, len = exps.length;i < len; i++){
  902. exps[i].compileExp = '$CBC.base(\'' + exps[i].orgExp + '\')';
  903. let regStr = exps[i].orgExp.replace(/\(/g, '\\\(');
  904. regStr = regStr.replace(/\)/g, '\\\)');
  905. v = v.replace(new RegExp(regStr, 'g'), exps[i].compileExp);
  906. }
  907. //去{}
  908. v = v.replace(/[{, },]/g, '');
  909. //行引用
  910. let fidArr = this.getFIDArr(v);
  911. let fExps = [];
  912. for(let i = 0, len = fidArr.length; i < len; i++){
  913. let fExp = Object.create(null);
  914. fExp.orgExp = fidArr[i];
  915. fExps.push(fExp);
  916. }
  917. for(let i = 0, len = fExps.length; i < len; i++){
  918. fExps[i].compileExp = '$CBC.ref(\'' + fExps[i].orgExp + '\')';
  919. v = v.replace(new RegExp(fExps[i].orgExp, 'g'), fExps[i].compileExp);
  920. }
  921. return v;
  922. }
  923. };
  924. let cbCalctor = {
  925. //计算基数
  926. base: function (figure) {
  927. if(figure === 'NONE'){
  928. return 0;
  929. }
  930. return baseFigureTemplate[calcBase.baseFigures[figure]['base']]();
  931. },
  932. //调价后计算基数
  933. tenderBase: function (figure) {
  934. if(figure === 'NONE'){
  935. return 0;
  936. }
  937. return baseFigureTemplate[calcBase.baseFigures[figure]['base']](true);
  938. },
  939. //ID引用
  940. ref: function (fExp) {
  941. let ID = cbParser.getUID([fExp]);
  942. if(ID.length === 1){
  943. let node = cbTools.getNodeByID(ID[0]);
  944. return cbTools.isDef(node) &&
  945. cbTools.isDef(node.data.feesIndex) &&
  946. cbTools.isDef(node.data.feesIndex.common) &&
  947. cbTools.isDef(node.data.feesIndex.common.totalFee) ?
  948. node.data.feesIndex.common.totalFee : 0;
  949. }
  950. return 0;
  951. },
  952. //计算
  953. exec: function () {
  954. }
  955. };
  956. let calcBase = {
  957. errMsg: '表达式不正确',
  958. success: false,
  959. //清单固定行
  960. fixedFlag: null,
  961. fixedBills: Object.create(null),
  962. //清单基数
  963. baseFigures: Object.create(null),
  964. //清单可选基数映射,分两类:组织措施项目:排除父项和计算的父项; 其他项目、规费、税金、工程造价,及新增部分:显示所有计算基数
  965. baseFigureClass: Object.create(null),
  966. //初始化
  967. init: function (project) {
  968. let me = this;
  969. me.project = project;
  970. me.fixedFlag = fixedFlag;
  971. cbTools.setFixedBills(project, me.fixedBills, me.fixedFlag);
  972. me.baseFigures = baseFigureMap;
  973. cbTools.setBaseBills(me.baseFigures, me.fixedBills);
  974. //cbTools.setBaseFigureClass(me.baseFigures, me.baseFigureClass);
  975. },
  976. getBase: function (figure) {
  977. return cbCalctor.base(figure);
  978. },
  979. getBaseByClass: function (node) {
  980. return cbTools.getFigure(node);
  981. },
  982. getBaseBill: function (node) {
  983. return cbTools.getBaseBill(node);
  984. },
  985. calculate: function (node, reCalc = null) {
  986. let me = calcBase,
  987. $CBA = cbAnalyzer,
  988. $CBP = cbParser,
  989. $CBC = cbCalctor;
  990. try {
  991. me.success = false;
  992. me.errMsg = '表达式不正确';
  993. //分析输入式合法性
  994. let exp = reCalc
  995. ? cbTools.isDef(node.data.calcBase)
  996. ? node.data.calcBase
  997. : ''
  998. : $CBA.legalExp(node);
  999. if(!cbTools.isDef(exp)){
  1000. throw '表达式不正确';
  1001. }
  1002. //输入式转换表达式
  1003. let compileExp = $CBP.toCompileExpr(exp);
  1004. //计算
  1005. let calcExp = $CBP.percentToNum(compileExp);
  1006. let calcBaseValue = eval(calcExp);
  1007. if(!cbTools.isNum(calcBaseValue)){
  1008. throw '基数计算结果不为数值';
  1009. }
  1010. //调价
  1011. let tenderCalcExp = calcExp.replace(new RegExp('base', 'g'), 'tenderBase');
  1012. let tenderCalcBaseValue = eval(tenderCalcExp);
  1013. if(!cbTools.isNum(tenderCalcBaseValue)){
  1014. throw '调价基数计算结果不为数值';
  1015. }
  1016. //存储
  1017. me.success = true;
  1018. node.updateData.calcBase = exp;
  1019. node.updateData.calcBaseValue = parseFloat(calcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1020. node.updateData.tenderCalcBaseValue = parseFloat(tenderCalcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1021. node.changed = true;
  1022. }
  1023. catch (err){
  1024. if(typeof err === 'object'){
  1025. err = '表达式不正确'
  1026. }
  1027. if (node) {
  1028. err = `第${node.serialNo() + 1}行${err}`;
  1029. }
  1030. alert(err);
  1031. }
  1032. }
  1033. };