calc_base.js 49 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174
  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. /*
  178. * {专项暂定合计}需要特殊处理,专项暂定有值的节点不可用此基数,否则会引起循环计算
  179. * */
  180. if (node.data.specialProvisional) {
  181. let filterMap = {};
  182. for (let baseName in calcBase.baseFigures) {
  183. if (baseName !== '专项暂定合计') {
  184. filterMap[baseName] = calcBase.baseFigures[baseName];
  185. }
  186. }
  187. return filterMap;
  188. } else {
  189. return calcBase.baseFigures;
  190. }
  191. },
  192. getBaseBill: function (node) {
  193. let calcBase = projectObj.project.calcBase;
  194. let parent = node.parent;
  195. if(this.isFlag(node.data) && (node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.SUB_ENGINERRING
  196. || node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.CONSTRUCTION_TECH)){
  197. return node;
  198. }
  199. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.CONSTRUCTION_ORGANIZATION){
  200. return node;
  201. }
  202. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.OTHER){
  203. return node;
  204. }
  205. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.CHARGE){
  206. return node;
  207. }
  208. else if(this.isFlag(node.data) && node.data.flagsIndex.fixed.flag === calcBase.fixedFlag.TAX){
  209. return node;
  210. }
  211. else {
  212. if(!parent){
  213. return node;
  214. }
  215. else {
  216. return this.getBaseBill(parent);
  217. }
  218. }
  219. },
  220. //获取清单(有基数计算)引用了的其他清单,(循环引用栈中的一块)
  221. getStackBlock: function (billID) {
  222. let tempBases = [], block = [];//存引用的清单ID
  223. let node = getBill(billID);
  224. if(!node){
  225. return tempBases;
  226. }
  227. else {
  228. //获取基数和行引用
  229. getBase(node);
  230. let bases = Array.from(new Set(tempBases));
  231. //根据基数和行引用获取清单ID
  232. for(let i = 0, len = bases.length; i < len; i++){
  233. //基数是跟清单直接关联的
  234. if(bases[i]['type'] === 'base' && cbTools.isDef(calcBase.baseFigures[bases[i]['value']])){
  235. let figureMultiRef= calcBase.baseFigures[bases[i]['value']]['multiRef'];
  236. let cycleCalcRef = calcBase.baseFigures[bases[i]['value']]['cycleCalcRef'];
  237. //重构后:
  238. if(cbTools.isDef(figureMultiRef)){
  239. if(cbTools.isDef(cycleCalcRef)){
  240. figureMultiRef = cycleCalcRef;
  241. }
  242. for(let flag of figureMultiRef){
  243. let bills = cbTools.findBill(flag);
  244. if(cbTools.isDef(bills)){
  245. block.push(bills.ID);
  246. }
  247. }
  248. } else if(cbTools.isDef(calcBase.baseFigures[bases[i]['value']]['fixedBill'])){
  249. block.push(calcBase.baseFigures[bases[i]['value']]['fixedBill']['bill']['ID']);
  250. }
  251. } else if(bases[i]['type'] === 'id'){
  252. let node = cbTools.getNodeByID(bases[i]['value']);
  253. if(cbTools.isDef(node)){
  254. block.push(node.data.ID);
  255. }
  256. }
  257. }
  258. return Array.from(new Set(block));
  259. }
  260. function getBase(node){
  261. if(node && node.children.length === 0){
  262. if(cbTools.isDef(node.data.calcBase) && node.data.calcBase !== ''){
  263. let figureF = cbParser.getFigureF(cbParser.getFigure(node.data.calcBase), cbParser.getUID(cbParser.getFIDArr(node.data.calcBase)));
  264. tempBases = tempBases.concat(figureF);
  265. }
  266. } else if(node && node.children.length > 0) {
  267. for(let i = 0, len = node.children.length; i < len; i++){
  268. getBase(node.children[i]);
  269. }
  270. }
  271. }
  272. function getBill(ID){
  273. let nodes = calcBase.project.mainTree.nodes;
  274. let node = nodes['id_' + ID];
  275. if(cbTools.isDef(node) && node.sourceType === calcBase.project.Bills.getSourceType()){
  276. return node;
  277. }
  278. return null;
  279. }
  280. },
  281. // 获取全部有公式的树节点清单。 CSL, 2018-01-05
  282. getFormulaNodes: function (needOrder = false) {
  283. // 给公式结点清单换照引用计算顺序排序。
  284. function orderFormulaNodes (nodesArr) {
  285. let orderArr = [];
  286. function recursionNode(nodes) {
  287. for (let node of nodes){
  288. if (orderArr.includes(node)) continue; // 已排过序的节点则跳过
  289. if (node.data.calcBase){
  290. let subNodes = cbTools.getNodesByExp(node, nodesArr);
  291. recursionNode(subNodes);
  292. };
  293. if (nodesArr.includes(node) && !orderArr.includes(node)) orderArr.push(node);
  294. };
  295. }
  296. recursionNode(nodesArr);
  297. return orderArr;
  298. };
  299. let nodes = [];
  300. for (let node of projectObj.project.mainTree.items){
  301. if (node.sourceType == ModuleNames.bills && node.data.calcBase && node.data.calcBase != '')
  302. nodes.push(node);
  303. };
  304. if (needOrder && nodes.length >= 2) return orderFormulaNodes(nodes)
  305. else return nodes;
  306. },
  307. // 刷新全部行引用的公式清单。 CSL, 2018-01-05
  308. refreshFormulaNodes: function () {
  309. try {
  310. let nodes = this.getFormulaNodes();
  311. if (nodes.length > 0) projectObj.mainController.refreshTreeNode(nodes);
  312. } catch (err) {
  313. alert('公式引用行号显示刷新失败:' + err.message);
  314. }
  315. },
  316. // 判断结点是否被其它结点的表达式引用。
  317. isUsedByFormula: function(node){
  318. let nodes = this.getFormulaNodes();
  319. if (nodes.length == 0) return false;
  320. let sID = '@' + node.data.ID;
  321. for (let node of nodes){
  322. if (node.data.calcBase.hasSubStr(sID)) return true;
  323. };
  324. },
  325. //获取清单节点的金额
  326. //@param {Number}fixedFlag(清单固定行类别) {String}feeField(外层金额字段: common) {String}subFeeField(子金额字段: totalFee)
  327. //@return {Number}
  328. getBillsFee: function(fixedFlag, feeField, subFeeField) {
  329. //固定清单类别与清单数据、关联基数的映射
  330. let fixedBills = calcBase.fixedBills[fixedFlag];
  331. if (this.isUnDef(fixedBills)) {
  332. return 0;
  333. }
  334. let bills = fixedBills.bill;
  335. if (this.isUnDef(bills)) {
  336. return 0;
  337. }
  338. if (this.isUnDef(bills.feesIndex) || _.isEmpty(bills.feesIndex)) {
  339. return 0;
  340. }
  341. return this.isDef(bills.feesIndex[feeField]) && this.isDef(bills.feesIndex[feeField][subFeeField]) ? bills.feesIndex[feeField][subFeeField] : 0;
  342. },
  343. //获取扣除固定项后的金额,扣除其节点后重新汇总
  344. //@param {Number}fixedFlag(基数取值固定行类别) {Array}deductFlags(扣除的固定类别组) {String}fullFeeField(完整的取费字段: 'rationCommon.totalFee')
  345. //@return {Number}
  346. getFeeWithDeduction: function (fixedFlag, deductFlags, fullFeeField) {
  347. let baseNode = this.findNodeByFlag(fixedFlag);
  348. if (!baseNode) {
  349. return 0;
  350. }
  351. //要扣除的节点
  352. let deductNodes = [];
  353. for (let deFlag of deductFlags) {
  354. let node = this.findNodeByFlag(deFlag);
  355. if (node) {
  356. deductNodes.push(node);
  357. }
  358. }
  359. return projectObj.project.calcProgram.getTotalFee([baseNode], deductNodes, fullFeeField);
  360. },
  361. //获取累进办法计算的金额
  362. //@param {Number}baseFee(相关基数金额) {String}name(使用累进计算的基数名称)
  363. //@return {Number}
  364. getProgressiveFee: function (baseFee, name) {
  365. let progressiveData = calcBase.project.property.progressiveInterval;
  366. if (!progressiveData) {
  367. throw '该项目不存在累进区间数据';
  368. }
  369. let matchData = _.find(progressiveData, function (data) {
  370. //根据基数名称匹配累进库数据,标准化以免(())等不同导致不匹配
  371. return cbAnalyzer.standar(data.name) === cbAnalyzer.standar(name);
  372. });
  373. if (!matchData) {
  374. //return 0;
  375. throw `计算基数{${name}}不存在累进区间数据`;
  376. }
  377. let progression = matchData.progression;
  378. //获取区间中的最小值(0, 10] = 0
  379. function getMin(intervalStr) {
  380. let str = cbAnalyzer.standar(intervalStr);
  381. let match = /\((\d+)?/g.exec(str);
  382. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null; //后台数据单位为万元,这里转为为元
  383. }
  384. //获取区间中的最大值(0, 10] = 10
  385. function getMax(intervalStr) {
  386. let str = cbAnalyzer.standar(intervalStr);
  387. let match = /[\,,,](\d+)?/g.exec(str);
  388. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null
  389. }
  390. //将累进区间进行排序
  391. progression.sort(function (a, b) {
  392. let aV = getMin(a.interval),
  393. bV = getMin(b.interval);
  394. if (aV > bV) {
  395. return 1;
  396. } else if (aV < bV) {
  397. return -1;
  398. }
  399. return 0;
  400. });
  401. //累进计算
  402. let fee = 0;
  403. //找到所在区间
  404. let within = _.find(progression, function (data) {
  405. let min = getMin(data.interval),
  406. max = getMax(data.interval);
  407. return min !== null && baseFee > min && (max !== null && baseFee <= max || max === null);
  408. });
  409. if (!within) {
  410. return 0;
  411. }
  412. //累进之前的区间
  413. for (let i = 0; i < progression.indexOf(within); i++) {
  414. let perData = progression[i],
  415. min = getMin(perData.interval),
  416. max = getMax(perData.interval);
  417. if (min !== null && max !== null) {
  418. fee += (max - min) * perData.feeRate * 0.01;
  419. }
  420. }
  421. //累进所在区间
  422. let min = getMin(within.interval);
  423. if (min !== null) {
  424. fee += (baseFee - min) * within.feeRate * 0.01;
  425. }
  426. return fee.toDecimal(decimalObj.bills.totalPrice);
  427. }
  428. };
  429. let baseFigureTemplate = {
  430. /*
  431. * 预算项目
  432. * */
  433. 'budget': {
  434. //{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  435. //取清单固定类别是“建筑安装工程”的定额建安费,但要扣除清单固定类别是“设备购置费”、及“专项费用”的定额建安费
  436. 'DEJZAZGCFBHSBZX': function (tender) {
  437. let fullFeeField = tender ? 'rationCommon.tenderTotalFee' : 'rationCommon.totalFee',
  438. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE, fixedFlag.SPECIAL_COST];
  439. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  440. },
  441. //{定额建筑安装工程(其中定额设备购置费按 40%计)} (定额建筑安装工程设备四十)
  442. //扣除设备购置费,再加上设备购置费的40%,扣除汇总算法不四舍五入,相当于汇总当中定额设备购置费就按照了40%计
  443. 'DEJZAZGCSBSS': function (tender) {
  444. let feeField = 'rationCommon',
  445. subFeeField = tender ? 'tenderTotalFee' : 'totalFee',
  446. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  447. //建安费扣除定额设备购置费
  448. let afterDeductFee = cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, `${feeField}.${subFeeField}`);
  449. //定额设备购置费
  450. let equipmentAcFee = cbTools.getBillsFee(deductFlags[0], feeField, subFeeField);
  451. return (afterDeductFee + equipmentAcFee * 0.4).toDecimal(decimalObj.bills.totalPrice);
  452. },
  453. //{建筑安装工程费(不含安全生产费)}
  454. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“安全生产费”的金额
  455. 'JZAZGCFBHSC': function (tender) {
  456. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  457. deductFlags = [fixedFlag.SAFE_COST];
  458. //建安费扣除安全生产费
  459. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  460. },
  461. //{建筑安装工程费(不含设备费)}
  462. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“设备购置费”的金额
  463. 'JZAZGCFBHSB': function (tender) {
  464. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  465. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  466. //建安费扣除设备费
  467. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  468. },
  469. //{建筑安装工程费}
  470. // 取清单固定类别是“建筑安装工程”的金额
  471. 'JZAZGCF': function (tender) {
  472. let feeField = 'common',
  473. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  474. return cbTools.getBillsFee(calcBase.fixedFlag.CONSTRUCTION_INSTALL_FEE, feeField, subFeeField);
  475. },
  476. //{土地使用及拆迁补偿费}
  477. // 取清单固定类别是“土地使用及拆迁补偿费”的金额
  478. 'TDSYJCQBCF': function (tender) {
  479. let feeField = 'common',
  480. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  481. return cbTools.getBillsFee(calcBase.fixedFlag.LAND_USED_DEMOLITION, feeField, subFeeField);
  482. },
  483. //{养护工程其他费}
  484. // 取清单固定类别是“养护工程其他费”的金额
  485. 'YHGCQTF': function (tender) {
  486. let feeField = 'common',
  487. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  488. return cbTools.getBillsFee(calcBase.fixedFlag.MAINTENANCE_EXPENSES, feeField, subFeeField);
  489. },
  490. //{预备费}
  491. // 取清单固定类别是“预备费”的金额
  492. 'YBF': function(tender) {
  493. let feeField = 'common',
  494. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  495. return cbTools.getBillsFee(calcBase.fixedFlag.BUDGET_FEE, feeField, subFeeField);
  496. },
  497. //{施工场地建设费}
  498. //使用累进办法计算,基数为{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  499. 'SGCDJSF': function (tender) {
  500. let baseFee = this['DEJZAZGCFBHSBZX'](tender);
  501. return cbTools.getProgressiveFee(baseFee, '施工场地建设费');
  502. },
  503. //{养护单位(业主)管理费}
  504. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  505. 'YHDWYZGLF': function (tender) {
  506. let baseFee = this['DEJZAZGCSBSS'](tender);
  507. return cbTools.getProgressiveFee(baseFee, '养护单位(业主)管理费');
  508. },
  509. //{信息化费}
  510. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  511. 'XXHF': function (tender) {
  512. let baseFee = this['DEJZAZGCSBSS'](tender);
  513. return cbTools.getProgressiveFee(baseFee, '信息化费');
  514. },
  515. //{路线工程监理费}
  516. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  517. 'LXGCJLF': function (tender) {
  518. let baseFee = this['DEJZAZGCSBSS'](tender),
  519. fee = cbTools.getProgressiveFee(baseFee, '路线工程监理费');
  520. return fee < 20000 ? 20000 : fee;
  521. },
  522. //{独立桥梁隧道工程监理费}
  523. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  524. 'QLSDGCJLF': function (tender) {
  525. let baseFee = this['DEJZAZGCSBSS'](tender),
  526. fee = cbTools.getProgressiveFee(baseFee, '独立桥梁隧道工程监理费');
  527. return fee < 20000 ? 20000 : fee;
  528. },
  529. //{设计文件审查费}
  530. // 使用累进办法计算,不足3千按3千,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  531. 'SJWJSCF': function (tender) {
  532. let baseFee = this['DEJZAZGCSBSS'](tender),
  533. fee = cbTools.getProgressiveFee(baseFee, '设计文件审查费');
  534. return fee < 3000 ? 3000 : fee;
  535. },
  536. //{路线勘察设计费}
  537. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  538. 'LXKCSJF': function (tender) {
  539. let baseFee = this['DEJZAZGCSBSS'](tender);
  540. return cbTools.getProgressiveFee(baseFee, '路线勘察设计费');
  541. },
  542. //{独立桥梁隧道维修加固勘察设计费}
  543. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  544. 'QLSDKCSJF': function (tender) {
  545. let baseFee = this['DEJZAZGCSBSS'](tender);
  546. return cbTools.getProgressiveFee(baseFee, '独立桥梁隧道维修加固勘察设计费');
  547. },
  548. //{招标代理及标底(最高投标限价)编制费} (招标代理及标底编制费ZBDLJBDBZF)
  549. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  550. 'ZBDLJBDBZF': function (tender) {
  551. let baseFee = this['DEJZAZGCSBSS'](tender);
  552. return cbTools.getProgressiveFee(baseFee, '招标代理及标底(最高投标限价)编制费');
  553. },
  554. //{价差预备费}
  555. //以建筑安装工程费为基数
  556. 'JCYBF': function (tender) {
  557. //建筑安装工程费作为基数
  558. let installFee = this['JZAZGCF'](tender);
  559. //年造价增涨
  560. let costGrowthRate = calcBase.project.property.costGrowthRate ?
  561. calcBase.project.property.costGrowthRate : 0;
  562. //增涨计费年限
  563. let growthPeriod = projectObj.project.property.growthPeriod ?
  564. calcBase.project.property.growthPeriod : 0;
  565. //= P * [(1+i)^(n-1) -1]
  566. return (installFee * (Math.pow(1 + costGrowthRate, growthPeriod - 1) - 1)).toDecimal(decimalObj.bills.totalPrice);
  567. }
  568. },
  569. /*
  570. * 工程量清单项目(bills of quantities)
  571. * */
  572. 'boq': {
  573. //{各章清单合计}
  574. // 取清单固定类别是“第100章至700章清单”的金额
  575. 'GZQDHJ': function (tender) {
  576. let feeField = 'common',
  577. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  578. return cbTools.getBillsFee(calcBase.fixedFlag.ONE_SEVEN_BILLS, feeField, subFeeField);
  579. },
  580. //{专项暂定合计}
  581. // 汇总专项暂定列有值的清单的金额
  582. 'ZXZDHJ': function (tender) {
  583. let rst = 0,
  584. feeField = 'common',
  585. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  586. let billsData = calcBase.project.Bills.datas,
  587. filterData = billsData.filter(function (data) {
  588. return data.specialProvisional;
  589. });
  590. for (let data of filterData) {
  591. if (cbTools.isUnDef(data.feesIndex) || _.isEmpty(data.feesIndex) ||
  592. cbTools.isUnDef(data.feesIndex[feeField]) || cbTools.isUnDef(data.feesIndex[feeField][subFeeField])) {
  593. continue;
  594. }
  595. rst += data.feesIndex[feeField][subFeeField];
  596. }
  597. return rst.toDecimal(decimalObj.bills.totalPrice);
  598. },
  599. //{100章以外清单合计}
  600. // 取清单固定类别是“第100章至700章清单”的金额,但扣除清单100章下的金额。
  601. 'YBZYHQDHJ': function (tender) {
  602. //获取清单100章下的节点(只需要找最底层的,排除了底层,父项金额即排除了子项)
  603. /*
  604. * 是否是100章下的节点
  605. * 判定规则:
  606. * 1.递归往上找父项,直至找到一个其父项没有编码的节点(基准节点)
  607. * 2.判断递归找到的节点编码规则:
  608. * ①其编码只有前三位是连续的数值 eg: 100s(属于) 101(属于) 10(不属于) 10001(不属于)
  609. * ②其前三位连续的数值编码在区间[100, 200)中
  610. * */
  611. function isWithin(node) {
  612. if (!node || node.sourceType !== calcBase.project.Bills.getSourceType() || node.source.children.length > 0) {
  613. return false;
  614. }
  615. //找判断基准节点
  616. while (node.parent && node.parent.data.code) {
  617. node = node.parent;
  618. }
  619. //判断编码规则
  620. let reg = /^\d{3,}/;
  621. if (cbTools.isUnDef(node.data.code)) {
  622. return false;
  623. }
  624. let matchCode = node.data.code.match(reg);
  625. return matchCode && matchCode[0].length === 3 && matchCode[0] >= 100 && matchCode[0] < 200;
  626. }
  627. let oneToSeven = cbTools.findNodeByFlag(fixedFlag.ONE_SEVEN_BILLS);
  628. if (!oneToSeven) {
  629. return 0;
  630. }
  631. //100-700章固定节点的所有子节点
  632. let allChildren = [];
  633. function getChildren(nodes) {
  634. allChildren = allChildren.concat(nodes);
  635. for (let node of nodes) {
  636. if (node.children.length > 0) {
  637. getChildren(node.children);
  638. }
  639. }
  640. }
  641. getChildren(oneToSeven.children);
  642. let deductNodes = allChildren.filter(isWithin);
  643. //计算金额
  644. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee';
  645. return projectObj.project.calcProgram.getTotalFee([oneToSeven], deductNodes, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  646. }
  647. }
  648. };
  649. let figureClassTemplate = {
  650. };
  651. //基数的值不是通过清单节点获得的,则该基数的fixedBill为空,如价差、甲供、分包; class:分类,用于基数选择界面分类显示
  652. //基数本身不与清单节点关联、但是其由与清单关联的节点四则运算得到,则拥有字段multiRef: [flags...]
  653. let baseFigureMap = {
  654. /*
  655. * 预算项目
  656. * */
  657. 'budget': {
  658. //与清单直接关联=======
  659. '定额建筑安装工程费(不含定额设备购置费及专项费用)': {base: 'DEJZAZGCFBHSBZX', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE}, //设备费及专项是其子节点,所以不需要用mulRef
  660. '定额建筑安装工程(其中定额设备购置费按40%计)': {base: 'DEJZAZGCSBSS', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  661. '建筑安装工程费(不含安全生产费)': {base: 'JZAZGCFBHSC', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  662. '建筑安装工程费(不含设备费)': {base: 'JZAZGCFBHSB', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  663. '建筑安装工程费': {base: 'JZAZGCF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  664. '土地使用及拆迁补偿费': {base: 'TDSYJCQBCF', fixedFlag: fixedFlag.LAND_USED_DEMOLITION},
  665. '养护工程其他费': {base: 'YHGCQTF', fixedFlag: fixedFlag.MAINTENANCE_EXPENSES},
  666. '预备费': {base: 'YBF', fixedFlag: fixedFlag.BUDGET_FEE},
  667. '施工场地建设费': {base: 'SGCDJSF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE},
  668. '养护单位(业主)管理费': {base: 'YHDWYZGLF',
  669. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  670. '信息化费': {base: 'XXHF',
  671. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  672. '路线工程监理费': {base: 'LXGCJLF',
  673. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  674. '独立桥梁隧道工程监理费': {base: 'QLSDGCJLF',
  675. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  676. '设计文件审查费': {base: 'SJWJSCF',
  677. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  678. '路线勘察设计费': {base: 'LXKCSJF',
  679. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  680. '独立桥梁隧道维修加固勘察设计费': {base: 'QLSDKCSJF',
  681. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  682. '招标代理及标底(最高投标限价)编制费': {base: 'ZBDLJBDBZF',
  683. multiRef: [fixedFlag.CONSTRUCTION_INSTALL_FEE]},
  684. '价差预备费': {base: 'JCYBF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE}
  685. },
  686. /*
  687. * 工程量清单项目
  688. * */
  689. 'boq': {
  690. '各章清单合计': {base: 'GZQDHJ', fixedFlag: fixedFlag.ONE_SEVEN_BILLS},
  691. '专项暂定合计': {base: 'ZXZDHJ', fixedFlag: null},
  692. '100章以外清单合计': {base: 'YBZYHQDHJ', fixedFlag: fixedFlag.ONE_SEVEN_BILLS},
  693. }
  694. };
  695. //输入式分析器
  696. let cbAnalyzer = {
  697. standar: function (exp) {
  698. //去空格
  699. exp = exp.replace(/\s/g, '');
  700. //( to (
  701. exp = exp.replace(/(/g, '(');
  702. //)to )
  703. exp = exp.replace(/)/g, ')');
  704. //,to ,
  705. exp = exp.replace(/,/g, ',');
  706. //f to F
  707. exp = exp.replace(new RegExp('f', 'g'), 'F');
  708. return exp;
  709. },
  710. //输入合法性
  711. inputLegal: function (exp) {
  712. let ilegalRex = /[^0-9,\u4e00-\u9fa5,\+,\-,\/,\*,\(,\),.,{,},F,%]/g;
  713. return !ilegalRex.test(exp);
  714. },
  715. //基数合法性、存在性
  716. baseLegal: function (baseFigures, exp) {
  717. //保证中文表达式在{}里
  718. let cnExps = cbParser.getCN(exp);
  719. let expFigures = cbParser.getFigure(exp);
  720. if(cnExps.length !== expFigures.length){
  721. throw '清单基数必须要用花括号{}括起来'
  722. return false;
  723. }
  724. for(let i = 0, len = cnExps.length; i < len; i++){
  725. if(cnExps[i] !== expFigures[i]){
  726. throw '清单基数必须要用花括号{}括起来'
  727. return false;
  728. }
  729. }
  730. //基数存在性
  731. for(let i = 0, len = expFigures.length; i < len; i++){
  732. if(cbTools.isUnDef(baseFigures[expFigures[i]])){
  733. throw `清单基数{${expFigures[i]}}不存在`;
  734. return false;
  735. }
  736. }
  737. return true;
  738. },
  739. //行引用合法性、存在性
  740. fLegal: function (items, exp) {
  741. //提取F标记
  742. let fmArr = cbParser.getFMArr(exp);
  743. //提取行引用
  744. let fArr = cbParser.getFArr(exp);
  745. if(fmArr.length !== fArr.length){
  746. return false;
  747. }
  748. //提取行数
  749. let rArr = cbParser.getXNum(fArr);
  750. if(fArr.length !== rArr.length){
  751. return false;
  752. }
  753. //判断合法性和存在性
  754. for(let i = 0, len = rArr.length; i < len; i++){
  755. let idx = rArr[i] - 1;
  756. if(cbTools.isUnDef(cbTools.getBillByRow(items, idx))){
  757. return false;
  758. }
  759. }
  760. return true;
  761. },
  762. //循环计算
  763. cycleCalc: function (node, baseFigures, exp) {
  764. let stack = [];
  765. if(node.sourceType !== calcBase.project.Bills.getSourceType()){
  766. return false;
  767. }
  768. //用于判断的起始清单ID
  769. let sIDs = cbTools.getNodeIDs(cbTools.getParents(node));
  770. let figureF = cbParser.getFigureF(cbParser.getFigure(exp), cbParser.getUID(cbParser.getFIDArr(exp)));
  771. for(let i = 0, len = figureF.length; i < len; i++){
  772. let figure = figureF[i];
  773. let billsIDs = [];
  774. if(figure.type === 'base' && cbTools.isDef(baseFigures[figure.value])){
  775. //重构后:
  776. //多重引用基数
  777. let figureMultiRef = baseFigures[figure.value]['multiRef'];
  778. let cycleCalcRef = baseFigures[figure.value]['cycleCalcRef'];
  779. if(cbTools.isDef(figureMultiRef)){
  780. if(cbTools.isDef(cycleCalcRef)){
  781. figureMultiRef = cycleCalcRef;
  782. }
  783. for(let flag of figureMultiRef){
  784. let bills = cbTools.findBill(flag);
  785. if(bills){
  786. billsIDs.push(bills.ID);
  787. }
  788. }
  789. } else {
  790. billsIDs = cbTools.isDef(baseFigures[figure.value]['fixedBill']) ? [baseFigures[figure.value]['fixedBill']['bill']['ID']] : [];
  791. }
  792. } else if(figure.type === 'id'){
  793. let node = cbTools.getNodeByID(figure.value);
  794. billsIDs = cbTools.isDef(node) ? [node.data.ID] : [];
  795. }
  796. if(cbTools.isDef(billsIDs) && billsIDs.length > 0 && isCycle(billsIDs)){
  797. console.log('循环计算');
  798. calcBase.errMsg = '表达式出现循环计算';
  799. return true;
  800. }
  801. }
  802. return false;
  803. function checkStack(stack, sIDs){
  804. //引用栈发现了初始引用
  805. for(let i = 0, len = sIDs.length; i < len; i++){
  806. if(stack.indexOf(sIDs[i]) !== -1){
  807. return true;
  808. }
  809. }
  810. return false;
  811. }
  812. function isCycle(billIDs){
  813. stack = Array.from(new Set(stack.concat(billIDs)));
  814. if (checkStack(stack, sIDs)) {
  815. return true;
  816. }
  817. for(let i = 0, len = billIDs.length; i < len; i++){
  818. let block = cbTools.getStackBlock(billIDs[i]);
  819. if(block.length > 0){
  820. stack = Array.from(new Set(stack.concat(block)));
  821. /* if(checkStack(stack, sIDs)){
  822. return true;
  823. }*/
  824. return isCycle(block);
  825. }
  826. }
  827. }
  828. },
  829. //四则运算合法性,控制不允许重复出现运算符,这里再判断一次,控制行引用只能F
  830. arithmeticLegal: function (exp) {
  831. let ilegalRex = /[\+,\-,\*,\/]{2}/g;
  832. let rex2 = /[{]{2}/g;
  833. let rex3 = /[}]{2}/g;
  834. let rex4 = /[F]{2}/g;
  835. let rex5 = /[.]{2}/g;
  836. let rex6 = /[%]{2}/g;
  837. return !ilegalRex.test(exp) && !rex2.test(exp) && !rex3.test(exp) && !rex4.test(exp) && !rex5.test(exp) && !rex6.test(exp);
  838. },
  839. //
  840. legalExp: function (node) {
  841. let exp = this.standar(node.data.userCalcBase);
  842. if(!this.inputLegal(exp)){
  843. throw '表达式含有无效字符';
  844. }
  845. if(!this.arithmeticLegal(exp)){
  846. throw '表达式含有无效字符';
  847. }
  848. if(!this.baseLegal(cbTools.getFigure(node), exp)){
  849. throw '清单基数不合法';
  850. }
  851. if(!this.fLegal(calcBase.project.mainTree.items, exp)){
  852. throw '行引用不合法';
  853. }
  854. //转换成ID引用
  855. exp = cbParser.toIDExpr(exp);
  856. if(this.cycleCalc(node, calcBase.baseFigures, exp)){
  857. throw '出现循环计算';
  858. }
  859. return exp;
  860. }
  861. };
  862. //输入式转换器
  863. let cbParser = {
  864. //获取标记F
  865. getFMArr: function (exp) {
  866. let fmRex = /F/g;
  867. let fmArr = exp.match(fmRex);
  868. return cbTools.isDef(fmArr) ? fmArr : [];
  869. },
  870. //获取行引用 eg: F10
  871. getFArr: function (exp) {
  872. let fRex = /F\d+/g;
  873. let fArr = exp.match(fRex);
  874. return cbTools.isDef(fArr) ? fArr : [];
  875. },
  876. //获取X+num eg: F10 10 @105 105
  877. getXNum: function (arr) {
  878. let rRex = /\d+/g;
  879. let tempArr = [];
  880. for(let i = 0, len = arr.length; i < len; i++){
  881. tempArr = tempArr.concat(arr[i].match(rRex));
  882. }
  883. let rArr = Array.from(new Set(tempArr));
  884. return rArr;
  885. },
  886. //获取uuid
  887. getUID: function (arr) {
  888. let rRex = /[\d,a-z,A-Z,-]{36}/g;
  889. let tempArr = [];
  890. for(let i = 0, len = arr.length; i < len; i++){
  891. tempArr = tempArr.concat(arr[i].match(rRex));
  892. }
  893. let rArr = Array.from(new Set(tempArr));
  894. return rArr;
  895. },
  896. //获取ID引用
  897. getFIDArr: function (exp) {
  898. let fidRex = /@[\d,a-z,A-Z,-]{36}/g;
  899. let fidArr = exp.match(fidRex);
  900. return cbTools.isDef(fidArr) ? fidArr : [];
  901. },
  902. //获取表达式中的中文式
  903. getCN: function(expr){
  904. let cnRex = /[\u4e00-\u9fa5]{1,}\({0,}[\u4e00-\u9fa5]{0,}\){0,}[\u4e00-\u9fa5]{0,}/g;
  905. return _.filter(expr.match(cnRex), function (data) {
  906. return data
  907. });
  908. },
  909. //获取表达式中的基数
  910. getFigure: function(expr){
  911. let rst = [];
  912. let rex = /\{([^}]*)\}/g;
  913. let temp = expr.match(rex);
  914. if(cbTools.isDef(temp)){
  915. for(let i = 0, len = temp.length; i < len; i++){
  916. rst.push(temp[i].replace(/[{,}]/g, ''));
  917. }
  918. }
  919. return rst;
  920. },
  921. //获取表达式中的基数和ID引用
  922. getFigureF: function (figures, fidArr) {
  923. let rst = [];
  924. for(let i = 0, len = figures.length; i < len; i++){
  925. let obj = Object.create(null);
  926. obj.type = 'base';
  927. obj.value = figures[i];
  928. rst.push(obj);
  929. }
  930. for(let i = 0, len = fidArr.length; i < len; i++){
  931. let obj = Object.create(null);
  932. obj.type = 'id';
  933. obj.value = fidArr[i];
  934. rst.push(obj);
  935. }
  936. return rst;
  937. },
  938. //表达式中的百分数转换成小数
  939. percentToNum: function (exp) {
  940. let rex = /\d+(\.\d+)?%/g;
  941. let percents = exp.match(rex);
  942. let numRex = /\d+(\.\d+)?/g;
  943. if(cbTools.isDef(percents)){
  944. for(let i = 0, len = percents.length; i < len; i++){
  945. let percentNum = percents[i].match(numRex);
  946. if(cbTools.isDef(percentNum) && percentNum.length === 1){
  947. exp = exp.replace(new RegExp(percents[i], 'g'), percentNum[0]/100);
  948. }
  949. }
  950. }
  951. return exp;
  952. },
  953. //将行引用转换成ID引用
  954. toIDExpr: function (exp) {
  955. let exps = [];
  956. //获得行引用
  957. let fArr = this.getFArr(exp);
  958. for(let i = 0, len = fArr.length; i < len; i++){
  959. let r = this.getXNum([fArr[i]]);
  960. if(r.length === 1){
  961. let node = cbTools.getBillByRow(calcBase.project.mainTree.items, r[0] - 1);
  962. if(cbTools.isUnDef(node)){
  963. //continue;
  964. calcBase.errMsg = '行引用错误';
  965. throw '行引用错误';
  966. }
  967. exps.push({orgExp: fArr[i], newExp: '@' + node.data.ID});
  968. }
  969. else {
  970. calcBase.errMsg = '行引用错误';
  971. throw '行引用错误';
  972. }
  973. }
  974. for(let i = 0, len = exps.length; i < len; i++){
  975. exp = exp.replace(new RegExp(exps[i].orgExp, 'g'), exps[i].newExp);
  976. }
  977. return exp;
  978. },
  979. //将ID引用转换成行引用
  980. toFExpr: function (exp) {
  981. let exps = [];
  982. //获得ID引用
  983. let fidArr = this.getFIDArr(exp);
  984. for(let i = 0, len = fidArr.length; i < len; i++){
  985. let id = this.getUID([fidArr[i]]);
  986. if(id.length === 1){
  987. let row = cbTools.getRowByID(calcBase.project.mainTree.items, id[0]);
  988. if(cbTools.isUnDef(row)){
  989. continue;
  990. }
  991. exps.push({orgExp: fidArr[i], newExp: 'F' + row});
  992. }
  993. }
  994. for(let i = 0, len = exps.length; i < len; i++){
  995. exp = exp.replace(new RegExp(exps[i].orgExp, 'g'), exps[i].newExp);
  996. }
  997. return exp;
  998. },
  999. //将表达式转换为可编译的表达式
  1000. toCompileExpr: function(v){
  1001. if(v === ''){
  1002. return '$CBC.base(\'NONE\')';
  1003. }
  1004. //基数
  1005. let strs = _.uniq(this.getFigure(v));
  1006. let exps = [];
  1007. for(let i = 0, len = strs.length; i < len; i++){
  1008. let exp = Object.create(null);
  1009. exp.orgExp = `{${strs[i]}}`;
  1010. exps.push(exp);
  1011. }
  1012. for(let i = 0, len = exps.length;i < len; i++){
  1013. exps[i].compileExp = '$CBC.base(\'' + exps[i].orgExp + '\')';
  1014. let regStr = exps[i].orgExp.replace(/\(/g, '\\\(');
  1015. regStr = regStr.replace(/\)/g, '\\\)');
  1016. v = v.replace(new RegExp(regStr, 'g'), exps[i].compileExp);
  1017. }
  1018. //去{}
  1019. v = v.replace(/[{, },]/g, '');
  1020. //行引用
  1021. let fidArr = this.getFIDArr(v);
  1022. let fExps = [];
  1023. for(let i = 0, len = fidArr.length; i < len; i++){
  1024. let fExp = Object.create(null);
  1025. fExp.orgExp = fidArr[i];
  1026. fExps.push(fExp);
  1027. }
  1028. for(let i = 0, len = fExps.length; i < len; i++){
  1029. fExps[i].compileExp = '$CBC.ref(\'' + fExps[i].orgExp + '\')';
  1030. v = v.replace(new RegExp(fExps[i].orgExp, 'g'), fExps[i].compileExp);
  1031. }
  1032. return v;
  1033. }
  1034. };
  1035. let cbCalctor = {
  1036. //计算基数
  1037. base: function (figure) {
  1038. if(figure === 'NONE'){
  1039. return 0;
  1040. }
  1041. if (calcBase.project.property.valuationType === 'bill') {//预算
  1042. return baseFigureTemplate.budget[calcBase.baseFigures[figure]['base']]();
  1043. } else {//工程量清单
  1044. return baseFigureTemplate.boq[calcBase.baseFigures[figure]['base']]();
  1045. }
  1046. },
  1047. //调价后计算基数
  1048. tenderBase: function (figure) {
  1049. if(figure === 'NONE'){
  1050. return 0;
  1051. }
  1052. if (calcBase.project.property.valuationType === 'bill') {//预算
  1053. return baseFigureTemplate.budget[calcBase.baseFigures[figure]['base']](true);
  1054. } else {//工程量清单
  1055. return baseFigureTemplate.boq[calcBase.baseFigures[figure]['base']](true);
  1056. }
  1057. },
  1058. //ID引用
  1059. ref: function (fExp) {
  1060. let ID = cbParser.getUID([fExp]);
  1061. if(ID.length === 1){
  1062. let node = cbTools.getNodeByID(ID[0]);
  1063. return cbTools.isDef(node) &&
  1064. cbTools.isDef(node.data.feesIndex) &&
  1065. cbTools.isDef(node.data.feesIndex.common) &&
  1066. cbTools.isDef(node.data.feesIndex.common.totalFee) ?
  1067. node.data.feesIndex.common.totalFee : 0;
  1068. }
  1069. return 0;
  1070. },
  1071. //计算
  1072. exec: function () {
  1073. }
  1074. };
  1075. let calcBase = {
  1076. errMsg: '表达式不正确',
  1077. success: false,
  1078. //清单固定行
  1079. fixedFlag: null,
  1080. fixedBills: Object.create(null),
  1081. //清单基数
  1082. baseFigures: Object.create(null),
  1083. //清单可选基数映射,分两类:组织措施项目:排除父项和计算的父项; 其他项目、规费、税金、工程造价,及新增部分:显示所有计算基数
  1084. baseFigureClass: Object.create(null),
  1085. //初始化
  1086. init: function (project) {
  1087. let me = this;
  1088. me.project = project;
  1089. me.fixedFlag = fixedFlag;
  1090. cbTools.setFixedBills(project, me.fixedBills, me.fixedFlag);
  1091. if (project.property.valuationType === 'bill') {//预算
  1092. me.baseFigures = baseFigureMap.budget;
  1093. } else {//工程量清单
  1094. me.baseFigures = baseFigureMap.boq;
  1095. }
  1096. cbTools.setBaseBills(me.baseFigures, me.fixedBills);
  1097. //cbTools.setBaseFigureClass(me.baseFigures, me.baseFigureClass);
  1098. },
  1099. getBase: function (figure) {
  1100. return cbCalctor.base(figure);
  1101. },
  1102. getBaseByClass: function (node) {
  1103. return cbTools.getFigure(node);
  1104. },
  1105. getBaseBill: function (node) {
  1106. return cbTools.getBaseBill(node);
  1107. },
  1108. calculate: function (node, reCalc = null) {
  1109. let me = calcBase,
  1110. $CBA = cbAnalyzer,
  1111. $CBP = cbParser,
  1112. $CBC = cbCalctor;
  1113. try {
  1114. me.success = false;
  1115. me.errMsg = '表达式不正确';
  1116. //分析输入式合法性
  1117. let exp = reCalc
  1118. ? cbTools.isDef(node.data.calcBase)
  1119. ? node.data.calcBase
  1120. : ''
  1121. : $CBA.legalExp(node);
  1122. if(!cbTools.isDef(exp)){
  1123. throw '表达式不正确';
  1124. }
  1125. //输入式转换表达式
  1126. let compileExp = $CBP.toCompileExpr(exp);
  1127. //计算
  1128. let calcExp = $CBP.percentToNum(compileExp);
  1129. let calcBaseValue = eval(calcExp);
  1130. if(!cbTools.isNum(calcBaseValue)){
  1131. throw '基数计算结果不为数值';
  1132. }
  1133. //调价
  1134. let tenderCalcExp = calcExp.replace(new RegExp('base', 'g'), 'tenderBase');
  1135. let tenderCalcBaseValue = eval(tenderCalcExp);
  1136. if(!cbTools.isNum(tenderCalcBaseValue)){
  1137. throw '调价基数计算结果不为数值';
  1138. }
  1139. //存储
  1140. me.success = true;
  1141. node.updateData.calcBase = exp;
  1142. node.updateData.calcBaseValue = parseFloat(calcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1143. node.updateData.tenderCalcBaseValue = parseFloat(tenderCalcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1144. node.changed = true;
  1145. }
  1146. catch (err){
  1147. if(typeof err === 'object'){
  1148. err = '表达式不正确'
  1149. }
  1150. if (node) {
  1151. err = `第${node.serialNo() + 1}行${err}`;
  1152. }
  1153. alert(err);
  1154. }
  1155. }
  1156. };