calc_base.js 57 KB

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