calc_base.js 55 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465
  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) && this.isDef(v.flagsIndex.fixed.flag);
  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. //通过行获取节点清单
  44. getBillByRow: function (items, row) {
  45. if (cbTools.isDef(items[row]) &&
  46. cbTools.isDef(items[row]['sourceType']) &&
  47. items[row]['sourceType'] === calcBase.project.Bills.getSourceType()) {
  48. return items[row];
  49. }
  50. return null;
  51. },
  52. //通过ID获取节点行
  53. getRowByID: function (items, ID) {
  54. for (let i = 0, len = items.length; i < len; i++) {
  55. if (items[i]['data']['ID'] == ID) {
  56. return i + 1;
  57. }
  58. }
  59. return null;
  60. },
  61. //通过ID获取节点
  62. getNodeByID: function (ID) {
  63. return this.isDef(calcBase.project.mainTree.nodes['id_' + ID]) ? calcBase.project.mainTree.nodes['id_' + ID] : null;
  64. },
  65. //获取该节点所有父节点
  66. getParents: function (node) {
  67. let rst = [];
  68. rst.push(node);
  69. rParent(node);
  70. return rst;
  71. function rParent(node) {
  72. if (cbTools.isDef(node.parent)) {
  73. rst.push(node.parent);
  74. rParent(node.parent);
  75. }
  76. }
  77. },
  78. //获取所有节点的ID
  79. getNodeIDs: function (nodes) {
  80. let rst = [];
  81. for (let i = 0, len = nodes.length; i < len; i++) {
  82. if (this.isDef(nodes[i]['data']['ID'])) {
  83. rst.push(nodes[i]['data']['ID']);
  84. }
  85. }
  86. return rst;
  87. },
  88. //根据公式获取相关的节点
  89. getNodesByExp: function (node, formulaNodesArr) {
  90. let exp = node.data.calcBase;
  91. let rst = [],
  92. ids = [];
  93. if (this.isUnDef(exp) || exp === '') {
  94. return rst;
  95. }
  96. let findChildNodes = []; //直接引用的节点,这些节点可能存在子节点,子节点才有公式,因此获取这些节点的子公式节点
  97. //获取表达式中的基数和行引用
  98. let figureF = cbParser.getFigureF(cbParser.getFigure(exp), cbParser.getUID(cbParser.getFIDArr(exp)));
  99. //首先提取出多处引用的进行排序
  100. for (let i = 0, len = figureF.length; i < len; i++) {
  101. let figure = figureF[i];
  102. if (figure.type === 'base' && calcBase.baseFigures && cbTools.isDef(calcBase.baseFigures[figure.value])) {
  103. let bill = this.isDef(calcBase.baseFigures[figure.value]['fixedBill']) ? calcBase.baseFigures[figure.value]['fixedBill']['bill'] : null;
  104. let figureMultiRef = calcBase.baseFigures[figure.value]['multiRef'];
  105. if (this.isDef(figureMultiRef)) {
  106. for (let flag of figureMultiRef) {
  107. let refNode = this.findBill(flag) ? this.getNodeByID(this.findBill(flag).ID) : null;
  108. if (refNode && !ids.includes(refNode.data.ID)) {
  109. findChildNodes.push(refNode);
  110. ids.push(refNode.data.ID);
  111. }
  112. }
  113. } else if (this.isDef(bill) && ids.indexOf(bill.ID) === -1) {
  114. let node = this.getNodeByID(bill.ID);
  115. if (this.isDef(node) && !ids.includes(node.data.ID)) {
  116. findChildNodes.push(node);
  117. ids.push(node.data.ID);
  118. }
  119. }
  120. } else if (figure.type === 'id') {
  121. let node = this.getNodeByID(figure.value);
  122. if (this.isDef(node) && !ids.includes(node.data.ID)) {
  123. findChildNodes.push(node);
  124. ids.push(node.data.ID);
  125. }
  126. }
  127. }
  128. if (findChildNodes.length > 0) {
  129. let childrenNodes = calcTools.getChildrenFormulaNodes(node, formulaNodesArr, findChildNodes);
  130. rst = rst.concat(childrenNodes);
  131. }
  132. return rst;
  133. },
  134. //需要用到计算基数的时候,先找出所有的固定清单,避免每个基数都要去遍历寻找清单
  135. setFixedBills: function (project, billsObj, fixedFlag) {
  136. let bills = project.Bills.datas;
  137. for (let i = 0, len = bills.length; i < len; i++) {
  138. if (this.isDef(bills[i].flagsIndex.fixed)) {
  139. for (let flag in fixedFlag) {
  140. if (fixedFlag[flag] === bills[i].flagsIndex.fixed.flag) {
  141. billsObj[fixedFlag[flag]] = Object.create(null);
  142. billsObj[fixedFlag[flag]]['base'] = Object.create(null);
  143. billsObj[fixedFlag[flag]]['bill'] = bills[i];
  144. }
  145. }
  146. }
  147. }
  148. },
  149. //清单基数设置所属固定清单属性
  150. setBaseBills: function (baseFigure, fixedBills) {
  151. for (let i in baseFigure) {
  152. let calcBase = baseFigure[i];
  153. calcBase.fixedBill = null;
  154. if (cbTools.isDef(calcBase.fixedFlag) && cbTools.isDef(fixedBills[calcBase.fixedFlag])) {
  155. fixedBills[calcBase.fixedFlag]['base'][i] = calcBase;
  156. calcBase.fixedBill = fixedBills[calcBase.fixedFlag];
  157. }
  158. }
  159. },
  160. //设置清单固定行下可用的基数映射
  161. //@param {Object}baseFigures(当前项目可用总基数配置表) {Object}mapping(可用基数映射,初始为空object,目标:{flag: Array(baseList)}) eg: {'1': ['xx费']}
  162. setValidBaseMapping: function (baseFigures, mapping) {
  163. //清单固定行数组[1, 2...]
  164. let allFlags = [];
  165. //清单固定行与子清单固定行映射
  166. let subFlagMapping = {};
  167. for (let attr in fixedFlag) {
  168. let flag = fixedFlag[attr];
  169. allFlags.push(flag);
  170. let subFlagList = this.getSubFlagList(flag);
  171. subFlagMapping[flag] = subFlagList;
  172. }
  173. for (let baseName in baseFigures) {
  174. let calcBase = baseFigures[baseName],
  175. filter = calcBase.filter || Object.values(fixedFlag), // filter为空则全部部分都可用该基数
  176. pick = calcBase.pick; //挑选或过滤
  177. /* if (!filter) {
  178. continue;
  179. } */
  180. //pick为true,则filter中的清单固定行可使用此基数(及其子清单固定行),
  181. //pick为false除去filter中的清单固定行(及其子清单固定行),其他可使用此基数(包括新增的大项费用)
  182. let allFilter = []; //filter及其子项
  183. for (let flag of filter) {
  184. if (subFlagMapping[flag].length > 0) {
  185. allFilter = allFilter.concat(subFlagMapping[flag]);
  186. }
  187. }
  188. allFilter = allFilter.concat(filter);
  189. allFilter = Array.from(new Set(allFilter));
  190. //获取可使用此基数的清单固定行
  191. let validFlags = pick ? allFilter : allFlags.filter(function (flag) {
  192. return !allFilter.includes(flag);
  193. });
  194. //其他节点可使用的基数(新增的大项费用),即基数配置表中过滤条件为“只允许非固定类别是xxx”的
  195. //允许非固定类别xx可用,则新增的大项费用也可用,新增的大项费用flag为null
  196. if (!pick) {
  197. if (mapping['other']) {
  198. mapping['other'].push(baseName);
  199. } else {
  200. mapping['other'] = [baseName];
  201. }
  202. }
  203. //设置清单固定行可使用此基数
  204. for (let flag of validFlags) {
  205. if (mapping[flag]) {
  206. mapping[flag].push(baseName)
  207. } else {
  208. mapping[flag] = [baseName];
  209. }
  210. }
  211. }
  212. },
  213. //该节点可使用的基数列表
  214. getValidFigures: function (node) {
  215. let filterMap = {},
  216. avaBaseNames = [];
  217. //该节点所属的固定行
  218. let belongFlag = this.getBelongFlag(node);
  219. //没有所属固定行,则属于新增的大项费用
  220. //获取可使用的基数
  221. if (!belongFlag) {
  222. avaBaseNames = calcBase.flagValidBase['other'];
  223. } else {
  224. avaBaseNames = calcBase.flagValidBase[belongFlag] ? calcBase.flagValidBase[belongFlag] : [];
  225. }
  226. for (let baseName of avaBaseNames) {
  227. let base = calcBase.baseFigures[baseName];
  228. if (baseName) {
  229. filterMap[baseName] = base;
  230. }
  231. }
  232. return filterMap
  233. },
  234. //根据清单固定行,获取子固定行
  235. getSubFlagList: function (flag) {
  236. let flagList = [];
  237. let node = this.findNodeByFlag(flag);
  238. if (!node) {
  239. return flagList;
  240. }
  241. let allChildren = [];
  242. function getChildren(nodes) {
  243. allChildren = allChildren.concat(nodes);
  244. for (let node of nodes) {
  245. if (node.children.length > 0) {
  246. getChildren(node.children);
  247. }
  248. }
  249. }
  250. getChildren(node.children);
  251. for (let child of allChildren) {
  252. if (child.data && this.isFlag(child.data)) {
  253. flagList.push(child.data.flagsIndex.fixed.flag);
  254. }
  255. }
  256. return flagList
  257. },
  258. //获取节点所属的清单固定行
  259. getBelongFlag: function (node) {
  260. while (node) {
  261. if (node.data && this.isFlag(node.data)) {
  262. return node.data.flagsIndex.fixed.flag;
  263. }
  264. node = node.parent;
  265. }
  266. return null;
  267. },
  268. //获取节点所属的清单固定列表
  269. getBelongFlagList: function (node) {
  270. let rst = [];
  271. while (node) {
  272. if (node.data && this.isFlag(node.data)) {
  273. rst.push(node.data.flagsIndex.fixed.flag);
  274. }
  275. node = node.parent;
  276. }
  277. return rst;
  278. },
  279. //获取清单(有基数计算)引用了的其他清单,(循环引用栈中的一块)
  280. getStackBlock: function (billID) {
  281. let tempBases = [],
  282. block = []; //存引用的清单ID
  283. let node = getBill(billID);
  284. if (!node) {
  285. return tempBases;
  286. } else {
  287. //获取基数和行引用
  288. getBase(node);
  289. let bases = Array.from(new Set(tempBases));
  290. //根据基数和行引用获取清单ID
  291. for (let i = 0, len = bases.length; i < len; i++) {
  292. //基数是跟清单直接关联的
  293. if (bases[i]['type'] === 'base' && cbTools.isDef(calcBase.baseFigures[bases[i]['value']])) {
  294. let figureMultiRef = calcBase.baseFigures[bases[i]['value']]['multiRef'];
  295. let cycleCalcRef = calcBase.baseFigures[bases[i]['value']]['cycleCalcRef'];
  296. if (cbTools.isDef(figureMultiRef)) {
  297. if (cbTools.isDef(cycleCalcRef)) {
  298. figureMultiRef = cycleCalcRef;
  299. }
  300. for (let flag of figureMultiRef) {
  301. let bills = cbTools.findBill(flag);
  302. if (cbTools.isDef(bills)) {
  303. block.push(bills.ID);
  304. }
  305. }
  306. } else if (cbTools.isDef(calcBase.baseFigures[bases[i]['value']]['fixedBill'])) {
  307. block.push(calcBase.baseFigures[bases[i]['value']]['fixedBill']['bill']['ID']);
  308. }
  309. } else if (bases[i]['type'] === 'id') {
  310. let node = cbTools.getNodeByID(bases[i]['value']);
  311. if (cbTools.isDef(node)) {
  312. block.push(node.data.ID);
  313. }
  314. }
  315. }
  316. return Array.from(new Set(block));
  317. }
  318. function getBase(node) {
  319. if (node && node.children.length === 0) {
  320. if (cbTools.isDef(node.data.calcBase) && node.data.calcBase !== '') {
  321. let figureF = cbParser.getFigureF(cbParser.getFigure(node.data.calcBase), cbParser.getUID(cbParser.getFIDArr(node.data.calcBase)));
  322. tempBases = tempBases.concat(figureF);
  323. }
  324. } else if (node && node.children.length > 0) {
  325. for (let i = 0, len = node.children.length; i < len; i++) {
  326. getBase(node.children[i]);
  327. }
  328. }
  329. }
  330. function getBill(ID) {
  331. let nodes = calcBase.project.mainTree.nodes;
  332. let node = nodes['id_' + ID];
  333. if (cbTools.isDef(node) && node.sourceType === calcBase.project.Bills.getSourceType()) {
  334. return node;
  335. }
  336. return null;
  337. }
  338. },
  339. // 获取全部有公式的树节点清单。 CSL, 2018-01-05
  340. getFormulaNodes: function (needOrder = false) {
  341. // 给公式结点清单换照引用计算顺序排序。
  342. function orderFormulaNodes(nodesArr) {
  343. let orderArr = [];
  344. function recursionNode(nodes) {
  345. for (let node of nodes) {
  346. if (orderArr.includes(node)) continue; // 已排过序的节点则跳过
  347. if (node.data.calcBase) {
  348. let subNodes = cbTools.getNodesByExp(node, nodesArr);
  349. recursionNode(subNodes);
  350. };
  351. if (nodesArr.includes(node) && !orderArr.includes(node)) orderArr.push(node);
  352. };
  353. }
  354. recursionNode(nodesArr);
  355. return orderArr;
  356. };
  357. let nodes = [];
  358. for (let node of projectObj.project.mainTree.items) {
  359. if (node.sourceType == ModuleNames.bills && node.data.calcBase && node.data.calcBase != '')
  360. nodes.push(node);
  361. };
  362. if (needOrder && nodes.length >= 2) return orderFormulaNodes(nodes)
  363. else return nodes;
  364. },
  365. // 刷新全部行引用的公式清单。 CSL, 2018-01-05
  366. refreshFormulaNodes: function () {
  367. try {
  368. let nodes = this.getFormulaNodes();
  369. if (nodes.length > 0) projectObj.mainController.refreshTreeNode(nodes, false, false);
  370. } catch (err) {
  371. alert('公式引用行号显示刷新失败:' + err.message);
  372. }
  373. },
  374. // 判断结点是否被其它结点的表达式引用。
  375. isUsedByFormula: function (node) {
  376. let nodes = this.getFormulaNodes();
  377. if (nodes.length == 0) return false;
  378. let sID = '@' + node.data.ID;
  379. for (let node of nodes) {
  380. if (node.data.calcBase.hasSubStr(sID)) return true;
  381. };
  382. },
  383. // 获取直接关联清单节点的基数金额
  384. getBaseFee: function (flag, tender, feeField) {
  385. const subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  386. return this.getBillsFee(flag, feeField, subFeeField);
  387. },
  388. //获取清单节点的金额
  389. //@param {Number}fixedFlag(清单固定行类别) {String}feeField(外层金额字段: common) {String}subFeeField(子金额字段: totalFee)
  390. //@return {Number}
  391. getBillsFee: function (fixedFlag, feeField, subFeeField) {
  392. //固定清单类别与清单数据、关联基数的映射
  393. let fixedBills = calcBase.fixedBills[fixedFlag];
  394. if (this.isUnDef(fixedBills)) {
  395. return 0;
  396. }
  397. let bills = fixedBills.bill;
  398. if (this.isUnDef(bills)) {
  399. return 0;
  400. }
  401. if (this.isUnDef(bills.feesIndex) || _.isEmpty(bills.feesIndex)) {
  402. return 0;
  403. }
  404. return this.isDef(bills.feesIndex[feeField]) &&
  405. this.isDef(bills.feesIndex[feeField][subFeeField]) && !isNaN(bills.feesIndex[feeField][subFeeField]) ?
  406. bills.feesIndex[feeField][subFeeField] :
  407. 0;
  408. },
  409. getFee: function (bills, feeField, subFeeField) {
  410. if (this.isUnDef(bills)) {
  411. return 0;
  412. }
  413. if (this.isUnDef(bills.feesIndex) || _.isEmpty(bills.feesIndex)) {
  414. return 0;
  415. }
  416. return this.isDef(bills.feesIndex[feeField]) &&
  417. this.isDef(bills.feesIndex[feeField][subFeeField]) &&
  418. !isNaN(bills.feesIndex[feeField][subFeeField]) ?
  419. bills.feesIndex[feeField][subFeeField] :
  420. 0;
  421. },
  422. /**
  423. * 获取扣除固定项后的金额,扣除其节点后重新汇总
  424. * @param {Number} fixedFlag - 基数取值固定行类别
  425. * @param {Array} deductFlags - 扣除的固定类别组
  426. * @param {Boolean} tender - 是否调价
  427. * @param {String} feeField - 价格字段
  428. * @param {Boolean = true} isRound - 是否取舍
  429. * @return {Number}
  430. */
  431. getFeeWithDeduction: function (fixedFlag, deductFlags, tender, feeField, isRound = true) {
  432. const fullFeeField = tender ? `${feeField}.tenderTotalFee` : `${feeField}.totalFee`;
  433. let baseNode = this.findNodeByFlag(fixedFlag);
  434. if (!baseNode) {
  435. return 0;
  436. }
  437. //要扣除的节点
  438. let deductNodes = [];
  439. for (let deFlag of deductFlags) {
  440. let node = this.findNodeByFlag(deFlag);
  441. if (node) {
  442. deductNodes.push(node);
  443. }
  444. }
  445. const fee = projectObj.project.calcProgram.getTotalFee([baseNode], deductNodes, fullFeeField);
  446. return isRound ? fee.toDecimal(decimalObj.bills.totalPrice) : fee;
  447. },
  448. /* getFeeWithDeduction: function (fixedFlag, deductFlags, fullFeeField) {
  449. let baseNode = this.findNodeByFlag(fixedFlag);
  450. if (!baseNode) {
  451. return 0;
  452. }
  453. //要扣除的节点
  454. let deductNodes = [];
  455. for (let deFlag of deductFlags) {
  456. let node = this.findNodeByFlag(deFlag);
  457. if (node) {
  458. deductNodes.push(node);
  459. }
  460. }
  461. return projectObj.project.calcProgram.getTotalFee([baseNode], deductNodes, fullFeeField);
  462. }, */
  463. //获取累进办法计算的金额
  464. //@param {Number}baseFee(相关基数金额) {String}name(使用累进计算的基数名称)
  465. //@return {Number}
  466. getProgressiveFee: function (baseFee, name) {
  467. const progressiveData = calcBase.project.property.progressiveInterval;
  468. if (!progressiveData) {
  469. throw '该项目不存在累进区间数据';
  470. }
  471. //根据基数名称匹配累进库数据,标准化以免(())等不同导致不匹配
  472. const matchProgressiveData = progressiveData.find(item => cbAnalyzer.standar(item.name) === cbAnalyzer.standar(name));
  473. if (!matchProgressiveData) {
  474. throw `计算基数{${name}}不存在累进区间数据`;
  475. }
  476. // 将原始数据转换成方便处理的数据:[{feeRate: xx, min: 0, max: 200, minOpr: '(', maxOpr: ']'}]
  477. const progression = matchProgressiveData.progression.map(item => {
  478. // item.interval内容: eg (0,200]、[300,500) [1000,+)....
  479. const interval = cbAnalyzer.standar(item.interval);
  480. // ( => 大于 [ => 大于等于 ) => 小于 ] => 小于等于
  481. const minReg = /([\(\[])(\d+)/;
  482. const minMatch = minReg.exec(interval);
  483. if (!minMatch || !minMatch[1] || !minMatch[2]) {
  484. throw `计算基数{${name}}累进区间数据错误`;
  485. }
  486. const minOpr = minMatch[1];
  487. // 后台数据单位为万元,这里转为为元
  488. const min = parseFloat(minMatch[2]) * 10000;
  489. const maxReg = /[\,,]([\d\+]+)([\)\]])/;
  490. const maxMatch = maxReg.exec(interval);
  491. if (!maxMatch || !maxMatch[1] || !maxMatch[2]) {
  492. throw `计算基数{${name}}累进区间数据错误`;
  493. }
  494. const max = maxMatch[1] === '+' ? 'infinity' : parseFloat(maxMatch[1]) * 10000;
  495. const maxOpr = maxMatch[2];
  496. return {
  497. feeRate: item.feeRate,
  498. min,
  499. minOpr,
  500. max,
  501. maxOpr
  502. }
  503. });
  504. progression.sort((a, b) => a.min - b.min);
  505. // 基数所在区间
  506. const withinData = progression.find(item => {
  507. const oprMiddle = item.max === 'infinity' ? '+' : '';
  508. const oprLink = item.minOpr + oprMiddle + item.maxOpr;
  509. switch (oprLink) {
  510. case '()':
  511. return baseFee > item.min && baseFee < item.max;
  512. case '(]':
  513. return baseFee > item.min && baseFee <= item.max;
  514. case '[)':
  515. return baseFee >= item.min && baseFee < item.max;
  516. case '[]':
  517. return baseFee >= item.min && baseFee <= item.max;
  518. case '(+)':
  519. case '(+]':
  520. return baseFee > item.min;
  521. case '[+)':
  522. case '[+]':
  523. return baseFee >= item.min;
  524. default:
  525. return false;
  526. }
  527. });
  528. if (!withinData) {
  529. return 0;
  530. }
  531. // 累进计算
  532. let fee = 0;
  533. //累进之前的区间
  534. for (let i = 0; i < progression.indexOf(withinData); i++) {
  535. const perData = progression[i];
  536. fee += (perData.max - perData.min) * perData.feeRate * 0.01;
  537. }
  538. //累进所在区间
  539. fee += (baseFee - withinData.min) * withinData.feeRate * 0.01;
  540. return fee.toDecimal(decimalObj.bills.totalPrice);
  541. },
  542. //获取清单100章下的节点(只需要找最底层的,排除了底层,父项金额即排除了子项)
  543. //@param {Object}node(判断的节点,最底层清单节点)
  544. //@return {Boolean}
  545. withingOneHundred: function (node) {
  546. if (!node || node.sourceType !== calcBase.project.Bills.getSourceType() || node.source.children.length > 0) {
  547. return false;
  548. }
  549. //节点所属的清单固定行为第100章清单
  550. let belongFlags = cbTools.getBelongFlagList(node);
  551. return belongFlags.includes(fixedFlag.ONE_HUNDRED_BILLS);
  552. }
  553. };
  554. let baseFigureTemplate = {
  555. /*
  556. * 预算项目
  557. * */
  558. 'budget': {
  559. //{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  560. //取清单固定类别是“建筑安装工程”的定额建安费,但要扣除清单固定类别是“设备购置费”、及“专项费用”的定额建安费
  561. 'DEJZAZGCFBHSBZX': function (tender) {
  562. const deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE, fixedFlag.SPECIAL_COST];
  563. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, tender, 'rationCommon');
  564. },
  565. //{定额建筑安装工程(其中定额设备购置费按 40%计)} (定额建筑安装工程设备四十)
  566. //扣除设备购置费,再加上设备购置费的40%,扣除汇总算法不四舍五入,相当于汇总当中定额设备购置费就按照了40%计
  567. 'DEJZAZGCSBSS': function (tender) {
  568. const feeField = 'rationCommon';
  569. const deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  570. //建安费扣除定额设备购置费
  571. const afterDeductFee = cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, tender, feeField, false);
  572. //定额设备购置费
  573. let equipmentAcFee = cbTools.getBaseFee(deductFlags[0], tender, feeField);
  574. return (afterDeductFee + equipmentAcFee * 0.4).toDecimal(decimalObj.bills.totalPrice);
  575. },
  576. //{建筑安装工程费(不含安全生产费)}
  577. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“安全生产费”的金额
  578. 'JZAZGCFBHSC': function (tender) {
  579. //建安费扣除安全生产费
  580. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, [fixedFlag.SAFE_COST], tender, 'common');
  581. },
  582. //{建筑安装工程费(不含设备费)}
  583. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“设备购置费”的金额
  584. 'JZAZGCFBHSB': function (tender) {
  585. //建安费扣除设备费
  586. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, [fixedFlag.EQUIPMENT_ACQUISITION_FEE], tender, 'common');
  587. },
  588. //{建筑安装工程费}
  589. // 取清单固定类别是“建筑安装工程”的金额
  590. 'JZAZGCF': function (tender) {
  591. return cbTools.getBaseFee(calcBase.fixedFlag.CONSTRUCTION_INSTALL_FEE, tender, 'common');
  592. },
  593. //{土地使用及拆迁补偿费}
  594. // 取清单固定类别是“土地使用及拆迁补偿费”的金额
  595. 'TDSYJCQBCF': function (tender) {
  596. return cbTools.getBaseFee(calcBase.fixedFlag.LAND_USED_DEMOLITION, tender, 'common');
  597. },
  598. //{养护工程其他费}
  599. // 取清单固定类别是“养护工程其他费”的金额
  600. 'YHGCQTF': function (tender) {
  601. return cbTools.getBaseFee(calcBase.fixedFlag.MAINTENANCE_EXPENSES, tender, 'common');
  602. },
  603. //{预备费}
  604. // 取清单固定类别是“预备费”的金额
  605. 'YBF': function (tender) {
  606. return cbTools.getBaseFee(calcBase.fixedFlag.BUDGET_FEE, tender, 'common');
  607. },
  608. //{施工场地建设费}
  609. //使用累进办法计算,基数为{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  610. 'SGCDJSF': function (tender) {
  611. let baseFee = this['DEJZAZGCFBHSBZX'](tender);
  612. if (!tender) {
  613. calcBase.baseProgressiveFee = baseFee;
  614. }
  615. return calculateUtil.getProgressiveFee(baseFee, '施工场地建设费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  616. },
  617. //{养护单位(业主)管理费}
  618. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  619. 'YHDWYZGLF': function (tender) {
  620. let baseFee = this['DEJZAZGCSBSS'](tender);
  621. if (!tender) {
  622. calcBase.baseProgressiveFee = baseFee;
  623. }
  624. return calculateUtil.getProgressiveFee(baseFee, '养护单位(业主)管理费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  625. },
  626. //{信息化费}
  627. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  628. 'XXHF': function (tender) {
  629. let baseFee = this['DEJZAZGCSBSS'](tender);
  630. if (!tender) {
  631. calcBase.baseProgressiveFee = baseFee;
  632. }
  633. return calculateUtil.getProgressiveFee(baseFee, '信息化费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  634. },
  635. //{路线工程监理费}
  636. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  637. 'LXGCJLF': function (tender) {
  638. let baseFee = this['DEJZAZGCSBSS'](tender);
  639. if (!tender) {
  640. calcBase.baseProgressiveFee = baseFee;
  641. }
  642. return calculateUtil.getProgressiveFee(baseFee, '路线工程监理费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  643. },
  644. //{独立桥梁隧道工程监理费}
  645. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  646. 'QLSDGCJLF': function (tender) {
  647. let baseFee = this['DEJZAZGCSBSS'](tender);
  648. if (!tender) {
  649. calcBase.baseProgressiveFee = baseFee;
  650. }
  651. return calculateUtil.getProgressiveFee(baseFee, '独立桥梁隧道工程监理费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  652. },
  653. //{设计文件审查费}
  654. // 使用累进办法计算,不足3千按3千,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  655. 'SJWJSCF': function (tender) {
  656. let baseFee = this['DEJZAZGCSBSS'](tender);
  657. if (!tender) {
  658. calcBase.baseProgressiveFee = baseFee;
  659. }
  660. return calculateUtil.getProgressiveFee(baseFee, '设计文件审查费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  661. },
  662. //{路线勘察设计费}
  663. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  664. 'LXKCSJF': function (tender) {
  665. let baseFee = this['DEJZAZGCSBSS'](tender);
  666. if (!tender) {
  667. calcBase.baseProgressiveFee = baseFee;
  668. }
  669. return calculateUtil.getProgressiveFee(baseFee, '路线勘察设计费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  670. },
  671. //{独立桥梁隧道维修加固勘察设计费}
  672. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  673. 'QLSDKCSJF': function (tender) {
  674. let baseFee = this['DEJZAZGCSBSS'](tender);
  675. if (!tender) {
  676. calcBase.baseProgressiveFee = baseFee;
  677. }
  678. return calculateUtil.getProgressiveFee(baseFee, '独立桥梁隧道维修加固勘察设计费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  679. },
  680. //{招标代理及标底(最高投标限价)编制费} (招标代理及标底编制费ZBDLJBDBZF)
  681. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  682. 'ZBDLJBDBZF': function (tender) {
  683. let baseFee = this['DEJZAZGCSBSS'](tender);
  684. if (!tender) {
  685. calcBase.baseProgressiveFee = baseFee;
  686. }
  687. return calculateUtil.getProgressiveFee(baseFee, '招标代理及标底(最高投标限价)编制费', projectObj.project.property.progressiveInterval, decimalObj.bills.totalPrice, deficiency);
  688. },
  689. //{价差预备费}
  690. //以建筑安装工程费为基数
  691. 'JCYBF': function (tender) {
  692. //建筑安装工程费作为基数
  693. let installFee = this['JZAZGCF'](tender);
  694. //年造价增涨
  695. let costGrowthRate = calcBase.project.property.costGrowthRate ?
  696. calcBase.project.property.costGrowthRate : 0;
  697. //增涨计费年限
  698. let growthPeriod = projectObj.project.property.growthPeriod ?
  699. calcBase.project.property.growthPeriod : 0;
  700. //= P * [(1+i)^(n-1) -1]
  701. return (installFee * (Math.pow(1 + costGrowthRate, growthPeriod - 1) - 1)).toDecimal(decimalObj.bills.totalPrice);
  702. }
  703. },
  704. /*
  705. * 工程量清单项目(bills of quantities)
  706. * */
  707. 'boq': {
  708. //{各章清单合计}
  709. // 取清单固定类别是“第100章至700章清单”的金额
  710. 'GZQDHJ': function (tender) {
  711. let feeField = 'common',
  712. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  713. return cbTools.getBillsFee(calcBase.fixedFlag.ONE_SEVEN_BILLS, feeField, subFeeField);
  714. },
  715. //{专项暂定合计}
  716. // 汇总专项暂定列有值的清单的金额
  717. /* 'ZXZDHJ': function (tender) {
  718. let rst = 0,
  719. feeField = 'common',
  720. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  721. let billsData = calcBase.project.Bills.datas,
  722. filterData = billsData.filter(function (data) {
  723. return data.specialProvisional;
  724. });
  725. for (let data of filterData) {
  726. if (cbTools.isUnDef(data.feesIndex) || _.isEmpty(data.feesIndex) ||
  727. cbTools.isUnDef(data.feesIndex[feeField]) || cbTools.isUnDef(data.feesIndex[feeField][subFeeField])) {
  728. continue;
  729. }
  730. rst += data.feesIndex[feeField][subFeeField];
  731. }
  732. return rst.toDecimal(decimalObj.bills.totalPrice);
  733. }, */
  734. // 第100章至700章清单行的暂估合价
  735. 'ZXZDHJ': function (tender) {
  736. return cbTools.getBaseFee(calcBase.fixedFlag.ONE_SEVEN_BILLS, tender, 'estimate');
  737. },
  738. //{100章以外清单合计}
  739. // 取清单固定清单[第100章至700章清单]的金额,但扣除清单100章下的金额。
  740. // 如果是固定清单[第100章至700章清单]下100章以外清单引用此基数,要排除自身(目前只允许100章的清单使用,所以暂时不需要此判断)
  741. 'YBZYHQDHJ': function (tender) {
  742. let oneToSeven = cbTools.findNodeByFlag(fixedFlag.ONE_SEVEN_BILLS);
  743. if (!oneToSeven) {
  744. return 0;
  745. }
  746. //100-700章固定节点的所有子节点
  747. let allChildren = [];
  748. function getChildren(nodes) {
  749. allChildren = allChildren.concat(nodes);
  750. for (let node of nodes) {
  751. if (node.children.length > 0) {
  752. getChildren(node.children);
  753. }
  754. }
  755. }
  756. getChildren(oneToSeven.children);
  757. //扣除的节点:100章的节点[100-200)
  758. let deductNodes = allChildren.filter(cbTools.withingOneHundred);
  759. //计算金额
  760. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee';
  761. return projectObj.project.calcProgram.getTotalFee([oneToSeven], deductNodes, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  762. },
  763. // {定额建安费(不含定额设备购置费)} 汇总非设备类型的定额的定额建安费
  764. 'DEJAFBHDESBGZF': function (tender) {
  765. const feeField = 'rationCommon';
  766. const subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  767. const rations = projectObj.project.Ration.datas.filter(ration => !(ration.type === rationType.gljRation && ration.subType === gljType.EQUIPMENT));
  768. const summaryFee = rations.reduce((total, ration) => {
  769. const fee = cbTools.getFee(ration, feeField, subFeeField);
  770. return total += fee;
  771. }, 0);
  772. return summaryFee.toDecimal(decimalObj.bills.totalPrice);
  773. }
  774. }
  775. };
  776. //基数的值不是通过直接引用某清单节点获得的,则该基数的fixedBill为空,如价差、甲供、分包; class:分类,用于基数选择界面分类显示
  777. //基数本身不与清单节点关联、但是其由与清单关联的节点四则运算得到,则拥有字段multiRef: [flags...]
  778. /*
  779. * 基数的过滤filter ,根据这个配置最终可以转换成清单固定行可使用的相应基数
  780. * 筛选和过滤由pick决定
  781. * 控制基数可被哪些清单固定行下的节点使用 //挑选 pick === true
  782. * 控制基数不可被哪些清单固定行下的节点使用 //过滤 pick === false
  783. * */
  784. //暂时特殊处理专项费用需要引用{定额建筑安装工程费(其中定额设备购置费按40%计)}等跟专项费用父项有关联的基数:将fixedFlag设置为null
  785. let baseFigureMap = {
  786. /*
  787. * 预算项目
  788. * */
  789. 'budget': {
  790. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  791. '定额建筑安装工程费(不含定额设备购置费及专项费用)': {
  792. base: 'DEJZAZGCFBHSBZX',
  793. fixedFlag: null,
  794. filter: [fixedFlag.SPECIAL_COST, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  795. pick: true
  796. },
  797. '定额建筑安装工程费(其中定额设备购置费按40%计)': {
  798. base: 'DEJZAZGCSBSS',
  799. fixedFlag: null,
  800. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  801. pick: true
  802. },
  803. //只允许固定类别是“安全生产费”
  804. '建筑安装工程费(不含安全生产费)': {
  805. base: 'JZAZGCFBHSC',
  806. fixedFlag: null,
  807. filter: [fixedFlag.SAFE_COST],
  808. pick: true
  809. },
  810. //只允许固定类别是“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  811. '建筑安装工程费(不含设备费)': {
  812. base: 'JZAZGCFBHSB',
  813. fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE,
  814. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  815. pick: true
  816. },
  817. //只允许非固定类别是“建筑安装工程费”下的清单引用
  818. '建筑安装工程费': {
  819. base: 'JZAZGCF',
  820. fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE,
  821. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE],
  822. pick: false
  823. },
  824. //只允许非固定类别是“建筑安装工程费”、非固定类别是“土地使用及拆迁补偿费”下的清单引用
  825. '土地使用及拆迁补偿费': {
  826. base: 'TDSYJCQBCF',
  827. fixedFlag: fixedFlag.LAND_USED_DEMOLITION,
  828. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE, fixedFlag.LAND_USED_DEMOLITION],
  829. pick: false,
  830. },
  831. //只允许非固定类别是“建筑安装工程费”、非固定类别是“土地使用及拆迁补偿费”、非固定类别是“养护工程其他费”下的清单引用
  832. '养护工程其他费': {
  833. base: 'YHGCQTF',
  834. fixedFlag: fixedFlag.MAINTENANCE_EXPENSES,
  835. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  836. pick: false
  837. },
  838. //只允许非固定类别是“建筑安装工程费”、非固定类别是“土地使用及拆迁补偿费”、非固定类别是“养护工程其他费”、非固定类别是“预备费”下的清单引用。
  839. '预备费': {
  840. base: 'YBF',
  841. fixedFlag: fixedFlag.BUDGET_FEE,
  842. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES, fixedFlag.BUDGET_FEE],
  843. pick: false
  844. },
  845. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  846. '施工场地建设费': {
  847. base: 'SGCDJSF',
  848. fixedFlag: null,
  849. filter: [fixedFlag.SPECIAL_COST, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  850. pick: true
  851. },
  852. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  853. '养护单位(业主)管理费': {
  854. base: 'YHDWYZGLF',
  855. fixedFlag: null,
  856. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  857. pick: true
  858. },
  859. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  860. '信息化费': {
  861. base: 'XXHF',
  862. fixedFlag: null,
  863. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  864. pick: true
  865. },
  866. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  867. '路线工程监理费': {
  868. base: 'LXGCJLF',
  869. fixedFlag: null,
  870. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  871. pick: true
  872. },
  873. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  874. '独立桥梁隧道工程监理费': {
  875. base: 'QLSDGCJLF',
  876. fixedFlag: null,
  877. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  878. pick: true
  879. },
  880. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  881. '设计文件审查费': {
  882. base: 'SJWJSCF',
  883. fixedFlag: null,
  884. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  885. pick: true
  886. },
  887. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  888. '路线勘察设计费': {
  889. base: 'LXKCSJF',
  890. fixedFlag: null,
  891. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  892. pick: true
  893. },
  894. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  895. '独立桥梁隧道维修加固勘察设计费': {
  896. base: 'QLSDKCSJF',
  897. fixedFlag: null,
  898. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  899. pick: true
  900. },
  901. '招标代理及标底(最高投标限价)编制费': {
  902. base: 'ZBDLJBDBZF',
  903. fixedFlag: null,
  904. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  905. pick: true
  906. },
  907. //只允许固定类别是“价差预备费”的清单使用
  908. '价差预备费': {
  909. base: 'JCYBF',
  910. fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE,
  911. filter: [fixedFlag.SPREAD_BUDGET_FEE],
  912. pick: true
  913. }
  914. },
  915. /*
  916. * 工程量清单项目
  917. * */
  918. 'boq': {
  919. //仅允许用于固定类别是“第100章至700章清单”以外的清单
  920. '各章清单合计': {
  921. base: 'GZQDHJ',
  922. fixedFlag: fixedFlag.ONE_SEVEN_BILLS,
  923. filter: [fixedFlag.ONE_SEVEN_BILLS],
  924. pick: false
  925. },
  926. //仅允许用于固定类别是“第100章至700章清单”以外的清单
  927. '专项暂定合计': {
  928. base: 'ZXZDHJ',
  929. fixedFlag: null,
  930. filter: [fixedFlag.ONE_SEVEN_BILLS],
  931. pick: false
  932. },
  933. /*
  934. * 清单固定行[第100章至700章清单]下的[第100章清单]需要允许清单可使用基数{100章以外合计}
  935. * 因此{100章以外合计}不设置关联的清单固定行
  936. * */
  937. //仅允许用于固定类别为“100章清单”引用
  938. '100章以外清单合计': {
  939. base: 'YBZYHQDHJ',
  940. fixedFlag: null,
  941. filter: [fixedFlag.ONE_HUNDRED_BILLS],
  942. pick: true
  943. },
  944. // 各部分都能用
  945. '定额建安费(不含定额设备购置费)': {
  946. base: 'DEJAFBHDESBGZF',
  947. fixedFlag: null,
  948. filter: null,
  949. pick: true
  950. }
  951. }
  952. };
  953. //输入式分析器
  954. let cbAnalyzer = {
  955. standar: function (exp) {
  956. //去空格
  957. exp = exp.replace(/\s/g, '');
  958. //( to (
  959. exp = exp.replace(/(/g, '(');
  960. //)to )
  961. exp = exp.replace(/)/g, ')');
  962. //,to ,
  963. exp = exp.replace(/,/g, ',');
  964. //f to F
  965. exp = exp.replace(new RegExp('f', 'g'), 'F');
  966. return exp;
  967. },
  968. //输入合法性
  969. inputLegal: function (exp) {
  970. let ilegalRex = /[^0-9,\u4e00-\u9fa5,\+,\-,\/,\*,\(,\),.,{,},F,%]/g;
  971. return !ilegalRex.test(exp);
  972. },
  973. //基数合法性、存在性
  974. baseLegal: function (baseFigures, exp) {
  975. //保证中文表达式在{}里
  976. let cnExps = cbParser.getCN(exp);
  977. let expFigures = cbParser.getFigure(exp);
  978. if (cnExps.length !== expFigures.length) {
  979. throw '清单基数必须要用花括号{}括起来';
  980. return false;
  981. }
  982. for (let i = 0, len = cnExps.length; i < len; i++) {
  983. if (cnExps[i] !== expFigures[i]) {
  984. throw '清单基数必须要用花括号{}括起来'
  985. return false;
  986. }
  987. }
  988. //基数存在性
  989. for (let i = 0, len = expFigures.length; i < len; i++) {
  990. if (cbTools.isUnDef(baseFigures[expFigures[i]])) {
  991. throw `清单基数{${expFigures[i]}}不存在`;
  992. return false;
  993. }
  994. }
  995. return true;
  996. },
  997. //行引用合法性、存在性
  998. fLegal: function (items, exp) {
  999. //提取F标记
  1000. let fmArr = cbParser.getFMArr(exp);
  1001. //提取行引用
  1002. let fArr = cbParser.getFArr(exp);
  1003. if (fmArr.length !== fArr.length) {
  1004. return false;
  1005. }
  1006. //提取行数
  1007. let rArr = cbParser.getXNum(fArr);
  1008. if (fArr.length !== rArr.length) {
  1009. return false;
  1010. }
  1011. rArr = Array.from(new Set(rArr));
  1012. //判断合法性和存在性
  1013. for (let i = 0, len = rArr.length; i < len; i++) {
  1014. let idx = rArr[i] - 1;
  1015. if (cbTools.isUnDef(cbTools.getBillByRow(items, idx))) {
  1016. return false;
  1017. }
  1018. }
  1019. return true;
  1020. },
  1021. //循环计算
  1022. cycleCalc: function (node, baseFigures, exp) {
  1023. let stack = [];
  1024. if (node.sourceType !== calcBase.project.Bills.getSourceType()) {
  1025. return false;
  1026. }
  1027. //用于判断的起始清单ID
  1028. let sIDs = cbTools.getNodeIDs(cbTools.getParents(node));
  1029. let figureF = cbParser.getFigureF(cbParser.getFigure(exp), cbParser.getUID(cbParser.getFIDArr(exp)));
  1030. for (let i = 0, len = figureF.length; i < len; i++) {
  1031. let figure = figureF[i];
  1032. let billsIDs = [];
  1033. if (figure.type === 'base' && cbTools.isDef(baseFigures[figure.value])) {
  1034. //多重引用基数
  1035. let figureMultiRef = baseFigures[figure.value]['multiRef'];
  1036. let cycleCalcRef = baseFigures[figure.value]['cycleCalcRef'];
  1037. if (cbTools.isDef(figureMultiRef)) {
  1038. if (cbTools.isDef(cycleCalcRef)) {
  1039. figureMultiRef = cycleCalcRef;
  1040. }
  1041. for (let flag of figureMultiRef) {
  1042. let bills = cbTools.findBill(flag);
  1043. if (bills) {
  1044. billsIDs.push(bills.ID);
  1045. }
  1046. }
  1047. } else {
  1048. billsIDs = cbTools.isDef(baseFigures[figure.value]['fixedBill']) ? [baseFigures[figure.value]['fixedBill']['bill']['ID']] : [];
  1049. }
  1050. } else if (figure.type === 'id') {
  1051. let node = cbTools.getNodeByID(figure.value);
  1052. billsIDs = cbTools.isDef(node) ? [node.data.ID] : [];
  1053. }
  1054. if (cbTools.isDef(billsIDs) && billsIDs.length > 0 && isCycle(billsIDs)) {
  1055. console.log('循环计算');
  1056. calcBase.errMsg = '表达式出现循环计算';
  1057. return true;
  1058. }
  1059. }
  1060. return false;
  1061. function checkStack(stack, sIDs) {
  1062. //引用栈发现了初始引用
  1063. for (let i = 0, len = sIDs.length; i < len; i++) {
  1064. if (stack.indexOf(sIDs[i]) !== -1) {
  1065. return true;
  1066. }
  1067. }
  1068. return false;
  1069. }
  1070. function isCycle(billIDs) {
  1071. stack = Array.from(new Set(stack.concat(billIDs)));
  1072. if (checkStack(stack, sIDs)) {
  1073. return true;
  1074. }
  1075. for (let i = 0, len = billIDs.length; i < len; i++) {
  1076. let block = cbTools.getStackBlock(billIDs[i]);
  1077. if (block.length > 0) {
  1078. stack = Array.from(new Set(stack.concat(block)));
  1079. let cycleFlag = isCycle(block);
  1080. if (cycleFlag === true) {
  1081. return cycleFlag;
  1082. }
  1083. //return isCycle(block);
  1084. }
  1085. }
  1086. return false;
  1087. }
  1088. },
  1089. //四则运算合法性,控制不允许重复出现运算符,这里再判断一次,控制行引用只能F
  1090. arithmeticLegal: function (exp) {
  1091. let ilegalRex = /[\+,\-,\*,\/]{2}/g;
  1092. let rex2 = /[{]{2}/g;
  1093. let rex3 = /[}]{2}/g;
  1094. let rex4 = /[F]{2}/g;
  1095. let rex5 = /[.]{2}/g;
  1096. let rex6 = /[%]{2}/g;
  1097. return !ilegalRex.test(exp) && !rex2.test(exp) && !rex3.test(exp) && !rex4.test(exp) && !rex5.test(exp) && !rex6.test(exp);
  1098. },
  1099. //
  1100. legalExp: function (node) {
  1101. let exp = this.standar(node.data.userCalcBase);
  1102. if (!this.inputLegal(exp)) {
  1103. throw '表达式含有无效字符';
  1104. }
  1105. if (!this.arithmeticLegal(exp)) {
  1106. throw '表达式含有无效字符';
  1107. }
  1108. if (!this.baseLegal(cbTools.getValidFigures(node), exp)) {
  1109. throw '清单基数不合法';
  1110. }
  1111. if (!this.fLegal(calcBase.project.mainTree.items, exp)) {
  1112. throw '行引用不合法';
  1113. }
  1114. //转换成ID引用
  1115. exp = cbParser.toIDExpr(exp);
  1116. if (this.cycleCalc(node, calcBase.baseFigures, exp)) {
  1117. throw '出现循环计算';
  1118. }
  1119. return exp;
  1120. }
  1121. };
  1122. //输入式转换器
  1123. let cbParser = {
  1124. //获取标记F
  1125. getFMArr: function (exp) {
  1126. let fmRex = /F/g;
  1127. let fmArr = exp.match(fmRex);
  1128. return cbTools.isDef(fmArr) ? fmArr : [];
  1129. },
  1130. //获取行引用 eg: F10
  1131. getFArr: function (exp) {
  1132. let fRex = /F\d+\b/g;
  1133. let fArr = exp.match(fRex);
  1134. return cbTools.isDef(fArr) ? fArr : [];
  1135. },
  1136. //获取X+num eg: F10 10 @105 105
  1137. getXNum: function (arr) {
  1138. let rRex = /\d+/g;
  1139. let tempArr = [];
  1140. for (let i = 0, len = arr.length; i < len; i++) {
  1141. tempArr = tempArr.concat(arr[i].match(rRex));
  1142. }
  1143. return tempArr;
  1144. //let rArr = Array.from(new Set(tempArr));
  1145. //return rArr;
  1146. },
  1147. //获取uuid
  1148. getUID: function (arr) {
  1149. let rRex = /[\d,a-z,A-Z,-]{36}/g;
  1150. let tempArr = [];
  1151. for (let i = 0, len = arr.length; i < len; i++) {
  1152. tempArr = tempArr.concat(arr[i].match(rRex));
  1153. }
  1154. let rArr = Array.from(new Set(tempArr));
  1155. return rArr;
  1156. },
  1157. //获取ID引用
  1158. getFIDArr: function (exp) {
  1159. return scMathUtil.getFIDArr(exp); //统一前后端调用方法
  1160. },
  1161. //获取表达式中的中文式
  1162. getCN: function (expr) {
  1163. let cnRex = /\d*[\u4e00-\u9fa5]{1,}\({0,}[\u4e00-\u9fa5]{0,}\d*%*[\u4e00-\u9fa5]{0,}\){0,}[\u4e00-\u9fa5]{0,}/g;
  1164. return _.filter(expr.match(cnRex), function (data) {
  1165. return data
  1166. });
  1167. },
  1168. //获取表达式中的基数
  1169. getFigure: function (expr) {
  1170. let rst = [];
  1171. let rex = /\{([^}]*)\}/g;
  1172. let temp = expr.match(rex);
  1173. if (cbTools.isDef(temp)) {
  1174. for (let i = 0, len = temp.length; i < len; i++) {
  1175. rst.push(temp[i].replace(/[{,}]/g, ''));
  1176. }
  1177. }
  1178. return rst;
  1179. },
  1180. //获取表达式中的基数和ID引用
  1181. getFigureF: function (figures, fidArr) {
  1182. let rst = [];
  1183. for (let i = 0, len = figures.length; i < len; i++) {
  1184. let obj = Object.create(null);
  1185. obj.type = 'base';
  1186. obj.value = figures[i];
  1187. rst.push(obj);
  1188. }
  1189. for (let i = 0, len = fidArr.length; i < len; i++) {
  1190. let obj = Object.create(null);
  1191. obj.type = 'id';
  1192. obj.value = fidArr[i];
  1193. rst.push(obj);
  1194. }
  1195. return rst;
  1196. },
  1197. //表达式中的百分数转换成小数
  1198. percentToNum: function (exp) {
  1199. let rex = /[\+,\-,\*,\/]{1}\d+(\.\d+)?%[\u4e00-\u9fa5]{0}/g;
  1200. let percents = exp.match(rex);
  1201. let numRex = /\d+(\.\d+)?/g;
  1202. if (cbTools.isDef(percents)) {
  1203. for (let i = 0, len = percents.length; i < len; i++) {
  1204. let percentNum = percents[i].match(numRex),
  1205. oprtor = percents[i].replace(`${percentNum}%`, '');
  1206. if (cbTools.isDef(percentNum) && percentNum.length === 1) {
  1207. exp = exp.replace(new RegExp(`\\${percents[i]}`, 'g'), `${oprtor}${percentNum[0]/100}`);
  1208. }
  1209. }
  1210. }
  1211. return exp;
  1212. },
  1213. //将行引用转换成ID引用
  1214. toIDExpr: function (exp) {
  1215. let exps = [];
  1216. //获得行引用
  1217. let fArr = this.getFArr(exp);
  1218. for (let i = 0, len = fArr.length; i < len; i++) {
  1219. let r = this.getXNum([fArr[i]]);
  1220. if (r.length === 1) {
  1221. let node = cbTools.getBillByRow(calcBase.project.mainTree.items, r[0] - 1);
  1222. if (cbTools.isUnDef(node)) {
  1223. //continue;
  1224. calcBase.errMsg = '行引用错误';
  1225. throw '行引用错误';
  1226. }
  1227. exps.push({
  1228. orgExp: fArr[i],
  1229. newExp: '@' + node.data.ID
  1230. });
  1231. } else {
  1232. calcBase.errMsg = '行引用错误';
  1233. throw '行引用错误';
  1234. }
  1235. }
  1236. for (let i = 0, len = exps.length; i < len; i++) {
  1237. exp = exp.replace(new RegExp(`${exps[i].orgExp}\\b`, 'g'), exps[i].newExp);
  1238. }
  1239. return exp;
  1240. },
  1241. //将ID引用转换成行引用
  1242. toFExpr: function (exp, items) {
  1243. let nodeItems = items ? items : calcBase.project.mainTree.items;
  1244. let exps = [];
  1245. //获得ID引用
  1246. let fidArr = this.getFIDArr(exp);
  1247. for (let i = 0, len = fidArr.length; i < len; i++) {
  1248. let id = this.getUID([fidArr[i]]);
  1249. if (id.length === 1) {
  1250. let row = cbTools.getRowByID(nodeItems, id[0]);
  1251. if (cbTools.isUnDef(row)) {
  1252. continue;
  1253. }
  1254. exps.push({
  1255. orgExp: fidArr[i],
  1256. newExp: 'F' + row
  1257. });
  1258. }
  1259. }
  1260. for (let i = 0, len = exps.length; i < len; i++) {
  1261. exp = exp.replace(new RegExp(`${exps[i].orgExp}\\b`, 'g'), exps[i].newExp);
  1262. }
  1263. return exp;
  1264. },
  1265. //将表达式转换为可编译的表达式
  1266. toCompileExpr: function (v) {
  1267. if (v === '') {
  1268. return '$CBC.base(\'NONE\')';
  1269. }
  1270. //基数
  1271. let strs = _.uniq(this.getFigure(v));
  1272. let exps = [];
  1273. for (let i = 0, len = strs.length; i < len; i++) {
  1274. let exp = Object.create(null);
  1275. exp.orgExp = `{${strs[i]}}`;
  1276. exps.push(exp);
  1277. }
  1278. for (let i = 0, len = exps.length; i < len; i++) {
  1279. exps[i].compileExp = '$CBC.base(\'' + exps[i].orgExp + '\')';
  1280. let regStr = exps[i].orgExp.replace(/\(/g, '\\\(');
  1281. regStr = regStr.replace(/\)/g, '\\\)');
  1282. v = v.replace(new RegExp(regStr, 'g'), exps[i].compileExp);
  1283. }
  1284. //去{}
  1285. v = v.replace(/[{, },]/g, '');
  1286. //行引用
  1287. let fidArr = _.uniq(this.getFIDArr(v));
  1288. let fExps = [];
  1289. for (let i = 0, len = fidArr.length; i < len; i++) {
  1290. let fExp = Object.create(null);
  1291. fExp.orgExp = fidArr[i];
  1292. fExps.push(fExp);
  1293. }
  1294. for (let i = 0, len = fExps.length; i < len; i++) {
  1295. fExps[i].compileExp = '$CBC.ref(\'' + fExps[i].orgExp + '\')';
  1296. v = v.replace(new RegExp(fExps[i].orgExp, 'g'), fExps[i].compileExp);
  1297. }
  1298. return v;
  1299. }
  1300. };
  1301. let cbCalctor = {
  1302. //计算基数
  1303. base: function (figure) {
  1304. if (figure === 'NONE') {
  1305. return 0;
  1306. }
  1307. if (calcBase.project.property.valuationType === 'bill') { //预算
  1308. return baseFigureTemplate.budget[calcBase.baseFigures[figure]['base']]();
  1309. } else { //工程量清单
  1310. return baseFigureTemplate.boq[calcBase.baseFigures[figure]['base']]();
  1311. }
  1312. },
  1313. //调价后计算基数
  1314. tenderBase: function (figure) {
  1315. if (figure === 'NONE') {
  1316. return 0;
  1317. }
  1318. if (calcBase.project.property.valuationType === 'bill') { //预算
  1319. return baseFigureTemplate.budget[calcBase.baseFigures[figure]['base']](true);
  1320. } else { //工程量清单
  1321. return baseFigureTemplate.boq[calcBase.baseFigures[figure]['base']](true);
  1322. }
  1323. },
  1324. //ID引用
  1325. ref: function (fExp) {
  1326. let ID = cbParser.getUID([fExp]);
  1327. if (ID.length === 1) {
  1328. let node = cbTools.getNodeByID(ID[0]);
  1329. return cbTools.isDef(node) &&
  1330. cbTools.isDef(node.data.feesIndex) &&
  1331. cbTools.isDef(node.data.feesIndex.common) &&
  1332. cbTools.isDef(node.data.feesIndex.common.totalFee) ?
  1333. node.data.feesIndex.common.totalFee : 0;
  1334. }
  1335. return 0;
  1336. },
  1337. tenderRef: function (fExp) {
  1338. let ID = cbParser.getUID([fExp]);
  1339. if (ID.length === 1) {
  1340. let node = cbTools.getNodeByID(ID[0]);
  1341. return cbTools.isDef(node) &&
  1342. cbTools.isDef(node.data.feesIndex) &&
  1343. cbTools.isDef(node.data.feesIndex.common) &&
  1344. cbTools.isDef(node.data.feesIndex.common.tenderTotalFee) ?
  1345. node.data.feesIndex.common.tenderTotalFee : 0;
  1346. }
  1347. return 0;
  1348. }
  1349. };
  1350. let calcBase = {
  1351. // 累进基数中的基准值,报表需要使用这个中间数据,因此需要入库处理,这里作为暂存
  1352. baseProgressiveFee: 0,
  1353. //正在执行计算的节点
  1354. activeNode: null,
  1355. errMsg: '表达式不正确',
  1356. success: false,
  1357. //清单固定行
  1358. fixedFlag: null,
  1359. fixedBills: Object.create(null),
  1360. //清单基数
  1361. baseFigures: Object.create(null),
  1362. //清单固定行可用基数对应 {flag: Number, baseList: Array}
  1363. flagValidBase: Object.create(null),
  1364. //清单可选基数映射,分两类:组织措施项目:排除父项和计算的父项; 其他项目、规费、税金、工程造价,及新增部分:显示所有计算基数
  1365. baseFigureClass: Object.create(null),
  1366. //初始化
  1367. init: function (project) {
  1368. let me = this;
  1369. me.project = project;
  1370. me.fixedFlag = fixedFlag;
  1371. cbTools.setFixedBills(project, me.fixedBills, me.fixedFlag);
  1372. if (project.property.valuationType === 'bill') { //预算
  1373. me.baseFigures = baseFigureMap.budget;
  1374. } else { //工程量清单
  1375. me.baseFigures = baseFigureMap.boq;
  1376. }
  1377. cbTools.setBaseBills(me.baseFigures, me.fixedBills);
  1378. //设置清单固定行可用基数映射
  1379. cbTools.setValidBaseMapping(me.baseFigures, me.flagValidBase);
  1380. },
  1381. getBase: function (figure) {
  1382. return cbCalctor.base(figure);
  1383. },
  1384. getBaseByClass: function (node) {
  1385. return cbTools.getValidFigures(node);
  1386. },
  1387. calculate: function (node, reCalc = null) {
  1388. let me = calcBase,
  1389. $CBA = cbAnalyzer,
  1390. $CBP = cbParser,
  1391. $CBC = cbCalctor;
  1392. try {
  1393. me.activeNode = node;
  1394. me.success = false;
  1395. me.errMsg = '表达式不正确';
  1396. //分析输入式合法性
  1397. let exp = reCalc ?
  1398. cbTools.isDef(node.data.calcBase) ?
  1399. node.data.calcBase :
  1400. '' :
  1401. $CBA.legalExp(node);
  1402. if (!cbTools.isDef(exp)) {
  1403. throw '表达式不正确';
  1404. }
  1405. //输入式转换表达式
  1406. let compileExp = $CBP.toCompileExpr(exp);
  1407. //计算
  1408. let calcExp = $CBP.percentToNum(compileExp);
  1409. let calcBaseValue = eval(calcExp);
  1410. if (!cbTools.isNum(calcBaseValue)) {
  1411. throw '基数计算结果不为数值';
  1412. }
  1413. //调价
  1414. let tenderCalcExp = calcExp
  1415. .replace(new RegExp('base', 'g'), 'tenderBase')
  1416. .replace(new RegExp('ref', 'g'), 'tenderRef')
  1417. let tenderCalcBaseValue = eval(tenderCalcExp);
  1418. if (!cbTools.isNum(tenderCalcBaseValue)) {
  1419. throw '调价基数计算结果不为数值';
  1420. }
  1421. //存储
  1422. me.success = true;
  1423. node.updateData.calcBase = exp;
  1424. node.updateData.calcBaseValue = parseFloat(calcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1425. node.updateData.tenderCalcBaseValue = parseFloat(tenderCalcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1426. // progression来自overwrite里配置的全局变量
  1427. if (typeof progression !== 'undefined' && calculateUtil.isProgressive(exp, progression)) {
  1428. node.updateData.baseProgressiveFee = me.baseProgressiveFee;
  1429. }
  1430. node.changed = true;
  1431. } catch (err) {
  1432. console.log(err);
  1433. if (typeof err === 'object') {
  1434. err = '表达式不正确'
  1435. }
  1436. if (node) {
  1437. err = `第${node.serialNo() + 1}行${err}`;
  1438. }
  1439. alert(err);
  1440. }
  1441. }
  1442. };