calc_base.js 56 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320
  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' && 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,
  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);
  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. //@param {Number}fixedFlag(清单固定行类别) {String}feeField(外层金额字段: common) {String}subFeeField(子金额字段: totalFee)
  384. //@return {Number}
  385. getBillsFee: function(fixedFlag, feeField, subFeeField) {
  386. //固定清单类别与清单数据、关联基数的映射
  387. let fixedBills = calcBase.fixedBills[fixedFlag];
  388. if (this.isUnDef(fixedBills)) {
  389. return 0;
  390. }
  391. let bills = fixedBills.bill;
  392. if (this.isUnDef(bills)) {
  393. return 0;
  394. }
  395. if (this.isUnDef(bills.feesIndex) || _.isEmpty(bills.feesIndex)) {
  396. return 0;
  397. }
  398. return this.isDef(bills.feesIndex[feeField]) &&
  399. this.isDef(bills.feesIndex[feeField][subFeeField]) && !isNaN(bills.feesIndex[feeField][subFeeField])
  400. ? bills.feesIndex[feeField][subFeeField]
  401. : 0;
  402. },
  403. //获取扣除固定项后的金额,扣除其节点后重新汇总
  404. //@param {Number}fixedFlag(基数取值固定行类别) {Array}deductFlags(扣除的固定类别组) {String}fullFeeField(完整的取费字段: 'rationCommon.totalFee')
  405. //@return {Number}
  406. getFeeWithDeduction: function (fixedFlag, deductFlags, fullFeeField) {
  407. let baseNode = this.findNodeByFlag(fixedFlag);
  408. if (!baseNode) {
  409. return 0;
  410. }
  411. //要扣除的节点
  412. let deductNodes = [];
  413. for (let deFlag of deductFlags) {
  414. let node = this.findNodeByFlag(deFlag);
  415. if (node) {
  416. deductNodes.push(node);
  417. }
  418. }
  419. return projectObj.project.calcProgram.getTotalFee([baseNode], deductNodes, fullFeeField);
  420. },
  421. //获取累进办法计算的金额
  422. //@param {Number}baseFee(相关基数金额) {String}name(使用累进计算的基数名称)
  423. //@return {Number}
  424. getProgressiveFee: function (baseFee, name) {
  425. let progressiveData = calcBase.project.property.progressiveInterval;
  426. if (!progressiveData) {
  427. throw '该项目不存在累进区间数据';
  428. }
  429. let matchData = _.find(progressiveData, function (data) {
  430. //根据基数名称匹配累进库数据,标准化以免(())等不同导致不匹配
  431. return cbAnalyzer.standar(data.name) === cbAnalyzer.standar(name);
  432. });
  433. if (!matchData) {
  434. //return 0;
  435. throw `计算基数{${name}}不存在累进区间数据`;
  436. }
  437. let progression = matchData.progression;
  438. //获取区间中的最小值(0, 10] = 0
  439. function getMin(intervalStr) {
  440. let str = cbAnalyzer.standar(intervalStr);
  441. let match = /\((\d+)?/g.exec(str);
  442. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null; //后台数据单位为万元,这里转为为元
  443. }
  444. //获取区间中的最大值(0, 10] = 10
  445. function getMax(intervalStr) {
  446. let str = cbAnalyzer.standar(intervalStr);
  447. let match = /[\,,,](\d+)?/g.exec(str);
  448. return typeof match[1] !== 'undefined' ? parseFloat(match[1]) * 10000 : null
  449. }
  450. //将累进区间进行排序
  451. progression.sort(function (a, b) {
  452. let aV = getMin(a.interval),
  453. bV = getMin(b.interval);
  454. if (aV > bV) {
  455. return 1;
  456. } else if (aV < bV) {
  457. return -1;
  458. }
  459. return 0;
  460. });
  461. //累进计算
  462. let fee = 0;
  463. //找到所在区间
  464. let within = _.find(progression, function (data) {
  465. let min = getMin(data.interval),
  466. max = getMax(data.interval);
  467. return min !== null && baseFee > min && (max !== null && baseFee <= max || max === null);
  468. });
  469. if (!within) {
  470. return 0;
  471. }
  472. //累进之前的区间
  473. for (let i = 0; i < progression.indexOf(within); i++) {
  474. let perData = progression[i],
  475. min = getMin(perData.interval),
  476. max = getMax(perData.interval);
  477. if (min !== null && max !== null) {
  478. fee += (max - min) * perData.feeRate * 0.01;
  479. }
  480. }
  481. //累进所在区间
  482. let min = getMin(within.interval);
  483. if (min !== null) {
  484. fee += (baseFee - min) * within.feeRate * 0.01;
  485. }
  486. return fee.toDecimal(decimalObj.bills.totalPrice);
  487. },
  488. //获取清单100章下的节点(只需要找最底层的,排除了底层,父项金额即排除了子项)
  489. //@param {Object}node(判断的节点,最底层清单节点)
  490. //@return {Boolean}
  491. withingOneHundred: function (node) {
  492. if (!node || node.sourceType !== calcBase.project.Bills.getSourceType() || node.source.children.length > 0) {
  493. return false;
  494. }
  495. //节点所属的清单固定行为第100章清单
  496. let belongFlags = cbTools.getBelongFlagList(node);
  497. return belongFlags.includes(fixedFlag.ONE_HUNDRED_BILLS);
  498. }
  499. };
  500. let baseFigureTemplate = {
  501. /*
  502. * 预算项目
  503. * */
  504. 'budget': {
  505. //{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  506. //取清单固定类别是“建筑安装工程”的定额建安费,但要扣除清单固定类别是“设备购置费”、及“专项费用”的定额建安费
  507. 'DEJZAZGCFBHSBZX': function (tender) {
  508. let fullFeeField = tender ? 'rationCommon.tenderTotalFee' : 'rationCommon.totalFee',
  509. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE, fixedFlag.SPECIAL_COST];
  510. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  511. },
  512. //{定额建筑安装工程(其中定额设备购置费按 40%计)} (定额建筑安装工程设备四十)
  513. //扣除设备购置费,再加上设备购置费的40%,扣除汇总算法不四舍五入,相当于汇总当中定额设备购置费就按照了40%计
  514. 'DEJZAZGCSBSS': function (tender) {
  515. let feeField = 'rationCommon',
  516. subFeeField = tender ? 'tenderTotalFee' : 'totalFee',
  517. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  518. //建安费扣除定额设备购置费
  519. let afterDeductFee = cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, `${feeField}.${subFeeField}`);
  520. //定额设备购置费
  521. let equipmentAcFee = cbTools.getBillsFee(deductFlags[0], feeField, subFeeField);
  522. return (afterDeductFee + equipmentAcFee * 0.4).toDecimal(decimalObj.bills.totalPrice);
  523. },
  524. //{建筑安装工程费(不含安全生产费)}
  525. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“安全生产费”的金额
  526. 'JZAZGCFBHSC': function (tender) {
  527. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  528. deductFlags = [fixedFlag.SAFE_COST];
  529. //建安费扣除安全生产费
  530. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  531. },
  532. //{建筑安装工程费(不含设备费)}
  533. // 取清单固定类别是“建筑安装工程”的金额,但要扣除清单固定类别是“设备购置费”的金额
  534. 'JZAZGCFBHSB': function (tender) {
  535. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee',
  536. deductFlags = [fixedFlag.EQUIPMENT_ACQUISITION_FEE];
  537. //建安费扣除设备费
  538. return cbTools.getFeeWithDeduction(fixedFlag.CONSTRUCTION_INSTALL_FEE, deductFlags, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  539. },
  540. //{建筑安装工程费}
  541. // 取清单固定类别是“建筑安装工程”的金额
  542. 'JZAZGCF': function (tender) {
  543. let feeField = 'common',
  544. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  545. return cbTools.getBillsFee(calcBase.fixedFlag.CONSTRUCTION_INSTALL_FEE, feeField, subFeeField);
  546. },
  547. //{土地使用及拆迁补偿费}
  548. // 取清单固定类别是“土地使用及拆迁补偿费”的金额
  549. 'TDSYJCQBCF': function (tender) {
  550. let feeField = 'common',
  551. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  552. return cbTools.getBillsFee(calcBase.fixedFlag.LAND_USED_DEMOLITION, feeField, subFeeField);
  553. },
  554. //{养护工程其他费}
  555. // 取清单固定类别是“养护工程其他费”的金额
  556. 'YHGCQTF': function (tender) {
  557. let feeField = 'common',
  558. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  559. return cbTools.getBillsFee(calcBase.fixedFlag.MAINTENANCE_EXPENSES, feeField, subFeeField);
  560. },
  561. //{预备费}
  562. // 取清单固定类别是“预备费”的金额
  563. 'YBF': function(tender) {
  564. let feeField = 'common',
  565. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  566. return cbTools.getBillsFee(calcBase.fixedFlag.BUDGET_FEE, feeField, subFeeField);
  567. },
  568. //{施工场地建设费}
  569. //使用累进办法计算,基数为{定额建筑安装工程费(不含定额设备购置费及专项费用)}
  570. 'SGCDJSF': function (tender) {
  571. let baseFee = this['DEJZAZGCFBHSBZX'](tender);
  572. return cbTools.getProgressiveFee(baseFee, '施工场地建设费');
  573. },
  574. //{养护单位(业主)管理费}
  575. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  576. 'YHDWYZGLF': function (tender) {
  577. let baseFee = this['DEJZAZGCSBSS'](tender);
  578. return cbTools.getProgressiveFee(baseFee, '养护单位(业主)管理费');
  579. },
  580. //{信息化费}
  581. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  582. 'XXHF': function (tender) {
  583. let baseFee = this['DEJZAZGCSBSS'](tender);
  584. return cbTools.getProgressiveFee(baseFee, '信息化费');
  585. },
  586. //{路线工程监理费}
  587. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  588. 'LXGCJLF': function (tender) {
  589. let baseFee = this['DEJZAZGCSBSS'](tender),
  590. fee = cbTools.getProgressiveFee(baseFee, '路线工程监理费');
  591. return fee > 0 && fee < 20000 ? 20000 : fee;
  592. },
  593. //{独立桥梁隧道工程监理费}
  594. //使用累进办法计算,不足2万按2万,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  595. 'QLSDGCJLF': function (tender) {
  596. let baseFee = this['DEJZAZGCSBSS'](tender),
  597. fee = cbTools.getProgressiveFee(baseFee, '独立桥梁隧道工程监理费');
  598. return fee > 0 && fee < 20000 ? 20000 : fee;
  599. },
  600. //{设计文件审查费}
  601. // 使用累进办法计算,不足3千按3千,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  602. 'SJWJSCF': function (tender) {
  603. let baseFee = this['DEJZAZGCSBSS'](tender),
  604. fee = cbTools.getProgressiveFee(baseFee, '设计文件审查费');
  605. return fee > 0 && fee < 3000 ? 3000 : fee;
  606. },
  607. //{路线勘察设计费}
  608. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  609. 'LXKCSJF': function (tender) {
  610. let baseFee = this['DEJZAZGCSBSS'](tender);
  611. return cbTools.getProgressiveFee(baseFee, '路线勘察设计费');
  612. },
  613. //{独立桥梁隧道维修加固勘察设计费}
  614. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  615. 'QLSDKCSJF': function (tender) {
  616. let baseFee = this['DEJZAZGCSBSS'](tender);
  617. return cbTools.getProgressiveFee(baseFee, '独立桥梁隧道维修加固勘察设计费');
  618. },
  619. //{招标代理及标底(最高投标限价)编制费} (招标代理及标底编制费ZBDLJBDBZF)
  620. // 使用累进办法计算,计算基数为{定额建筑安装工程(其中定额设备购置费按 40%计)}
  621. 'ZBDLJBDBZF': function (tender) {
  622. let baseFee = this['DEJZAZGCSBSS'](tender);
  623. return cbTools.getProgressiveFee(baseFee, '招标代理及标底(最高投标限价)编制费');
  624. },
  625. //{价差预备费}
  626. //以建筑安装工程费为基数
  627. 'JCYBF': function (tender) {
  628. //建筑安装工程费作为基数
  629. let installFee = this['JZAZGCF'](tender);
  630. //年造价增涨
  631. let costGrowthRate = calcBase.project.property.costGrowthRate ?
  632. calcBase.project.property.costGrowthRate : 0;
  633. //增涨计费年限
  634. let growthPeriod = projectObj.project.property.growthPeriod ?
  635. calcBase.project.property.growthPeriod : 0;
  636. //= P * [(1+i)^(n-1) -1]
  637. return (installFee * (Math.pow(1 + costGrowthRate, growthPeriod - 1) - 1)).toDecimal(decimalObj.bills.totalPrice);
  638. }
  639. },
  640. /*
  641. * 工程量清单项目(bills of quantities)
  642. * */
  643. 'boq': {
  644. //{各章清单合计}
  645. // 取清单固定类别是“第100章至700章清单”的金额
  646. 'GZQDHJ': function (tender) {
  647. let feeField = 'common',
  648. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  649. return cbTools.getBillsFee(calcBase.fixedFlag.ONE_SEVEN_BILLS, feeField, subFeeField);
  650. },
  651. //{专项暂定合计}
  652. // 汇总专项暂定列有值的清单的金额
  653. 'ZXZDHJ': function (tender) {
  654. let rst = 0,
  655. feeField = 'common',
  656. subFeeField = tender ? 'tenderTotalFee' : 'totalFee';
  657. let billsData = calcBase.project.Bills.datas,
  658. filterData = billsData.filter(function (data) {
  659. return data.specialProvisional;
  660. });
  661. for (let data of filterData) {
  662. if (cbTools.isUnDef(data.feesIndex) || _.isEmpty(data.feesIndex) ||
  663. cbTools.isUnDef(data.feesIndex[feeField]) || cbTools.isUnDef(data.feesIndex[feeField][subFeeField])) {
  664. continue;
  665. }
  666. rst += data.feesIndex[feeField][subFeeField];
  667. }
  668. return rst.toDecimal(decimalObj.bills.totalPrice);
  669. },
  670. //{100章以外清单合计}
  671. // 取清单固定清单[第100章至700章清单]的金额,但扣除清单100章下的金额。
  672. // 如果是固定清单[第100章至700章清单]下100章以外清单引用此基数,要排除自身(目前只允许100章的清单使用,所以暂时不需要此判断)
  673. 'YBZYHQDHJ': function (tender) {
  674. let oneToSeven = cbTools.findNodeByFlag(fixedFlag.ONE_SEVEN_BILLS);
  675. if (!oneToSeven) {
  676. return 0;
  677. }
  678. //100-700章固定节点的所有子节点
  679. let allChildren = [];
  680. function getChildren(nodes) {
  681. allChildren = allChildren.concat(nodes);
  682. for (let node of nodes) {
  683. if (node.children.length > 0) {
  684. getChildren(node.children);
  685. }
  686. }
  687. }
  688. getChildren(oneToSeven.children);
  689. //扣除的节点:100章的节点[100-200)
  690. let deductNodes = allChildren.filter(cbTools.withingOneHundred);
  691. //计算金额
  692. let fullFeeField = tender ? 'common.tenderTotalFee' : 'common.totalFee';
  693. return projectObj.project.calcProgram.getTotalFee([oneToSeven], deductNodes, fullFeeField).toDecimal(decimalObj.bills.totalPrice);
  694. }
  695. }
  696. };
  697. //基数的值不是通过直接引用某清单节点获得的,则该基数的fixedBill为空,如价差、甲供、分包; class:分类,用于基数选择界面分类显示
  698. //基数本身不与清单节点关联、但是其由与清单关联的节点四则运算得到,则拥有字段multiRef: [flags...]
  699. /*
  700. * 基数的过滤filter ,根据这个配置最终可以转换成清单固定行可使用的相应基数
  701. * 筛选和过滤由pick决定
  702. * 控制基数可被哪些清单固定行下的节点使用 //挑选 pick === true
  703. * 控制基数不可被哪些清单固定行下的节点使用 //过滤 pick === false
  704. * */
  705. //暂时特殊处理专项费用需要引用{定额建筑安装工程费(其中定额设备购置费按40%计)}等跟专项费用父项有关联的基数:将fixedFlag设置为null
  706. let baseFigureMap = {
  707. /*
  708. * 预算项目
  709. * */
  710. 'budget': {
  711. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  712. '定额建筑安装工程费(不含定额设备购置费及专项费用)': {
  713. base: 'DEJZAZGCFBHSBZX', fixedFlag: null,
  714. filter: [fixedFlag.SPECIAL_COST, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  715. pick: true
  716. },
  717. '定额建筑安装工程费(其中定额设备购置费按40%计)': {
  718. base: 'DEJZAZGCSBSS', fixedFlag: null,
  719. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  720. pick: true
  721. },
  722. //只允许固定类别是“安全生产费”
  723. '建筑安装工程费(不含安全生产费)': {
  724. base: 'JZAZGCFBHSC', fixedFlag: null,
  725. filter: [fixedFlag.SAFE_COST],
  726. pick: true
  727. },
  728. //只允许固定类别是“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  729. '建筑安装工程费(不含设备费)': {
  730. base: 'JZAZGCFBHSB', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE,
  731. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  732. pick: true
  733. },
  734. //只允许非固定类别是“建筑安装工程费”下的清单引用
  735. '建筑安装工程费': {
  736. base: 'JZAZGCF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE,
  737. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE],
  738. pick: false
  739. },
  740. //只允许非固定类别是“建筑安装工程费”、非固定类别是“土地使用及拆迁补偿费”下的清单引用
  741. '土地使用及拆迁补偿费': {
  742. base: 'TDSYJCQBCF', fixedFlag: fixedFlag.LAND_USED_DEMOLITION,
  743. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE, fixedFlag.LAND_USED_DEMOLITION],
  744. pick: false,
  745. },
  746. //只允许非固定类别是“建筑安装工程费”、非固定类别是“土地使用及拆迁补偿费”、非固定类别是“养护工程其他费”下的清单引用
  747. '养护工程其他费': {
  748. base: 'YHGCQTF', fixedFlag: fixedFlag.MAINTENANCE_EXPENSES,
  749. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  750. pick: false
  751. },
  752. //只允许非固定类别是“建筑安装工程费”、非固定类别是“土地使用及拆迁补偿费”、非固定类别是“养护工程其他费”、非固定类别是“预备费”下的清单引用。
  753. '预备费': {
  754. base: 'YBF', fixedFlag: fixedFlag.BUDGET_FEE,
  755. filter: [fixedFlag.CONSTRUCTION_INSTALL_FEE, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES, fixedFlag.BUDGET_FEE],
  756. pick: false
  757. },
  758. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  759. '施工场地建设费': {
  760. base: 'SGCDJSF', fixedFlag: null,
  761. filter: [fixedFlag.SPECIAL_COST, fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  762. pick: true
  763. },
  764. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  765. '养护单位(业主)管理费': {
  766. base: 'YHDWYZGLF', fixedFlag: null,
  767. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  768. pick: true
  769. },
  770. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  771. '信息化费': {
  772. base: 'XXHF', fixedFlag: null,
  773. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  774. pick: true
  775. },
  776. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  777. '路线工程监理费': {
  778. base: 'LXGCJLF', fixedFlag: null,
  779. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  780. pick: true
  781. },
  782. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  783. '独立桥梁隧道工程监理费': {
  784. base: 'QLSDGCJLF', fixedFlag: null,
  785. filter: [ fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  786. pick: true
  787. },
  788. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  789. '设计文件审查费': {
  790. base: 'SJWJSCF', fixedFlag: null,
  791. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  792. pick: true
  793. },
  794. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  795. '路线勘察设计费': {
  796. base: 'LXKCSJF', fixedFlag: null,
  797. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  798. pick: true
  799. },
  800. //只允许固定类别是“专项费用”、“土地使用及拆迁补偿费“、“养护工程其他费”下的清单使用
  801. '独立桥梁隧道维修加固勘察设计费': {
  802. base: 'QLSDKCSJF', fixedFlag: null,
  803. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  804. pick: true
  805. },
  806. '招标代理及标底(最高投标限价)编制费': {
  807. base: 'ZBDLJBDBZF', fixedFlag: null,
  808. filter: [fixedFlag.LAND_USED_DEMOLITION, fixedFlag.MAINTENANCE_EXPENSES],
  809. pick: true
  810. },
  811. //只允许固定类别是“价差预备费”的清单使用
  812. '价差预备费': {
  813. base: 'JCYBF', fixedFlag: fixedFlag.CONSTRUCTION_INSTALL_FEE,
  814. filter: [fixedFlag.SPREAD_BUDGET_FEE],
  815. pick: true
  816. }
  817. },
  818. /*
  819. * 工程量清单项目
  820. * */
  821. 'boq': {
  822. //仅允许用于固定类别是“第100章至700章清单”以外的清单
  823. '各章清单合计': {base: 'GZQDHJ', fixedFlag: fixedFlag.ONE_SEVEN_BILLS,
  824. filter: [fixedFlag.ONE_SEVEN_BILLS],
  825. pick: false
  826. },
  827. //仅允许用于固定类别是“第100章至700章清单”以外的清单
  828. '专项暂定合计': {base: 'ZXZDHJ', fixedFlag: null,
  829. filter: [fixedFlag.ONE_SEVEN_BILLS],
  830. pick: false
  831. },
  832. /*
  833. * 清单固定行[第100章至700章清单]下的[第100章清单]需要允许清单可使用基数{100章以外合计}
  834. * 因此{100章以外合计}不设置关联的清单固定行
  835. * */
  836. //仅允许用于固定类别为“100章清单”引用
  837. '100章以外清单合计': {base: 'YBZYHQDHJ', fixedFlag: null,
  838. filter: [fixedFlag.ONE_HUNDRED_BILLS],
  839. pick: true
  840. },
  841. }
  842. };
  843. //输入式分析器
  844. let cbAnalyzer = {
  845. standar: function (exp) {
  846. //去空格
  847. exp = exp.replace(/\s/g, '');
  848. //( to (
  849. exp = exp.replace(/(/g, '(');
  850. //)to )
  851. exp = exp.replace(/)/g, ')');
  852. //,to ,
  853. exp = exp.replace(/,/g, ',');
  854. //f to F
  855. exp = exp.replace(new RegExp('f', 'g'), 'F');
  856. return exp;
  857. },
  858. //输入合法性
  859. inputLegal: function (exp) {
  860. let ilegalRex = /[^0-9,\u4e00-\u9fa5,\+,\-,\/,\*,\(,\),.,{,},F,%]/g;
  861. return !ilegalRex.test(exp);
  862. },
  863. //基数合法性、存在性
  864. baseLegal: function (baseFigures, exp) {
  865. //保证中文表达式在{}里
  866. let cnExps = cbParser.getCN(exp);
  867. let expFigures = cbParser.getFigure(exp);
  868. if(cnExps.length !== expFigures.length){
  869. throw '清单基数必须要用花括号{}括起来';
  870. return false;
  871. }
  872. for(let i = 0, len = cnExps.length; i < len; i++){
  873. if(cnExps[i] !== expFigures[i]){
  874. throw '清单基数必须要用花括号{}括起来'
  875. return false;
  876. }
  877. }
  878. //基数存在性
  879. for(let i = 0, len = expFigures.length; i < len; i++){
  880. if(cbTools.isUnDef(baseFigures[expFigures[i]])){
  881. throw `清单基数{${expFigures[i]}}不存在`;
  882. return false;
  883. }
  884. }
  885. return true;
  886. },
  887. //行引用合法性、存在性
  888. fLegal: function (items, exp) {
  889. //提取F标记
  890. let fmArr = cbParser.getFMArr(exp);
  891. //提取行引用
  892. let fArr = cbParser.getFArr(exp);
  893. if(fmArr.length !== fArr.length){
  894. return false;
  895. }
  896. //提取行数
  897. let rArr = cbParser.getXNum(fArr);
  898. if(fArr.length !== rArr.length){
  899. return false;
  900. }
  901. rArr = Array.from(new Set(rArr));
  902. //判断合法性和存在性
  903. for(let i = 0, len = rArr.length; i < len; i++){
  904. let idx = rArr[i] - 1;
  905. if(cbTools.isUnDef(cbTools.getBillByRow(items, idx))){
  906. return false;
  907. }
  908. }
  909. return true;
  910. },
  911. //循环计算
  912. cycleCalc: function (node, baseFigures, exp) {
  913. let stack = [];
  914. if(node.sourceType !== calcBase.project.Bills.getSourceType()){
  915. return false;
  916. }
  917. //用于判断的起始清单ID
  918. let sIDs = cbTools.getNodeIDs(cbTools.getParents(node));
  919. let figureF = cbParser.getFigureF(cbParser.getFigure(exp), cbParser.getUID(cbParser.getFIDArr(exp)));
  920. for(let i = 0, len = figureF.length; i < len; i++){
  921. let figure = figureF[i];
  922. let billsIDs = [];
  923. if(figure.type === 'base' && cbTools.isDef(baseFigures[figure.value])){
  924. //多重引用基数
  925. let figureMultiRef = baseFigures[figure.value]['multiRef'];
  926. let cycleCalcRef = baseFigures[figure.value]['cycleCalcRef'];
  927. if(cbTools.isDef(figureMultiRef)){
  928. if(cbTools.isDef(cycleCalcRef)){
  929. figureMultiRef = cycleCalcRef;
  930. }
  931. for(let flag of figureMultiRef){
  932. let bills = cbTools.findBill(flag);
  933. if(bills){
  934. billsIDs.push(bills.ID);
  935. }
  936. }
  937. } else {
  938. billsIDs = cbTools.isDef(baseFigures[figure.value]['fixedBill']) ? [baseFigures[figure.value]['fixedBill']['bill']['ID']] : [];
  939. }
  940. } else if(figure.type === 'id'){
  941. let node = cbTools.getNodeByID(figure.value);
  942. billsIDs = cbTools.isDef(node) ? [node.data.ID] : [];
  943. }
  944. if(cbTools.isDef(billsIDs) && billsIDs.length > 0 && isCycle(billsIDs)){
  945. console.log('循环计算');
  946. calcBase.errMsg = '表达式出现循环计算';
  947. return true;
  948. }
  949. }
  950. return false;
  951. function checkStack(stack, sIDs){
  952. //引用栈发现了初始引用
  953. for(let i = 0, len = sIDs.length; i < len; i++){
  954. if(stack.indexOf(sIDs[i]) !== -1){
  955. return true;
  956. }
  957. }
  958. return false;
  959. }
  960. function isCycle(billIDs){
  961. stack = Array.from(new Set(stack.concat(billIDs)));
  962. if (checkStack(stack, sIDs)) {
  963. return true;
  964. }
  965. for(let i = 0, len = billIDs.length; i < len; i++){
  966. let block = cbTools.getStackBlock(billIDs[i]);
  967. if(block.length > 0){
  968. stack = Array.from(new Set(stack.concat(block)));
  969. let cycleFlag = isCycle(block);
  970. if (cycleFlag === true) {
  971. return cycleFlag;
  972. }
  973. //return isCycle(block);
  974. }
  975. }
  976. return false;
  977. }
  978. },
  979. //四则运算合法性,控制不允许重复出现运算符,这里再判断一次,控制行引用只能F
  980. arithmeticLegal: function (exp) {
  981. let ilegalRex = /[\+,\-,\*,\/]{2}/g;
  982. let rex2 = /[{]{2}/g;
  983. let rex3 = /[}]{2}/g;
  984. let rex4 = /[F]{2}/g;
  985. let rex5 = /[.]{2}/g;
  986. let rex6 = /[%]{2}/g;
  987. return !ilegalRex.test(exp) && !rex2.test(exp) && !rex3.test(exp) && !rex4.test(exp) && !rex5.test(exp) && !rex6.test(exp);
  988. },
  989. //
  990. legalExp: function (node) {
  991. let exp = this.standar(node.data.userCalcBase);
  992. if(!this.inputLegal(exp)){
  993. throw '表达式含有无效字符';
  994. }
  995. if(!this.arithmeticLegal(exp)){
  996. throw '表达式含有无效字符';
  997. }
  998. if(!this.baseLegal(cbTools.getValidFigures(node), exp)){
  999. throw '清单基数不合法';
  1000. }
  1001. if(!this.fLegal(calcBase.project.mainTree.items, exp)){
  1002. throw '行引用不合法';
  1003. }
  1004. //转换成ID引用
  1005. exp = cbParser.toIDExpr(exp);
  1006. if(this.cycleCalc(node, calcBase.baseFigures, exp)){
  1007. throw '出现循环计算';
  1008. }
  1009. return exp;
  1010. }
  1011. };
  1012. //输入式转换器
  1013. let cbParser = {
  1014. //获取标记F
  1015. getFMArr: function (exp) {
  1016. let fmRex = /F/g;
  1017. let fmArr = exp.match(fmRex);
  1018. return cbTools.isDef(fmArr) ? fmArr : [];
  1019. },
  1020. //获取行引用 eg: F10
  1021. getFArr: function (exp) {
  1022. let fRex = /F\d+\b/g;
  1023. let fArr = exp.match(fRex);
  1024. return cbTools.isDef(fArr) ? fArr : [];
  1025. },
  1026. //获取X+num eg: F10 10 @105 105
  1027. getXNum: function (arr) {
  1028. let rRex = /\d+/g;
  1029. let tempArr = [];
  1030. for(let i = 0, len = arr.length; i < len; i++){
  1031. tempArr = tempArr.concat(arr[i].match(rRex));
  1032. }
  1033. return tempArr;
  1034. //let rArr = Array.from(new Set(tempArr));
  1035. //return rArr;
  1036. },
  1037. //获取uuid
  1038. getUID: function (arr) {
  1039. let rRex = /[\d,a-z,A-Z,-]{36}/g;
  1040. let tempArr = [];
  1041. for(let i = 0, len = arr.length; i < len; i++){
  1042. tempArr = tempArr.concat(arr[i].match(rRex));
  1043. }
  1044. let rArr = Array.from(new Set(tempArr));
  1045. return rArr;
  1046. },
  1047. //获取ID引用
  1048. getFIDArr: function (exp) {
  1049. return scMathUtil.getFIDArr(exp);//统一前后端调用方法
  1050. },
  1051. //获取表达式中的中文式
  1052. getCN: function(expr){
  1053. let cnRex = /\d*[\u4e00-\u9fa5]{1,}\({0,}[\u4e00-\u9fa5]{0,}\d*%*[\u4e00-\u9fa5]{0,}\){0,}[\u4e00-\u9fa5]{0,}/g;
  1054. return _.filter(expr.match(cnRex), function (data) {
  1055. return data
  1056. });
  1057. },
  1058. //获取表达式中的基数
  1059. getFigure: function(expr){
  1060. let rst = [];
  1061. let rex = /\{([^}]*)\}/g;
  1062. let temp = expr.match(rex);
  1063. if(cbTools.isDef(temp)){
  1064. for(let i = 0, len = temp.length; i < len; i++){
  1065. rst.push(temp[i].replace(/[{,}]/g, ''));
  1066. }
  1067. }
  1068. return rst;
  1069. },
  1070. //获取表达式中的基数和ID引用
  1071. getFigureF: function (figures, fidArr) {
  1072. let rst = [];
  1073. for(let i = 0, len = figures.length; i < len; i++){
  1074. let obj = Object.create(null);
  1075. obj.type = 'base';
  1076. obj.value = figures[i];
  1077. rst.push(obj);
  1078. }
  1079. for(let i = 0, len = fidArr.length; i < len; i++){
  1080. let obj = Object.create(null);
  1081. obj.type = 'id';
  1082. obj.value = fidArr[i];
  1083. rst.push(obj);
  1084. }
  1085. return rst;
  1086. },
  1087. //表达式中的百分数转换成小数
  1088. percentToNum: function (exp) {
  1089. let rex = /[\+,\-,\*,\/]{1}\d+(\.\d+)?%[\u4e00-\u9fa5]{0}/g;
  1090. let percents = exp.match(rex);
  1091. let numRex = /\d+(\.\d+)?/g;
  1092. if(cbTools.isDef(percents)){
  1093. for(let i = 0, len = percents.length; i < len; i++){
  1094. let percentNum = percents[i].match(numRex),
  1095. oprtor = percents[i].replace(`${percentNum}%`, '');
  1096. if(cbTools.isDef(percentNum) && percentNum.length === 1){
  1097. exp = exp.replace(new RegExp(`\\${percents[i]}`, 'g'), `${oprtor}${percentNum[0]/100}`);
  1098. }
  1099. }
  1100. }
  1101. return exp;
  1102. },
  1103. //将行引用转换成ID引用
  1104. toIDExpr: function (exp) {
  1105. let exps = [];
  1106. //获得行引用
  1107. let fArr = this.getFArr(exp);
  1108. for(let i = 0, len = fArr.length; i < len; i++){
  1109. let r = this.getXNum([fArr[i]]);
  1110. if(r.length === 1){
  1111. let node = cbTools.getBillByRow(calcBase.project.mainTree.items, r[0] - 1);
  1112. if(cbTools.isUnDef(node)){
  1113. //continue;
  1114. calcBase.errMsg = '行引用错误';
  1115. throw '行引用错误';
  1116. }
  1117. exps.push({orgExp: fArr[i], newExp: '@' + node.data.ID});
  1118. }
  1119. else {
  1120. calcBase.errMsg = '行引用错误';
  1121. throw '行引用错误';
  1122. }
  1123. }
  1124. for(let i = 0, len = exps.length; i < len; i++){
  1125. exp = exp.replace(new RegExp(`${exps[i].orgExp}\\b`, 'g'), exps[i].newExp);
  1126. }
  1127. return exp;
  1128. },
  1129. //将ID引用转换成行引用
  1130. toFExpr: function (exp) {
  1131. let exps = [];
  1132. //获得ID引用
  1133. let fidArr = this.getFIDArr(exp);
  1134. for(let i = 0, len = fidArr.length; i < len; i++){
  1135. let id = this.getUID([fidArr[i]]);
  1136. if(id.length === 1){
  1137. let row = cbTools.getRowByID(calcBase.project.mainTree.items, id[0]);
  1138. if(cbTools.isUnDef(row)){
  1139. continue;
  1140. }
  1141. exps.push({orgExp: fidArr[i], newExp: 'F' + row});
  1142. }
  1143. }
  1144. for(let i = 0, len = exps.length; i < len; i++){
  1145. exp = exp.replace(new RegExp(`${exps[i].orgExp}\\b`, 'g'), exps[i].newExp);
  1146. }
  1147. return exp;
  1148. },
  1149. //将表达式转换为可编译的表达式
  1150. toCompileExpr: function(v){
  1151. if(v === ''){
  1152. return '$CBC.base(\'NONE\')';
  1153. }
  1154. //基数
  1155. let strs = _.uniq(this.getFigure(v));
  1156. let exps = [];
  1157. for(let i = 0, len = strs.length; i < len; i++){
  1158. let exp = Object.create(null);
  1159. exp.orgExp = `{${strs[i]}}`;
  1160. exps.push(exp);
  1161. }
  1162. for(let i = 0, len = exps.length;i < len; i++){
  1163. exps[i].compileExp = '$CBC.base(\'' + exps[i].orgExp + '\')';
  1164. let regStr = exps[i].orgExp.replace(/\(/g, '\\\(');
  1165. regStr = regStr.replace(/\)/g, '\\\)');
  1166. v = v.replace(new RegExp(regStr, 'g'), exps[i].compileExp);
  1167. }
  1168. //去{}
  1169. v = v.replace(/[{, },]/g, '');
  1170. //行引用
  1171. let fidArr = _.uniq(this.getFIDArr(v));
  1172. let fExps = [];
  1173. for(let i = 0, len = fidArr.length; i < len; i++){
  1174. let fExp = Object.create(null);
  1175. fExp.orgExp = fidArr[i];
  1176. fExps.push(fExp);
  1177. }
  1178. for(let i = 0, len = fExps.length; i < len; i++){
  1179. fExps[i].compileExp = '$CBC.ref(\'' + fExps[i].orgExp + '\')';
  1180. v = v.replace(new RegExp(fExps[i].orgExp, 'g'), fExps[i].compileExp);
  1181. }
  1182. return v;
  1183. }
  1184. };
  1185. let cbCalctor = {
  1186. //计算基数
  1187. base: function (figure) {
  1188. if(figure === 'NONE'){
  1189. return 0;
  1190. }
  1191. if (calcBase.project.property.valuationType === 'bill') {//预算
  1192. return baseFigureTemplate.budget[calcBase.baseFigures[figure]['base']]();
  1193. } else {//工程量清单
  1194. return baseFigureTemplate.boq[calcBase.baseFigures[figure]['base']]();
  1195. }
  1196. },
  1197. //调价后计算基数
  1198. tenderBase: function (figure) {
  1199. if(figure === 'NONE'){
  1200. return 0;
  1201. }
  1202. if (calcBase.project.property.valuationType === 'bill') {//预算
  1203. return baseFigureTemplate.budget[calcBase.baseFigures[figure]['base']](true);
  1204. } else {//工程量清单
  1205. return baseFigureTemplate.boq[calcBase.baseFigures[figure]['base']](true);
  1206. }
  1207. },
  1208. //ID引用
  1209. ref: function (fExp) {
  1210. let ID = cbParser.getUID([fExp]);
  1211. if(ID.length === 1){
  1212. let node = cbTools.getNodeByID(ID[0]);
  1213. return cbTools.isDef(node) &&
  1214. cbTools.isDef(node.data.feesIndex) &&
  1215. cbTools.isDef(node.data.feesIndex.common) &&
  1216. cbTools.isDef(node.data.feesIndex.common.totalFee) ?
  1217. node.data.feesIndex.common.totalFee : 0;
  1218. }
  1219. return 0;
  1220. }
  1221. };
  1222. let calcBase = {
  1223. //正在执行计算的节点(计算基数的排除本项可能会用到)
  1224. errMsg: '表达式不正确',
  1225. success: false,
  1226. //清单固定行
  1227. fixedFlag: null,
  1228. fixedBills: Object.create(null),
  1229. //清单基数
  1230. baseFigures: Object.create(null),
  1231. //清单固定行可用基数对应 {flag: Number, baseList: Array}
  1232. flagValidBase: Object.create(null),
  1233. //清单可选基数映射,分两类:组织措施项目:排除父项和计算的父项; 其他项目、规费、税金、工程造价,及新增部分:显示所有计算基数
  1234. baseFigureClass: Object.create(null),
  1235. //初始化
  1236. init: function (project) {
  1237. let me = this;
  1238. me.project = project;
  1239. me.fixedFlag = fixedFlag;
  1240. cbTools.setFixedBills(project, me.fixedBills, me.fixedFlag);
  1241. if (project.property.valuationType === 'bill') {//预算
  1242. me.baseFigures = baseFigureMap.budget;
  1243. } else {//工程量清单
  1244. me.baseFigures = baseFigureMap.boq;
  1245. }
  1246. cbTools.setBaseBills(me.baseFigures, me.fixedBills);
  1247. //设置清单固定行可用基数映射
  1248. cbTools.setValidBaseMapping(me.baseFigures, me.flagValidBase);
  1249. },
  1250. getBase: function (figure) {
  1251. return cbCalctor.base(figure);
  1252. },
  1253. getBaseByClass: function (node) {
  1254. return cbTools.getValidFigures(node);
  1255. },
  1256. calculate: function (node, reCalc = null) {
  1257. let me = calcBase,
  1258. $CBA = cbAnalyzer,
  1259. $CBP = cbParser,
  1260. $CBC = cbCalctor;
  1261. try {
  1262. me.success = false;
  1263. me.errMsg = '表达式不正确';
  1264. //分析输入式合法性
  1265. let exp = reCalc
  1266. ? cbTools.isDef(node.data.calcBase)
  1267. ? node.data.calcBase
  1268. : ''
  1269. : $CBA.legalExp(node);
  1270. if(!cbTools.isDef(exp)){
  1271. throw '表达式不正确';
  1272. }
  1273. //输入式转换表达式
  1274. let compileExp = $CBP.toCompileExpr(exp);
  1275. //计算
  1276. let calcExp = $CBP.percentToNum(compileExp);
  1277. let calcBaseValue = eval(calcExp);
  1278. if(!cbTools.isNum(calcBaseValue)){
  1279. throw '基数计算结果不为数值';
  1280. }
  1281. //调价
  1282. let tenderCalcExp = calcExp.replace(new RegExp('base', 'g'), 'tenderBase');
  1283. let tenderCalcBaseValue = eval(tenderCalcExp);
  1284. if(!cbTools.isNum(tenderCalcBaseValue)){
  1285. throw '调价基数计算结果不为数值';
  1286. }
  1287. //存储
  1288. me.success = true;
  1289. node.updateData.calcBase = exp;
  1290. node.updateData.calcBaseValue = parseFloat(calcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1291. node.updateData.tenderCalcBaseValue = parseFloat(tenderCalcBaseValue).toDecimal(decimalObj.decimal('totalPrice', node));
  1292. node.changed = true;
  1293. }
  1294. catch (err){
  1295. if(typeof err === 'object'){
  1296. err = '表达式不正确'
  1297. }
  1298. if (node) {
  1299. err = `第${node.serialNo() + 1}行${err}`;
  1300. }
  1301. alert(err);
  1302. }
  1303. }
  1304. };