calc_base.js 61 KB

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