budget_controller.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date 2021/10/27
  7. * @version
  8. */
  9. const stdDataAddType = {
  10. withParent: 1,
  11. child: 2,
  12. next: 3,
  13. };
  14. const auditConst = require('../const/audit');
  15. const changeConst = require('../const/change');
  16. const LzString = require('lz-string');
  17. const accountGroup = require('../const/account_group').group;
  18. const ValidDskProject = {
  19. gu: [15, 16],
  20. gai: [5],
  21. yu: [1, 18],
  22. zb: [4, 18, 19],
  23. ctrl: [4, 18, 19],
  24. };
  25. const DSK = require('../lib/dsk');
  26. const sendToWormhole = require('stream-wormhole');
  27. const streamToArray = require('stream-to-array');
  28. module.exports = app => {
  29. class BudgetController extends app.BaseController {
  30. /**
  31. * 概算投资
  32. *
  33. * @param ctx
  34. * @returns {Promise<void>}
  35. */
  36. async list(ctx) {
  37. try {
  38. if (!ctx.subProject.page_show.openBudget) throw '该功能已关闭或无法查看';
  39. const renderData = {
  40. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.list),
  41. auditConst,
  42. };
  43. renderData.budgetList = await ctx.service.subProject.getBudgetProject(ctx.session.sessionProject.id, ctx.session.sessionUser.accountId, ctx.session.sessionUser.is_admin);
  44. renderData.budgetStd = await ctx.service.budgetStd.getDataByProjectId(ctx.session.sessionProject.id);
  45. for (const bl of renderData.budgetList) {
  46. if (bl.is_folder) continue;
  47. bl.gu_tp = await ctx.service.budgetGu.getSumTp(bl.budget_id);
  48. bl.gai_tp = await ctx.service.budgetGai.getSumTp(bl.budget_id);
  49. bl.yu_tp = await ctx.service.budgetYu.getSumTp(bl.budget_id);
  50. bl.zb_tp = await ctx.service.budgetZb.getSumTp(bl.budget_id);
  51. bl.ctrl_tp = await ctx.service.budgetCtrl.getSumTp(bl.budget_id);
  52. }
  53. renderData.tenderList = await ctx.service.tender.getList4Select('stage');
  54. renderData.categoryData = await this.ctx.service.category.getAllCategory(ctx.subProject);
  55. await this.layout('budget/list.ejs', renderData, 'budget/list_modal.ejs');
  56. } catch (err) {
  57. console.log(err);
  58. ctx.log(err);
  59. ctx.session.postError = err.toString();
  60. ctx.redirect(this.menu.menu.dashboard.url);
  61. }
  62. }
  63. async budgetInfo(ctx) {
  64. try {
  65. // 获取变更费用前10的变更令
  66. const changeList = await ctx.service.change.getListByBudgetInfo(ctx.budget.rela_tender);
  67. // 获取变更后总金额
  68. const total_change_tp = await ctx.service.change.getTotalTpByBudgetInfo(ctx.budget.rela_tender);
  69. const renderData = {
  70. changeList,
  71. changeConst,
  72. total_change_tp,
  73. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.info),
  74. auditConst,
  75. };
  76. // const relaTenderId = ctx.helper._.map(ctx.budget.rela_tender.split(','), ctx.helper._.toInteger);
  77. // const tenderList = await ctx.service.tender.getList4Select('stage');
  78. // renderData.tenderList = relaTenderId.length > 0 ? tenderList.filter(x => {
  79. // return relaTenderId.indexOf(x.id) >= 0;
  80. // }) : tenderList;
  81. // renderData.tenderList = renderData.tenderList.map(y => {
  82. // return { id: y.id, name: y.name, lastStageOrder: y.lastStage.order, lastStageStatus: auditConst.stage.statusString[y.lastStage.status], category: y.category };
  83. // });
  84. // renderData.categoryData = await this.ctx.service.category.getAllCategory(ctx.subProject);
  85. await this.layout('budget/info.ejs', renderData);
  86. } catch (err) {
  87. ctx.log(err);
  88. }
  89. }
  90. async compare(ctx) {
  91. try {
  92. const renderData = {
  93. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.compare),
  94. auditConst,
  95. };
  96. const tenderList = await ctx.service.tender.getList4Select('stage');
  97. renderData.tenderList = tenderList.filter(x => {
  98. return !x.filter_budget;
  99. });
  100. renderData.tenderList = renderData.tenderList.map(y => {
  101. return { id: y.id, name: y.name, lastStageOrder: y.lastStage.order, lastStageStatus: auditConst.stage.statusString[y.lastStage.status], category: y.category };
  102. });
  103. renderData.categoryData = await this.ctx.service.category.getAllCategory(ctx.subProject);
  104. await this.layout('budget/compare.ejs', renderData, 'budget/compare_modal.ejs');
  105. } catch (err) {
  106. ctx.log(err);
  107. }
  108. }
  109. async compareLoad(ctx) {
  110. try {
  111. const data = {};
  112. if (ctx.budget.final_id) {
  113. data.final = await ctx.service.budgetFinal.getAllDataByCondition({ where: { final_id: ctx.budget.final_id } });
  114. data.finalInfo = await ctx.service.budgetFinalList.getFinal(ctx.budget.final_id);
  115. } else {
  116. data.gu = await ctx.service.budgetGu.getData(ctx.budget.id);
  117. data.gai = await ctx.service.budgetGai.getData(ctx.budget.id);
  118. data.yu = await ctx.service.budgetYu.getData(ctx.budget.id);
  119. data.zb = await ctx.service.budgetZb.getData(ctx.budget.id);
  120. data.ctrl = await ctx.service.budgetCtrl.getData(ctx.budget.id);
  121. }
  122. ctx.body = { err: 0, msg: '', data };
  123. } catch (err) {
  124. ctx.log(err);
  125. ctx.ajaxErrorBody(err, '获取数据错误');
  126. }
  127. }
  128. async compareFinal(ctx) {
  129. try {
  130. const data = JSON.parse(ctx.request.body.data);
  131. if (data.final_type !== ctx.budget.final_type) {
  132. await ctx.service.budget.defaultUpdate({ id: ctx.budget.id, final_type: data.final_type });
  133. ctx.budget.final_type = data.final_type;
  134. }
  135. if (ctx.budget.final_id && data.final_id !== ctx.budget.final_id) {
  136. const final = await ctx.service.budgetFinal.getAllDataByCondition({ where: { final_id: ctx.budget.final_id } });
  137. const finalInfo = await ctx.service.budgetFinalList.getFinal(ctx.budget.final_id);
  138. ctx.body = { err: 0, msg: `决算数据已在${ctx.moment(finalInfo.update_time).format('YYYY-DD-MM HH:mm:ss')}更新,请先查看后再决定是否生成`, data: { final, finalInfo } };
  139. } else {
  140. const finalInfo = await ctx.service.budgetFinalList.addFinal(ctx.budget, data.id);
  141. const final = await ctx.service.budget.doFinal(ctx.budget, finalInfo);
  142. ctx.body = { err: 0, msg: '', data: { final, finalInfo } };
  143. }
  144. } catch (err) {
  145. ctx.log(err);
  146. ctx.ajaxErrorBody(err, '获取决算数据错误');
  147. }
  148. }
  149. _getSpreadSetting(needGcl) {
  150. const spreadSetting = {
  151. cols: [
  152. {title: '项目节编号', colSpan: '1', rowSpan: '2', field: 'code', hAlign: 0, width: 180, formatter: '@', cellType: 'tree'},
  153. {title: '清单编号', colSpan: '1', rowSpan: '2', field: 'b_code', hAlign: 0, width: 120, formatter: '@'},
  154. {title: '名称', colSpan: '1', rowSpan: '2', field: 'name', hAlign: 0, width: 230, formatter: '@'},
  155. {title: '单位', colSpan: '1', rowSpan: '2', field: 'unit', hAlign: 1, width: 50, formatter: '@', cellType: 'unit', comboEdit: true},
  156. {title: '设计数量|数量1', colSpan: '2|1', rowSpan: '1|1', field: 'dgn_qty1', hAlign: 2, width: 80, type: 'Number'},
  157. {title: '|数量2', colSpan: '|1', rowSpan: '|1', field: 'dgn_qty2', hAlign: 2, width: 80, type: 'Number'},
  158. {title: '经济指标', colSpan: '1', rowSpan: '2', field: 'dgn_price', hAlign: 2, width: 80, type: 'Number', readOnly: true},
  159. {title: '清单数量', colSpan: '1', rowSpan: '2', field: 'quantity', hAlign: 2, width: 80, type: 'Number'},
  160. {title: '单价', colSpan: '1', rowSpan: '2', field: 'unit_price', hAlign: 2, width: 80, type: 'Number'},
  161. {title: '金额', colSpan: '1', rowSpan: '2', field: 'total_price', hAlign: 2, width: 80, type: 'Number'},
  162. {title: '图册号', colSpan: '1', rowSpan: '2', field: 'drawing_code', hAlign: 0, width: 100, formatter: '@'},
  163. {title: '备注', colSpan: '1', rowSpan: '2', field: 'memo', hAlign: 0, width: 100, formatter: '@'},
  164. ],
  165. emptyRows: 0,
  166. headRows: 2,
  167. headRowHeight: [25, 25],
  168. defaultRowHeight: 21,
  169. headerFont: '12px 微软雅黑',
  170. font: '12px 微软雅黑',
  171. localCache: { key: 'budget', colWidth: true },
  172. };
  173. if (!needGcl) {
  174. spreadSetting.cols = spreadSetting.cols.filter(x => {
  175. return ['b_code', 'quantity', 'unit_price'].indexOf(x.field) < 0;
  176. });
  177. }
  178. return spreadSetting;
  179. }
  180. _getRelaService(type) {
  181. switch(type) {
  182. case 'gu': return this.ctx.service.budgetGu;
  183. case 'gai': return this.ctx.service.budgetGai;
  184. case 'yu': return this.ctx.service.budgetYu;
  185. case 'zb': return this.ctx.service.budgetZb;
  186. case 'ctrl': return this.ctx.service.budgetCtrl;
  187. default: return null;
  188. }
  189. }
  190. async _getNeedGcl() {
  191. if (!this.ctx.params.btype) throw '参数错误';
  192. const funRela = this.ctx.subProject.fun_rela;
  193. return ['yu', 'zb', 'ctrl'].indexOf(this.ctx.params.btype) >= 0 && !!funRela.needGcl;
  194. }
  195. async detail(ctx) {
  196. try {
  197. const needGcl = await this._getNeedGcl();
  198. const renderData = {
  199. spreadSetting: this._getSpreadSetting(needGcl),
  200. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.detail),
  201. needGcl,
  202. validDskProjType: ValidDskProject[ctx.params.btype],
  203. };
  204. [renderData.stdBills, renderData.stdChapters] = await ctx.service.budgetStd.getStdList(ctx.budget.std_id, ctx.params.btype);
  205. await this.layout('budget/detail.ejs', renderData, 'budget/detail_modal.ejs');
  206. } catch (err) {
  207. ctx.log(err);
  208. }
  209. }
  210. async detailLoad(ctx) {
  211. try {
  212. const relaService = this._getRelaService(ctx.params.btype);
  213. ctx.body = {
  214. err: 0, msg: '',
  215. data: await relaService.getData(ctx.budget.id),
  216. }
  217. } catch (err) {
  218. ctx.log(err);
  219. ctx.ajaxErrorBody(err, '获取数据错误');
  220. }
  221. }
  222. async _billsBase(relaService, type, data) {
  223. if (isNaN(data.id) || data.id <= 0) throw '数据错误';
  224. if (type !== 'add') {
  225. if (isNaN(data.count) || data.count <= 0) data.count = 1;
  226. }
  227. switch (type) {
  228. case 'add':
  229. return await relaService.addNodeBatch(this.ctx.budget.id, data.id, {}, data.count);
  230. case 'delete':
  231. return await relaService.delete(this.ctx.budget.id, data.id, data.count);
  232. case 'up-move':
  233. return await relaService.upMoveNode(this.ctx.budget.id, data.id, data.count);
  234. case 'down-move':
  235. return await relaService.downMoveNode(this.ctx.budget.id, data.id, data.count);
  236. case 'up-level':
  237. return await relaService.upLevelNode(this.ctx.budget.id, data.id, data.count);
  238. case 'down-level':
  239. return await relaService.downLevelNode(this.ctx.budget.id, data.id, data.count);
  240. }
  241. }
  242. async _addStd(relaService, data) {
  243. if (!data.stdType || !data.stdNode) throw '参数错误';
  244. let stdLib, addType;
  245. switch (data.stdType) {
  246. case 'xmj':
  247. stdLib = this.ctx.service.stdXmj;
  248. addType = stdDataAddType.withParent;
  249. break;
  250. case 'gcl':
  251. stdLib = this.ctx.service.stdGcl;
  252. const selectNode = await relaService.getDataByKid(this.ctx.budget.id, data.id);
  253. addType = selectNode.b_code ? stdDataAddType.next : stdDataAddType.child;
  254. break;
  255. default:
  256. throw '未知标准库';
  257. }
  258. const stdData = await stdLib.getDataByDataId(data.stdLibId, data.stdNode);
  259. switch (addType) {
  260. case stdDataAddType.child:
  261. return await relaService.addStdNodeAsChild(this.ctx.budget.id, data.id, stdData);
  262. case stdDataAddType.next:
  263. return await relaService.addStdNode(this.ctx.budget.id, data.id, stdData);
  264. case stdDataAddType.withParent:
  265. return await relaService.addStdNodeWithParent(this.ctx.budget.id, stdData, stdLib);
  266. default:
  267. throw '未知添加方式';
  268. }
  269. }
  270. async _pasteBlock(relaService, data) {
  271. if ((isNaN(data.id) || data.id <= 0) ||
  272. (!data.tid && data.tid <= 0) ||
  273. (!data.block || data.block.length <= 0)) throw '参数错误';
  274. return await relaService.pasteBlockData(this.ctx.budget.id, data.id, data.block);
  275. }
  276. async detailUpdate(ctx) {
  277. try {
  278. if (!ctx.budget) throw '项目数据错误';
  279. if (ctx.budget.readOnly) throw '你无权修改数据';
  280. const relaService = this._getRelaService(ctx.params.btype);
  281. const data = JSON.parse(ctx.request.body.data);
  282. if (!data.postType || !data.postData) throw '数据错误';
  283. const responseData = { err: 0, msg: '', data: {} };
  284. switch (data.postType) {
  285. case 'add':
  286. case 'delete':
  287. case 'up-move':
  288. case 'down-move':
  289. case 'up-level':
  290. case 'down-level':
  291. responseData.data = await this._billsBase(relaService, data.postType, data.postData);
  292. break;
  293. case 'update':
  294. ctx.helper.checkDgnQtyPrecision(data.postData);
  295. responseData.data = await relaService.updateCalc(ctx.budget.id, data.postData);
  296. break;
  297. case 'add-std':
  298. responseData.data = await this._addStd(relaService, data.postData);
  299. break;
  300. case 'paste-block':
  301. responseData.data = await this._pasteBlock(relaService, data.postData);
  302. break;
  303. default:
  304. throw '未知操作';
  305. }
  306. ctx.body = responseData;
  307. } catch (err) {
  308. this.log(err);
  309. ctx.body = this.ajaxErrorBody(err, '数据错误');
  310. }
  311. }
  312. async detailUploadExcel(ctx) {
  313. try {
  314. if (!ctx.budget) throw '项目数据错误';
  315. if (ctx.budget.readOnly) throw '你无权导入数据';
  316. const relaService = this._getRelaService(ctx.params.btype);
  317. const needGcl = await this._getNeedGcl();
  318. const ueType = ctx.params.ueType;
  319. const compressData = ctx.request.body.data;
  320. const data = JSON.parse(LzString.decompressFromUTF16(compressData));
  321. const responseData = { err: 0, msg: '', data: {} };
  322. switch (ueType) {
  323. case 'tz':
  324. const templateId = await this.ctx.service.budgetStd.getTemplateId(this.ctx.budget.std_id, ctx.params.btype);
  325. responseData.data = await relaService.importExcel(templateId, data.sheet, needGcl, data.filter);
  326. break;
  327. case 'gcl2xmj':
  328. responseData.data = await relaService.importGclExcel(data.id, data.sheet);
  329. break;
  330. default:
  331. throw '数据错误';
  332. }
  333. ctx.body = responseData;
  334. } catch (err) {
  335. this.log(err);
  336. ctx.body = { err: 1, msg: err.toString(), data: null };
  337. }
  338. }
  339. async uploadYbp(ctx) {
  340. const stream = await ctx.getFileStream();
  341. try {
  342. const relaService = this._getRelaService(ctx.params.btype);
  343. const needGcl = await this._getNeedGcl();
  344. // 读取字节流
  345. const parts = await streamToArray(stream);
  346. // 转化为buffer
  347. const buffer = Buffer.concat(parts);
  348. const bills = await relaService.importYbp(ctx.budget, buffer.toString(), needGcl);
  349. ctx.body = { err: 0, msg: '', data: bills };
  350. } catch (err) {
  351. await sendToWormhole(stream);
  352. ctx.log(err);
  353. ctx.ajaxErrorBody(err, '导入计价文件出错');
  354. }
  355. }
  356. async importDsk(ctx) {
  357. try {
  358. const relaService = this._getRelaService(ctx.params.btype);
  359. const needGcl = await this._getNeedGcl();
  360. const data = JSON.parse(ctx.request.body.data);
  361. if (!data.subjects || data.subjects.length === 0) throw '无导入计价数据';
  362. // 拼凑ybp数据
  363. const ybpData = { header: {}, subjects: [] };
  364. const dsk = new DSK(ctx);
  365. for (const s of data.subjects) {
  366. const subject = { project: { ID: s.ID, parentID: s.parentID, type: s.type, name: s.name, compilationID: s.compilationId }, rations: [], };
  367. subject.bills = await dsk.getProjectBills(s.compilationId, s.subjectId);
  368. // 拼凑fees.marketCommon.totalFee,fees.marketLabour.totalFee防止出错
  369. subject.bills.forEach(b => {
  370. b.fees = {
  371. marketCommon: { unitPrice: b.unitPrice, totalFee: b.totalPrice },
  372. marketLabour: { unitPrice: 0, totalFee: 0 },
  373. };
  374. });
  375. ybpData.subjects.push(subject);
  376. }
  377. const bills = await relaService.importYbpData(ctx.budget, ybpData, needGcl);
  378. ctx.body = { err: 0, msg: '', data: bills };
  379. } catch (err) {
  380. ctx.log(err);
  381. ctx.ajaxErrorBody(err, '导入计价分段出错');
  382. }
  383. }
  384. async init(ctx) {
  385. const relaService = this._getRelaService(ctx.params.btype);
  386. try {
  387. const data = await relaService.getData(ctx.budget.id);
  388. if (data.length > 0) throw '已有数据,请勿重复初始化';
  389. const budgetStd = await this.ctx.service.budgetStd.getDataById(this.ctx.budget.std_id);
  390. await relaService.initByTemplate(null, this.ctx.budget.id, budgetStd[ctx.params.btype + '_template_id']);
  391. } catch(err) {
  392. ctx.log(err);
  393. ctx.postError(err, '初始化数据错误');
  394. }
  395. ctx.redirect(ctx.request.header.referer);
  396. }
  397. async decimal(ctx) {
  398. try {
  399. if (!ctx.budget) throw '项目数据错误';
  400. if (ctx.budget.readOnly) throw '你无权修改小数位数';
  401. const data = JSON.parse(ctx.request.body.data);
  402. if (!data || !data.decimal || !data.page) throw '缺少必要参数';
  403. const refreshData = await ctx.service.budget.saveDecimal(data.decimal, data.page);
  404. ctx.body = { err: 0, msg: '', data: refreshData };
  405. } catch (err) {
  406. this.log(err);
  407. this.ajaxErrorBody(err, '保存小数位数失败');
  408. }
  409. }
  410. }
  411. return BudgetController;
  412. };