budget_final.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date
  7. * @version
  8. */
  9. const itemsPre = 'id_';
  10. const BillsTree = require('./ledger').billsTree;
  11. const TreeUtils = require('./ledger').treeUtils;
  12. const auditConst = require('../const/audit');
  13. class FinalTree extends BillsTree {
  14. constructor(ctx, setting) {
  15. super(ctx, setting);
  16. this._newId = 1;
  17. this.checkField = ['code', 'name'];
  18. }
  19. get newId() {
  20. return this._newId++;
  21. }
  22. loadNode(node, parent, loadFun) {
  23. if (node.b_code) return;
  24. const checkField = this.checkField;
  25. const siblings = parent ? parent.children : this.children;
  26. let cur = siblings.find(function (x) {
  27. for (const cf of checkField) {
  28. if (x[cf] !== node[cf]) return false;
  29. }
  30. return true;
  31. // return x.code === node.code && x.name === node.name;
  32. });
  33. if (!cur) {
  34. cur = { children: [], code: node.code || '', name: node.name || '', unit: node.unit || '' };
  35. const id = this.newId;
  36. cur[this.setting.id] = id;
  37. cur[this.setting.pid] = parent ? parent[this.setting.id] : this.setting.rootId;
  38. cur[this.setting.fullPath] = parent ? parent[this.setting.fullPath] + '-' + id : '' + id;
  39. cur[this.setting.level] = parent ? parent[this.setting.level] + 1 : 1;
  40. cur[this.setting.order] = siblings.length + 1;
  41. siblings.push(cur);
  42. this.datas.push(cur);
  43. }
  44. loadFun(cur, node);
  45. for (const c of node.children) {
  46. this.loadNode(c, cur, loadFun);
  47. }
  48. }
  49. loadTree(tree, loadFun) {
  50. for (const node of tree.children) {
  51. this.loadNode(node, null, loadFun);
  52. }
  53. }
  54. generateSortNodes() {
  55. const self = this;
  56. const addSortNode = function (node) {
  57. self.nodes.push(node);
  58. for (const c of node.children) {
  59. addSortNode(c);
  60. }
  61. };
  62. this.nodes = [];
  63. for (const n of this.children) {
  64. addSortNode(n);
  65. }
  66. }
  67. afterLoad(fun) {
  68. for (const d of this.datas) {
  69. fun && fun(d);
  70. d.is_leaf = d.children.length === 0;
  71. d.expanded = true;
  72. d.visible = true;
  73. this.items[itemsPre + d[this.setting.id]] = d;
  74. }
  75. this.generateSortNodes();
  76. }
  77. resortChildrenByCustom(fun) {
  78. for (const n of this.nodes) {
  79. if (n.children && n.children.length > 1) {
  80. n.children.sort(fun);
  81. n.children.forEach((x, y) => { x.order = y + 1; });
  82. }
  83. }
  84. this.generateSortNodes();
  85. }
  86. }
  87. class BudgetFinal {
  88. constructor (ctx) {
  89. this.ctx = ctx;
  90. this.budgetSetting = { id: 'tree_id', pid: 'tree_pid', order: 'order', level: 'level', rootId: -1, calcFields: ['total_price'] };
  91. this.tenderSetting = { id: 'ledger_id', pid: 'ledger_pid', order: 'order', level: 'level', rootId: -1, calcFields: ['total_price', 'end_gather_tp', 'end_contract_tp', 'end_qc_tp', 'tz_qc_tp'] };
  92. this.finalTree = new FinalTree(this.ctx, { id: 'id', pid: 'pid', order: 'order', level: 'level', fullPath: 'full_path', rootId: -1 });
  93. }
  94. async _loadGu(budget) {
  95. const helper = this.ctx.helper;
  96. const gu = await this.ctx.service.budgetGu.getData(budget.id);
  97. const guTree = new BillsTree(this.ctx, { id: 'tree_id', pid: 'tree_pid', order: 'order', level: 'level', rootId: -1, calcFields: ['total_price'] });
  98. guTree.loadDatas(gu);
  99. guTree.calculateAll();
  100. this.finalTree.loadTree(guTree, function (cur, source) {
  101. cur.base = true;
  102. cur.gu_dgn_qty1 = helper.add(cur.gu_dgn_qty1, source.dgn_qty1);
  103. cur.gu_dgn_qty2 = helper.add(cur.gu_dgn_qty2, source.dgn_qty2);
  104. cur.gu_tp = helper.add(cur.gu_tp, source.total_price);
  105. });
  106. }
  107. async _loadGai(budget) {
  108. const helper = this.ctx.helper;
  109. const gai = await this.ctx.service.budgetGai.getData(budget.id);
  110. const gaiTree = new BillsTree(this.ctx, { id: 'tree_id', pid: 'tree_pid', order: 'order', level: 'level', rootId: -1, calcFields: ['total_price'] });
  111. gaiTree.loadDatas(gai);
  112. gaiTree.calculateAll();
  113. this.finalTree.loadTree(gaiTree, function (cur, source) {
  114. cur.base = true;
  115. cur.gai_dgn_qty1 = helper.add(cur.gai_dgn_qty1, source.dgn_qty1);
  116. cur.gai_dgn_qty2 = helper.add(cur.gai_dgn_qty2, source.dgn_qty2);
  117. cur.gai_tp = helper.add(cur.gai_tp, source.total_price);
  118. });
  119. }
  120. async _loadYu(budget) {
  121. const helper = this.ctx.helper;
  122. const yu = await this.ctx.service.budgetYu.getData(budget.id);
  123. const yuTree = new BillsTree(this.ctx, { id: 'tree_id', pid: 'tree_pid', order: 'order', level: 'level', rootId: -1, calcFields: ['total_price'] });
  124. yuTree.loadDatas(yu);
  125. yuTree.calculateAll();
  126. this.finalTree.loadTree(yuTree, function (cur, source) {
  127. cur.base = true;
  128. cur.yu_dgn_qty1 = helper.add(cur.yu_dgn_qty1, source.dgn_qty1);
  129. cur.yu_dgn_qty2 = helper.add(cur.yu_dgn_qty2, source.dgn_qty2);
  130. cur.yu_tp = helper.add(cur.yu_tp, source.total_price);
  131. });
  132. }
  133. async _loadZb(budget) {
  134. const helper = this.ctx.helper;
  135. const zb = await this.ctx.service.budgetZb.getData(budget.id);
  136. const zbTree = new BillsTree(this.ctx, { id: 'tree_id', pid: 'tree_pid', order: 'order', level: 'level', rootId: -1, calcFields: ['total_price'] });
  137. zbTree.loadDatas(zb);
  138. zbTree.calculateAll();
  139. this.finalTree.loadTree(zbTree, function (cur, source) {
  140. cur.base = true;
  141. cur.zb_dgn_qty1 = helper.add(cur.zb_dgn_qty1, source.dgn_qty1);
  142. cur.zb_dgn_qty2 = helper.add(cur.zb_dgn_qty2, source.dgn_qty2);
  143. cur.zb_tp = helper.add(cur.zb_tp, source.total_price);
  144. });
  145. }
  146. async _loadCtrl(budget) {
  147. const helper = this.ctx.helper;
  148. const ctrl = await this.ctx.service.budgetCtrl.getData(budget.id);
  149. const ctrlTree = new BillsTree(this.ctx, { id: 'tree_id', pid: 'tree_pid', order: 'order', level: 'level', rootId: -1, calcFields: ['total_price'] });
  150. ctrlTree.loadDatas(ctrl);
  151. ctrlTree.calculateAll();
  152. this.finalTree.loadTree(ctrlTree, function (cur, source) {
  153. cur.base = true;
  154. cur.ctrl_dgn_qty1 = helper.add(cur.ctrl_dgn_qty1, source.dgn_qty1);
  155. cur.ctrl_dgn_qty2 = helper.add(cur.ctrl_dgn_qty2, source.dgn_qty2);
  156. cur.ctrl_tp = helper.add(cur.ctrl_tp, source.total_price);
  157. });
  158. }
  159. async _loadTender(id) {
  160. const helper = this.ctx.helper;
  161. const bills = await this.ctx.service.ledger.getFinalData(id, ['id', 'ledger_id', 'ledger_pid', 'level', 'order', 'full_path', 'is_leaf',
  162. 'code', 'b_code', 'name', 'unit', 'unit_price', 'dgn_qty1', 'dgn_qty2', 'total_price']);
  163. const dgnData = await this.ctx.service.stageBillsDgn.getDgnData(id);
  164. // 使用最新一期对比
  165. const stage = await this.ctx.service.stage.getLastestStage(id);
  166. if (!stage) {
  167. helper.assignRelaData(bills, [
  168. { data: dgnData, fields: ['deal_dgn_qty1', 'deal_dgn_qty2', 'c_dgn_qty1', 'c_dgn_qty2'], prefix: '', relaId: 'id' },
  169. ]);
  170. this.final.tender_info.push({ id, stageCount: 0 });
  171. } else if (stage.status === auditConst.stage.status.checked) {
  172. const finalBills = await this.ctx.service.stageBillsFinal.getFinalData({id}, stage.order);
  173. helper.assignRelaData(bills, [
  174. { data: finalBills, fields: ['contract_tp', 'qc_tp'], prefix: 'end_', relaId: 'lid' },
  175. { data: dgnData, fields: ['deal_dgn_qty1', 'deal_dgn_qty2', 'c_dgn_qty1', 'c_dgn_qty2'], prefix: '', relaId: 'id' },
  176. ]);
  177. bills.forEach(b => {
  178. b.end_gather_tp = helper.add(b.end_qc_tp, b.end_contract_tp);
  179. });
  180. this.final.tender_info.push({ id, stageOrder: stage.order });
  181. } else {
  182. await this.ctx.service.stage.doCheckStage(stage);
  183. const curBills = stage.readOnly
  184. ? await this.ctx.service.stageBills.getAuditorStageData2(id, stage.id, stage.curTimes, stage.curOrder)
  185. : await this.ctx.service.stageBills.getLastestStageData2(id, stage.id);
  186. const preBills = stage.preCheckedStage ? await this.ctx.service.stageBillsFinal.getFinalData({id}, stage.preCheckedStage.order) : [];
  187. const bpcData = await this.ctx.service.stageBillsPc.getAllDataByCondition({ where: { sid: stage.id } });
  188. helper.assignRelaData(bills, [
  189. { data: curBills, fields: ['contract_tp', 'qc_tp'], prefix: '', relaId: 'lid' },
  190. { data: preBills, fields: ['contract_tp', 'qc_tp'], prefix: 'pre_', relaId: 'lid' },
  191. { data: dgnData, fields: ['deal_dgn_qty1', 'deal_dgn_qty2', 'c_dgn_qty1', 'c_dgn_qty2'], prefix: '', relaId: 'id' },
  192. { data: bpcData, fields: ['contract_pc_tp', 'qc_pc_tp', 'pc_tp', 'org_price'], prefix: '', relaId: 'lid' },
  193. ]);
  194. bills.forEach(b => {
  195. b.end_contract_tp = helper.sum([b.contract_tp, b.pre_contract_tp, b.contract_pc_tp]);
  196. b.end_qc_tp = helper.sum([b.qc_tp, b.pre_qc_tp, b.qc_pc_tp]);
  197. b.end_gather_tp = helper.sum([b.qc_tp, b.contract_tp, b.pre_qc_tp, b.pre_contract_tp, b.pc_tp]);
  198. });
  199. this.final.tender_info.push({ id, stageOrder: stage.order, stageStatus: stage.status, stageFlow: stage.curTimes + '-' + stage.curOrder });
  200. }
  201. const tree = new BillsTree(this.ctx, this.tenderSetting);
  202. tree.loadDatas(bills);
  203. const tenderInfo = await this.ctx.service.tenderInfo.getTenderInfo(id);
  204. const reCalcChange = await this.ctx.service.change.getReCalcChangeData(id, tenderInfo);
  205. TreeUtils.loadChangeData(tree, null, reCalcChange, tenderInfo, helper);
  206. tree.calculateAll();
  207. this.finalTree.loadTree(tree, function (cur, source) {
  208. cur.total_price = helper.add(cur.total_price, source.total_price);
  209. cur.dgn_qty1 = helper.add(cur.dgn_qty1, source.dgn_qty1);
  210. cur.dgn_qty2 = helper.add(cur.dgn_qty2, source.dgn_qty2);
  211. cur.deal_dgn_qty1 = helper.add(cur.deal_dgn_qty1, source.deal_dgn_qty1);
  212. cur.deal_dgn_qty2 = helper.add(cur.deal_dgn_qty2, source.deal_dgn_qty2);
  213. cur.c_dgn_qty1 = helper.add(cur.c_dgn_qty1, source.c_dgn_qty1);
  214. cur.c_dgn_qty2 = helper.add(cur.c_dgn_qty2, source.c_dgn_qty2);
  215. cur.final_dgn_qty1 = helper.sum([cur.final_dgn_qty1, source.deal_dgn_qty1, source.c_dgn_qty1]);
  216. cur.final_dgn_qty2 = helper.sum([cur.final_dgn_qty2, source.deal_dgn_qty2, source.c_dgn_qty2]);
  217. cur.final_contract_tp = helper.add(cur.final_contract_tp, source.end_contract_tp);
  218. cur.final_qc_tp = helper.add(cur.final_qc_tp, source.end_qc_tp);
  219. cur.final_tp = helper.add(cur.final_tp, source.end_gather_tp);
  220. cur.tz_qc_qty = helper.add(cur.tz_qc_qty, source.tz_qc_qty);
  221. cur.tz_qc_tp = helper.add(cur.tz_qc_tp, source.tz_qc_tp);
  222. });
  223. }
  224. async _afterLoad() {
  225. const helper = this.ctx.helper;
  226. const checkGaiExist = this.finalTree.datas.findIndex(x => {
  227. if (x.gai_tp || x.gai_dgn_qty1 || x.gai_dgn_qty2) return true;
  228. }) >= 0;
  229. this.finalTree.afterLoad(node => {
  230. node.dgn_price = helper.div(node.total_price, node.dgn_qty1, 2);
  231. node.dgn_qty = node.dgn_qty1
  232. ? (node.dgn_qty2 ? node.dgn_qty1 + '/' + node.dgn_qty2 : node.dgn_qty1)
  233. : (node.dgn_qty2 ? '/' + node.dgn_qty2 : '');
  234. node.gu_dgn_price = helper.div(node.gu_tp, node.gu_dgn_qty1, 2);
  235. node.gu_dgn_qty = node.gu_dgn_qty1
  236. ? (node.gu_dgn_qty2 ? node.gu_dgn_qty1 + '/' + node.gu_dgn_qty2 : node.gu_dgn_qty1 + '')
  237. : (node.gu_dgn_qty2 ? '/' + node.gu_dgn_qty2 : '');
  238. node.gai_dgn_price = helper.div(node.gai_tp, node.gai_dgn_qty1, 2);
  239. node.gai_dgn_qty = node.gai_dgn_qty1
  240. ? (node.gai_dgn_qty2 ? node.gai_dgn_qty1 + '/' + node.gai_dgn_qty2 : node.gai_dgn_qty1 + '')
  241. : (node.gai_dgn_qty2 ? '/' + node.gai_dgn_qty2 : '');
  242. node.yu_dgn_price = helper.div(node.yu_tp, node.yu_dgn_qty1, 2);
  243. node.yu_dgn_qty = node.yu_dgn_qty1
  244. ? (node.yu_dgn_qty2 ? node.yu_dgn_qty1 + '/' + node.yu_dgn_qty2 : node.yu_dgn_qty1 + '')
  245. : (node.yu_dgn_qty2 ? '/' + node.yu_dgn_qty2 : '');
  246. node.zb_dgn_price = helper.div(node.zb_tp, node.zb_dgn_qty1, 2);
  247. node.zb_dgn_qty = node.zb_dgn_qty1
  248. ? (node.zb_dgn_qty2 ? node.zb_dgn_qty1 + '/' + node.zb_dgn_qty2 : node.zb_dgn_qty1 + '')
  249. : (node.zb_dgn_qty2 ? '/' + node.zb_dgn_qty2 : '');
  250. node.ctrl_dgn_price = helper.div(node.ctrl_tp, node.ctrl_dgn_qty1, 2);
  251. node.ctrl_dgn_qty = node.ctrl_dgn_qty1
  252. ? (node.ctrl_dgn_qty2 ? node.ctrl_dgn_qty1 + '/' + node.ctrl_dgn_qty2 : node.ctrl_dgn_qty1 + '')
  253. : (node.ctrl_dgn_qty2 ? '/' + node.ctrl_dgn_qty2 : '');
  254. node.final_dgn_price = helper.div(node.final_tp, node.final_dgn_qty1, 2);
  255. node.final_dgn_qty = node.final_dgn_qty1
  256. ? (node.final_dgn_qty2 ? node.final_dgn_qty1 + '/' + node.final_dgn_qty2 : node.final_dgn_qty1)
  257. : (node.final_dgn_qty2 ? '/' + node.final_dgn_qty2 : '');
  258. if (checkGaiExist) {
  259. node.grow_dgn_qty1 = helper.mul(helper.div(helper.sub(node.final_dgn_qty1, node.gai_dgn_qty1), node.gai_dgn_qty1, 4), 100);
  260. node.grow_dgn_qty2 = helper.mul(helper.div(helper.sub(node.final_dgn_qty2, node.gai_dgn_qty2), node.gai_dgn_qty2, 4), 100);
  261. node.grow_dgn_qty = node.grow_dgn_qty1
  262. ? (node.grow_dgn_qty2 ? node.grow_dgn_qty1 + '/' + node.grow_dgn_qty2 : node.grow_dgn_qty1)
  263. : (node.grow_dgn_qty2 ? '/' + node.grow_dgn_qty2 : '');
  264. node.grow_tp = helper.mul(helper.div(helper.sub(node.final_tp, node.gai_tp), node.gai_tp, 4), 100);
  265. } else {
  266. node.grow_dgn_qty1 = helper.mul(helper.div(helper.sub(node.final_dgn_qty1, node.yu_dgn_qty1), node.yu_dgn_qty1, 4), 100);
  267. node.grow_dgn_qty2 = helper.mul(helper.div(helper.sub(node.final_dgn_qty2, node.yu_dgn_qty2), node.yu_dgn_qty2, 4), 100);
  268. node.grow_dgn_qty = node.grow_dgn_qty1
  269. ? (node.grow_dgn_qty2 ? node.grow_dgn_qty1 + '/' + node.grow_dgn_qty2 : node.grow_dgn_qty1)
  270. : (node.grow_dgn_qty2 ? '/' + node.grow_dgn_qty2 : '');
  271. node.grow_tp = helper.mul(helper.div(helper.sub(node.final_tp, node.yu_tp), node.yu_tp, 4), 100);
  272. }
  273. });
  274. this.finalTree.resortChildrenByCustom(function (x, y) {
  275. const iCode = helper.compareCode(x.code, y.code);
  276. if (iCode) return iCode;
  277. if (!x.name) return -1;
  278. if (!y.name) return 1;
  279. return x.name.localeCompare(y.name);
  280. });
  281. }
  282. getFinalData() {
  283. const data = [], ctx = this.ctx, bid = this.budget.id, final_id = this.final.id;
  284. this.finalTree.datas.forEach(x => {
  285. data.push({
  286. id: ctx.app.uuid.v4(), bid, final_id,
  287. tree_id: x.id, tree_pid: x.pid, order: x.order, level: x.level, full_path: x.full_path, is_leaf: x.children && x.children.length > 0 ? 0 : 1,
  288. code: x.code || '', name: x.name || '', unit: x.unit || '',
  289. gu_dgn_qty1: x.gu_dgn_qty1 || 0, gu_dgn_qty2: x.gu_dgn_qty2 || 0, gu_dgn_qty: x.gu_dgn_qty || '', gu_dgn_price: x.gu_dgn_price || 0, gu_tp: x.gu_tp || 0,
  290. gai_dgn_qty1: x.gai_dgn_qty1 || 0, gai_dgn_qty2: x.gai_dgn_qty2 || 0, gai_dgn_qty: x.gai_dgn_qty || '', gai_dgn_price: x.gai_dgn_price || 0, gai_tp: x.gai_tp || 0,
  291. yu_dgn_qty1: x.yu_dgn_qty1 || 0, yu_dgn_qty2: x.yu_dgn_qty2 || 0, yu_dgn_qty: x.yu_dgn_qty || '', yu_dgn_price: x.yu_dgn_price || 0, yu_tp: x.yu_tp || 0,
  292. zb_dgn_qty1: x.zb_dgn_qty1 || 0, zb_dgn_qty2: x.zb_dgn_qty2 || 0, zb_dgn_qty: x.zb_dgn_qty || '', zb_dgn_price: x.zb_dgn_price || 0, zb_tp: x.zb_tp || 0,
  293. ctrl_dgn_qty1: x.ctrl_dgn_qty1 || 0, ctrl_dgn_qty2: x.ctrl_dgn_qty2 || 0, ctrl_dgn_qty: x.ctrl_dgn_qty || '', ctrl_dgn_price: x.ctrl_dgn_price || 0, ctrl_tp: x.ctrl_tp || 0,
  294. dgn_qty1: x.dgn_qty1 || 0, dgn_qty2: x.dgn_qty2 || 0, total_price: x.total_price || 0,
  295. dgn_price: x.dgn_price || 0, dgn_qty: x.dgn_qty || '',
  296. deal_dgn_qty1: x.deal_dgn_qty1 || 0, deal_dgn_qty2: x.deal_dgn_qty2 || 0, c_dgn_qty1: x.c_dgn_qty1 || 0, c_dgn_qty2: x.c_dgn_qty2 || 0,
  297. final_dgn_qty1: x.final_dgn_qty1 || 0, final_dgn_qty2: x.final_dgn_qty2 || 0, final_tp: x.final_tp || 0,
  298. final_contract_tp: x.final_contract_tp || 0, final_qc_tp: x.final_qc_tp || 0,
  299. final_dgn_price: x.final_dgn_price || 0, final_dgn_qty: x.final_dgn_qty || '',
  300. tz_qc_qty: x.tz_qc_qty || 0, tz_qc_tp: x.tz_qc_tp || 0,
  301. grow_dgn_qty1: x.grow_dgn_qty1 || 0, grow_dgn_qty2: x.grow_dgn_qty2 || 0, grow_dgn_qty: x.grow_dgn_qty || '', grow_tp: x.grow_tp || 0,
  302. })
  303. });
  304. return data;
  305. }
  306. async doFinal(budget, final) {
  307. this.budget = budget;
  308. this.final = final;
  309. this.finalTree.checkField = budget.final_type.split('_');
  310. for (const t of final.tender) {
  311. await this._loadTender(t);
  312. }
  313. await this._loadCtrl(budget);
  314. await this._loadZb(budget);
  315. await this._loadYu(budget);
  316. await this._loadGai(budget);
  317. await this._loadGu(budget);
  318. this._afterLoad();
  319. return this.getFinalData();
  320. }
  321. }
  322. module.exports = BudgetFinal;