bills_pos_convert.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date
  7. * @version
  8. */
  9. const Ledger = require('./ledger');
  10. const splitChar = '-';
  11. const mergeChar = ';';
  12. class BillsPosConvert {
  13. constructor(ctx) {
  14. const self = this;
  15. this.ctx = ctx;
  16. this._ = this.ctx.helper._;
  17. this.tpFields = ['total_price', 'contract_tp', 'qc_tp', 'gather_tp',
  18. 'pre_contract_tp', 'pre_qc_tp', 'pre_gather_tp', 'end_contract_tp', 'end_qc_tp'];
  19. this.baseCalcFields = ['quantity', 'contract_qty', 'qc_qty', 'gather_qty',
  20. 'pre_contract_qty', 'pre_qc_qty', 'pre_gather_qty', 'end_contract_qty', 'end_qc_qty', 'end_gather_qty'];
  21. // mainData
  22. this.bpcTree = new Ledger.billsTree(this.ctx, {
  23. id: 'ledger_id',
  24. pid: 'ledger_pid',
  25. order: 'order',
  26. level: 'level',
  27. rootId: -1,
  28. keys: ['id', 'tender_id', 'ledger_id'],
  29. stageId: 'id',
  30. updateFields: ['contract_qty', 'contract_tp', 'qc_qty', 'qc_tp'],
  31. calcFields: [],
  32. });
  33. this.bpcPos = new Ledger.pos({
  34. id: 'id', ledgerId: 'lid',
  35. updateFields: ['contract_qty', 'qc_qty'],
  36. });
  37. this.bpcChange = [];
  38. // reusltData
  39. this.resultTreeSetting = {
  40. id: 'ledger_id',
  41. pid: 'ledger_pid',
  42. order: 'order',
  43. level: 'level',
  44. rootId: -1,
  45. fullPath: 'full_path',
  46. };
  47. this.resultTree = new Ledger.filterTree(this.ctx, this.resultTreeSetting);
  48. }
  49. // 加载基础数据
  50. _loadLedgerData (ledger) {
  51. const self = this;
  52. // 加载树结构数据
  53. this.bpcTree.loadDatas(ledger);
  54. }
  55. _loadPosData(pos) {
  56. this.bpcPos.loadDatas(pos);
  57. }
  58. loadData (ledger, pos, change) {
  59. this._loadLedgerData(ledger);
  60. this._loadPosData(pos);
  61. this.bpcChange = change;
  62. }
  63. _convertXmj(data) {
  64. const xmj = this.resultTree.addData(data, ['ledger_id', 'ledger_pid', 'order', 'level', 'tender_id', 'full_path',
  65. 'code', 'name', 'unit', 'dgn_qty1', 'dgn_qty2', 'drawing_code', 'postil', 'memo', 'gxby_status', 'dagl_status', 'dagl_url']);
  66. return xmj;
  67. }
  68. // v1
  69. _loadField(node, data, fields) {
  70. for (const prop in data) {
  71. if (fields.indexOf(prop) >= 0) node[prop] = data[prop];
  72. }
  73. }
  74. // v2
  75. _loadPosCalcFields(node, data) {
  76. const tpDecimal = this.ctx.tender.info.decimal.tp;
  77. node.quantity = this.ctx.helper.add(node.quantity, data.quantity);
  78. node.total_price = this.ctx.helper.add(node.total_price, this.ctx.helper.mul(node.unit_price, data.quantity, tpDecimal));
  79. node.contract_qty = this.ctx.helper.add(node.contract_qty, data.contract_qty);
  80. node.contract_tp = this.ctx.helper.add(node.contract_tp, this.ctx.helper.mul(node.unit_price, data.contract_qty, tpDecimal));
  81. node.qc_qty = this.ctx.helper.add(node.qc_qty, data.qc_qty);
  82. node.qc_tp = this.ctx.helper.add(node.qc_tp, this.ctx.helper.mul(node.unit_price, data.qc_qty, tpDecimal));
  83. node.gather_qty = this.ctx.helper.add(node.gather_qty, data.gather_qty);
  84. node.pre_contract_qty = this.ctx.helper.add(node.pre_contract_qty, data.pre_contract_qty);
  85. node.pre_contract_tp = this.ctx.helper.add(node.pre_contract_tp, this.ctx.helper.mul(node.unit_price, data.pre_contract_qty, tpDecimal));
  86. node.pre_qc_qty = this.ctx.helper.add(node.pre_qc_qty, data.pre_qc_qty);
  87. node.pre_qc_tp = this.ctx.helper.add(node.pre_qc_tp, this.ctx.helper.mul(node.unit_price, data.pre_qc_qty, tpDecimal));
  88. node.pre_gather_qty = this.ctx.helper.add(node.pre_gather_qty, data.pre_gather_qty);
  89. node.real_qty = this.ctx.helper.add(node.real_qty, data.real_qty);
  90. }
  91. _loadBillsCalcFields(node, data) {
  92. node.quantity = this.ctx.helper.add(node.quantity, data.quantity);
  93. node.total_price = this.ctx.helper.add(node.total_price, data.total_price);
  94. node.contract_qty = this.ctx.helper.add(node.contract_qty, data.contract_qty);
  95. node.contract_tp = this.ctx.helper.add(node.contract_tp, data.contract_tp);
  96. node.qc_qty = this.ctx.helper.add(node.qc_qty, data.qc_qty);
  97. node.qc_tp = this.ctx.helper.add(node.qc_tp, data.qc_tp);
  98. node.gather_qty = this.ctx.helper.add(node.gather_qty, data.gather_qty);
  99. node.pre_contract_qty = this.ctx.helper.add(node.pre_contract_qty, data.pre_contract_qty);
  100. node.pre_contract_tp = this.ctx.helper.add(node.pre_contract_tp, data.pre_contract_tp);
  101. node.pre_qc_qty = this.ctx.helper.add(node.pre_qc_qty, data.pre_qc_qty);
  102. node.pre_qc_tp = this.ctx.helper.add(node.pre_qc_tp, data.pre_qc_tp);
  103. node.pre_gather_qty = this.ctx.helper.add(node.pre_gather_qty, data.pre_gather_qty);
  104. }
  105. _convertGcl(node, xmj) {
  106. if (!xmj) return;
  107. if (!xmj.unitTree) {
  108. xmj.unitTree = new Ledger.filterGatherTree(this.ctx, this.resultTreeSetting);
  109. }
  110. const pos = this.bpcPos.getLedgerPos(node.id);
  111. if (pos && pos.length > 0) {
  112. for (const p of pos) {
  113. const posUnit = xmj.unitTree.addNode({pos_name: p.name,
  114. b_code: null, name: '', unit: null, unit_price: null,
  115. gxby_status: p.gxby_status, dagl_status: p.dagl_status, dagl_url: p.dagl_url});
  116. if (p.dagl_status > 0) console.log(posUnit);
  117. if (p.drawing_code && posUnit.drawing_code.indexOf(p.drawing_code) < 0)
  118. posUnit.drawing_code.push(p.drawing_code);
  119. if (p.memo) posUnit.memo.push(p.memo);
  120. if (p.postil) posUnit.postil.push(p.postil);
  121. const gclUnit = xmj.unitTree.addNode({pos_name: '',
  122. b_code: node.b_code, name: node.name, unit: node.unit, unit_price: node.unit_price
  123. }, posUnit);
  124. if (node.drawing_code && gclUnit.drawing_code.indexOf(node.drawing_code) < 0)
  125. gclUnit.drawing_code.push(node.drawing_code);
  126. if (node.memo) gclUnit.memo.push(node.memo);
  127. if (node.postil) gclUnit.postil.push(node.postil);
  128. //loadField(gclUnit, p, baseCalcFields);
  129. this._loadPosCalcFields(gclUnit, p);
  130. if (!gclUnit.changes) gclUnit.changes = [];
  131. for (const c of this.bpcChange) {
  132. if (c.lid === node.id && c.pid === p.id && c.qty && c.qty !== 0) {
  133. gclUnit.changes.push(c);
  134. }
  135. }
  136. }
  137. } else {
  138. const unit = xmj.unitTree.addNode({
  139. pos_name: '',
  140. b_code: node.b_code, name: node.name, unit: node.unit, unit_price: node.unit_price,
  141. });
  142. if (node.drawing_code && unit.drawing_code.indexOf(node.drawing_code) < 0)
  143. unit.drawing_code.push(node.drawing_code);
  144. if (node.memo) unit.memo.push(node.memo);
  145. if (node.postil) unit.postil.push(node.postil);
  146. this._loadBillsCalcFields(unit, node);
  147. if (!unit.changes) unit.changes = [];
  148. for (const c of this.bpcChange) {
  149. if (c.lid === node.id && c.pid == -1 && c.qty && c.qty !== 0) {
  150. unit.changes.push(c);
  151. }
  152. }
  153. }
  154. }
  155. _recursiveConvertNode(nodes, xmj = null) {
  156. if (!nodes || !nodes instanceof Array || nodes.length === 0) return;
  157. for (const node of nodes) {
  158. if (node.b_code && node.b_code !== '') {
  159. if (!node.children || node.children.length === 0) {
  160. this._convertGcl(node, xmj);
  161. }
  162. this._recursiveConvertNode(node.children, xmj);
  163. } else {
  164. const curXmj = this._convertXmj(node);
  165. this._recursiveConvertNode(node.children, curXmj);
  166. if (!node.children || node.children.length === 0) {
  167. this._loadField(curXmj, node, this.tpFields);
  168. }
  169. }
  170. }
  171. }
  172. _calculateChild(child) {
  173. const tpDecimal = this.ctx.tender.info.decimal.tp;
  174. child.gather_qty = this.ctx.helper.add(child.contract_qty, child.qc_qty);
  175. child.pre_gather_qty = this.ctx.helper.add(child.pre_contract_qty, child.pre_qc_qty);
  176. child.end_contract_qty = this.ctx.helper.add(child.contract_qty, child.pre_contract_qty);
  177. child.end_qc_qty = this.ctx.helper.add(child.qc_qty, child.pre_qc_qty);
  178. child.end_gather_qty = this.ctx.helper.add(child.gather_qty, child.pre_gather_qty);
  179. child.estimate_qty = !this.ctx.helper.checkZero(child.real_qty)
  180. ? this.ctx.helper.sub(this.ctx.helper.sub(child.real_qty, child.quantity), child.end_qc_qty)
  181. : null;
  182. // v1
  183. // child.total_price = this.ctx.helper.mul(child.unit_price, child.quantity, tpDecimal);
  184. // child.contract_tp = this.ctx.helper.mul(child.unit_price, child.contract_qty, tpDecimal);
  185. // child.qc_tp = this.ctx.helper.mul(child.unit_price, child.qc_qty, tpDecimal);
  186. // child.gather_tp = this.ctx.helper.mul(child.unit_price, child.gather_qty, tpDecimal);
  187. // child.pre_contract_tp = this.ctx.helper.mul(child.unit_price, child.pre_contract_qty, tpDecimal);
  188. // child.pre_qc_tp = this.ctx.helper.mul(child.unit_price, child.pre_qc_qty, tpDecimal);
  189. // child.pre_gather_tp = this.ctx.helper.mul(child.unit_price, child.pre_gather_qty, tpDecimal);
  190. // child.end_contract_tp = this.ctx.helper.mul(child.unit_price, child.end_contract_qty, tpDecimal);
  191. // child.end_qc_tp = this.ctx.helper.mul(child.unit_price, child.end_qc_qty, tpDecimal);
  192. // child.end_gather_tp = this.ctx.helper.mul(child.unit_price, child.end_gather_qty, tpDecimal);
  193. // v2
  194. child.gather_tp = this.ctx.helper.add(child.contract_tp, child.qc_tp);
  195. child.pre_gather_tp = this.ctx.helper.add(child.pre_contract_tp, child.pre_qc_tp);
  196. child.end_contract_tp = this.ctx.helper.add(child.pre_contract_tp, child.contract_tp);
  197. child.end_qc_tp = this.ctx.helper.add(child.pre_qc_tp, child.qc_tp);
  198. child.end_gather_tp = this.ctx.helper.add(child.end_contract_tp, child.end_qc_tp);
  199. child.final_tp = this.ctx.helper.add(child.total_price, child.end_qc_tp);
  200. child.end_gather_percent = this.ctx.helper.mul(this.ctx.helper.div(child.end_gather_tp, child.final_tp, 4), 100);
  201. child.real_tp = this.ctx.helper.mul(child.real_qty, child.unit_price, tpDecimal);
  202. child.estimate_tp = this.ctx.helper.mul(child.estimate_qty, child.unit_price, tpDecimal);
  203. child.bgl_code = this.ctx.helper._.uniq(this.ctx.helper._.map(child.changes, 'c_code')).join(';');
  204. }
  205. _calculateNode(node, children) {
  206. for (const child of children) {
  207. node.total_price = this.ctx.helper.add(node.total_price, child.total_price);
  208. node.contract_tp = this.ctx.helper.add(node.contract_tp, child.contract_tp);
  209. node.qc_tp = this.ctx.helper.add(node.qc_tp, child.qc_tp);
  210. node.gather_tp = this.ctx.helper.add(node.gather_tp, child.gather_tp);
  211. node.pre_contract_tp = this.ctx.helper.add(node.pre_contract_tp, child.pre_contract_tp);
  212. node.pre_qc_tp = this.ctx.helper.add(node.pre_qc_tp, child.pre_qc_tp);
  213. node.pre_gather_tp = this.ctx.helper.add(node.pre_gather_tp, child.pre_gather_tp);
  214. node.end_contract_tp = this.ctx.helper.add(node.end_contract_tp, child.end_contract_tp);
  215. node.end_qc_tp = this.ctx.helper.add(node.end_qc_tp, child.end_qc_tp);
  216. node.end_gather_tp = this.ctx.helper.add(node.end_gather_tp, child.end_gather_tp);
  217. node.real_tp = this.ctx.helper.add(node.real_tp, child.real_tp);
  218. node.estimate_tp = this.ctx.helper.add(node.estimate_tp, child.estimate_tp);
  219. }
  220. if (node.dgn_qty1)
  221. node.dgn_price = this.ctx.helper.div(node.total_price, node.dgn_qty1, this.ctx.tender.info.decimal.up);
  222. node.final_tp = this.ctx.helper.add(node.total_price, node.end_qc_tp);
  223. node.end_gather_percent = this.ctx.helper.mul(this.ctx.helper.div(node.end_gather_tp, node.final_tp, 4), 100);
  224. }
  225. _recursiveCalcUnitNodes(nodes) {
  226. if (!nodes || !nodes instanceof Array || nodes.length === 0) return;
  227. for (const node of nodes) {
  228. this._recursiveCalcUnitNodes(node.children);
  229. if (node.children && node.children.length > 0) {
  230. this._calculateNode(node, node.children);
  231. } else {
  232. this._calculateChild(node);
  233. }
  234. if (node.drawing_code && node.drawing_code.length > 0)
  235. node.drawing_code_merge = node.drawing_code.join(mergeChar);
  236. if (node.memo && node.memo.length > 0)
  237. node.memo_merge = node.memo.join(mergeChar);
  238. if (node.postil && node.postil.length > 0)
  239. node.postil_merge = node.postil.join(mergeChar);
  240. }
  241. }
  242. _recursiveCalculateAndSort(nodes) {
  243. const self = this;
  244. if (!nodes || !nodes instanceof Array || nodes.length === 0) return;
  245. for (const node of nodes) {
  246. this._recursiveCalculateAndSort(node.children);
  247. if (node.unitTree) {
  248. node.unitTree.sortTreeNodeCustom(function (a, b) {
  249. if (a.b_code && a.b_code !== '') {
  250. return b.b_code && b.b_code !== '' ? self.ctx.helper.compareCode(a.b_code, b.b_code) : 1;
  251. } else {
  252. return b.b_code && b.b_code !== '' ? -1 : a.order - b.order;
  253. }
  254. });
  255. this._recursiveCalcUnitNodes(node.unitTree.children);
  256. this._calculateNode(node, node.unitTree.children, this.tpFields);
  257. } else if (node.children && node.children.length > 0) {
  258. this._calculateNode(node, node.children, this.tpFields);
  259. } else {
  260. this._calculateChild(node);
  261. }
  262. }
  263. }
  264. _calculateAndSortResult() {
  265. this.resultTree.sortTreeNode();
  266. this._recursiveCalculateAndSort(this.resultTree.children);
  267. }
  268. _generateCodeByWbsCode(wbsCode) {
  269. let needUpdate = false;
  270. for (const node of this.resultTree.nodes) {
  271. if (!node.unitTree) continue;
  272. let xmj = wbsCode.find(function (x) {
  273. return x.xmj_code === node.code;
  274. });
  275. for (const unit of node.unitTree.nodes) {
  276. if (!unit.pos_name || unit.pos_name === '') continue;
  277. if (!xmj) {
  278. xmj = {xmj_code: node.code, pos: []};
  279. wbsCode.push(xmj);
  280. needUpdate = true;
  281. }
  282. let index = xmj.pos.indexOf(unit.pos_name);
  283. if (index < 0) {
  284. unit.code = xmj.xmj_code + splitChar + (xmj.pos.length + 1);
  285. xmj.pos.push(unit.pos_name);
  286. needUpdate = true;
  287. } else {
  288. unit.code = xmj.xmj_code + splitChar + (index + 1);
  289. }
  290. }
  291. }
  292. return needUpdate;
  293. }
  294. _getResultData(filterFun) {
  295. const result = this.resultTree.getDefaultDatas(filterFun);
  296. for (const r of result) {
  297. if (r.unitTree) {
  298. r.unitTreeData = r.unitTree.getDefaultDatas(filterFun);
  299. delete r.unitTree;
  300. }
  301. }
  302. return result;
  303. }
  304. // 转换数据
  305. convert(filter) {
  306. this._recursiveConvertNode(this.bpcTree.children);
  307. this._calculateAndSortResult();
  308. this._generateCodeByWbsCode([]);
  309. switch (filter) {
  310. case 'cur':
  311. return this._getResultData(function (node) {
  312. for (const field of ['contract_tp', 'qc_tp', 'gather_tp']) {
  313. if (node[field]) {
  314. return false;
  315. }
  316. }
  317. return true;
  318. });
  319. case 'end':
  320. return this._getResultData(function (node) {
  321. for (const field of ['end_contract_tp', 'end_qc_tp', 'end_gather_tp']) {
  322. if (node[field]) {
  323. return false;
  324. }
  325. }
  326. return true;
  327. });
  328. default:
  329. return this._getResultData();
  330. }
  331. }
  332. /**
  333. * 转换数据,会根据wbsCodeHis生成计量单元的code
  334. * 如果计量单元在wbsCodeHis中没有历史,则会更新传入的wbsCodeHis,并返回[result, true], 反之返回[result, false]
  335. * @param wbsCodeHis
  336. * @returns {*[]}
  337. */
  338. convertByWbsCode(wbsCodeHis) {
  339. this._recursiveConvertNode(this.bpcTree.children);
  340. this._calculateAndSortResult();
  341. const needUpdate = this._generateCodeByWbsCode(wbsCodeHis);
  342. return [this._getResultData(), needUpdate];
  343. }
  344. }
  345. module.exports = BillsPosConvert;