calc_base.js 56 KB

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