budget_controller.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  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. module.exports = app => {
  19. class BudgetController extends app.BaseController {
  20. /**
  21. * 概算投资
  22. *
  23. * @param ctx
  24. * @returns {Promise<void>}
  25. */
  26. async list(ctx) {
  27. try {
  28. const renderData = {
  29. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.list),
  30. auditConst,
  31. };
  32. renderData.budgetList = await ctx.service.subProject.getBudgetProject(ctx.session.sessionProject.id, ctx.session.sessionUser.accountId, ctx.session.sessionUser.is_admin);
  33. renderData.budgetStd = await ctx.service.budgetStd.getDataByProjectId(ctx.session.sessionProject.id);
  34. for (const bl of renderData.budgetList) {
  35. if (bl.is_folder) continue;
  36. bl.gu_tp = await ctx.service.budgetGu.getSumTp(bl.budget_id);
  37. bl.gai_tp = await ctx.service.budgetGai.getSumTp(bl.budget_id);
  38. bl.yu_tp = await ctx.service.budgetYu.getSumTp(bl.budget_id);
  39. bl.zb_tp = await ctx.service.budgetZb.getSumTp(bl.budget_id);
  40. }
  41. renderData.tenderList = await ctx.service.tender.getList4Select('stage');
  42. renderData.categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  43. await this.layout('budget/list.ejs', renderData, 'budget/list_modal.ejs');
  44. } catch (err) {
  45. ctx.log(err);
  46. }
  47. }
  48. async budgetInfo(ctx) {
  49. try {
  50. // 获取变更费用前10的变更令
  51. const changeList = await ctx.service.change.getListByBudgetInfo(ctx.budget.rela_tender);
  52. // 获取变更后总金额
  53. const total_change_tp = await ctx.service.change.getTotalTpByBudgetInfo(ctx.budget.rela_tender);
  54. const renderData = {
  55. changeList,
  56. changeConst,
  57. total_change_tp,
  58. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.info),
  59. auditConst,
  60. };
  61. // const relaTenderId = ctx.helper._.map(ctx.budget.rela_tender.split(','), ctx.helper._.toInteger);
  62. // const tenderList = await ctx.service.tender.getList4Select('stage');
  63. // renderData.tenderList = relaTenderId.length > 0 ? tenderList.filter(x => {
  64. // return relaTenderId.indexOf(x.id) >= 0;
  65. // }) : tenderList;
  66. // renderData.tenderList = renderData.tenderList.map(y => {
  67. // return { id: y.id, name: y.name, lastStageOrder: y.lastStage.order, lastStageStatus: auditConst.stage.statusString[y.lastStage.status], category: y.category };
  68. // });
  69. // renderData.categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  70. await this.layout('budget/info.ejs', renderData);
  71. } catch (err) {
  72. ctx.log(err);
  73. }
  74. }
  75. async compare(ctx) {
  76. try {
  77. const renderData = {
  78. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.compare),
  79. auditConst,
  80. };
  81. const relaTenderId = ctx.helper._.map(ctx.budget.rela_tender.split(','), ctx.helper._.toInteger);
  82. const tenderList = await ctx.service.tender.getList4Select('stage');
  83. renderData.tenderList = relaTenderId.length > 0 ? tenderList.filter(x => {
  84. return relaTenderId.indexOf(x.id) >= 0;
  85. }) : tenderList;
  86. renderData.tenderList = renderData.tenderList.map(y => {
  87. return { id: y.id, name: y.name, lastStageOrder: y.lastStage.order, lastStageStatus: auditConst.stage.statusString[y.lastStage.status], category: y.category };
  88. });
  89. renderData.categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  90. await this.layout('budget/compare.ejs', renderData, 'budget/compare_modal.ejs');
  91. } catch (err) {
  92. ctx.log(err);
  93. }
  94. }
  95. async compareLoad(ctx) {
  96. try {
  97. const data = {};
  98. if (ctx.budget.final_id) {
  99. data.final = await ctx.service.budgetFinal.getAllDataByCondition({ where: { final_id: ctx.budget.final_id } });
  100. data.finalInfo = await ctx.service.budgetFinalList.getFinal(ctx.budget.final_id);
  101. } else {
  102. data.gu = await ctx.service.budgetGu.getData(ctx.budget.id);
  103. data.gai = await ctx.service.budgetGai.getData(ctx.budget.id);
  104. data.yu = await ctx.service.budgetYu.getData(ctx.budget.id);
  105. data.zb = await ctx.service.budgetZb.getData(ctx.budget.id);
  106. }
  107. ctx.body = { err: 0, msg: '', data };
  108. } catch (err) {
  109. ctx.log(err);
  110. ctx.ajaxErrorBody(err, '获取数据错误');
  111. }
  112. }
  113. async compareFinal(ctx) {
  114. try {
  115. const data = JSON.parse(ctx.request.body.data);
  116. if (ctx.budget.final_id && data.final_id !== ctx.budget.final_id) {
  117. const final = await ctx.service.budgetFinal.getAllDataByCondition({ where: { final_id: ctx.budget.final_id } });
  118. const finalInfo = await ctx.service.budgetFinalList.getFinal(ctx.budget.final_id);
  119. ctx.body = { err: 0, msg: `决算数据已在${ctx.moment(finalInfo.update_time).format('YYYY-DD-MM HH:mm:ss')}更新,请先查看后再决定是否生成`, data: { final, finalInfo } };
  120. } else {
  121. const finalInfo = await ctx.service.budgetFinalList.addFinal(ctx.budget, data.id);
  122. const final = await ctx.service.budget.doFinal(ctx.budget, finalInfo);
  123. ctx.body = { err: 0, msg: '', data: { final, finalInfo } };
  124. }
  125. } catch (err) {
  126. ctx.log(err);
  127. ctx.ajaxErrorBody(err, '获取决算数据错误');
  128. }
  129. }
  130. _getSpreadSetting(needGcl) {
  131. const spreadSetting = {
  132. cols: [
  133. {title: '项目节编号', colSpan: '1', rowSpan: '2', field: 'code', hAlign: 0, width: 180, formatter: '@', cellType: 'tree'},
  134. {title: '清单编号', colSpan: '1', rowSpan: '2', field: 'b_code', hAlign: 0, width: 120, formatter: '@'},
  135. {title: '名称', colSpan: '1', rowSpan: '2', field: 'name', hAlign: 0, width: 230, formatter: '@'},
  136. {title: '单位', colSpan: '1', rowSpan: '2', field: 'unit', hAlign: 1, width: 50, formatter: '@', cellType: 'unit', comboEdit: true},
  137. {title: '设计数量|数量1', colSpan: '2|1', rowSpan: '1|1', field: 'dgn_qty1', hAlign: 2, width: 80, type: 'Number'},
  138. {title: '|数量2', colSpan: '|1', rowSpan: '|1', field: 'dgn_qty2', hAlign: 2, width: 80, type: 'Number'},
  139. {title: '经济指标', colSpan: '1', rowSpan: '2', field: 'dgn_price', hAlign: 2, width: 80, type: 'Number', readOnly: true},
  140. {title: '清单数量', colSpan: '1', rowSpan: '2', field: 'quantity', hAlign: 2, width: 80, type: 'Number'},
  141. {title: '单价', colSpan: '1', rowSpan: '2', field: 'unit_price', hAlign: 2, width: 80, type: 'Number'},
  142. {title: '金额', colSpan: '1', rowSpan: '2', field: 'total_price', hAlign: 2, width: 80, type: 'Number'},
  143. {title: '图册号', colSpan: '1', rowSpan: '2', field: 'drawing_code', hAlign: 0, width: 100, formatter: '@'},
  144. {title: '备注', colSpan: '1', rowSpan: '2', field: 'memo', hAlign: 0, width: 100, formatter: '@'},
  145. ],
  146. emptyRows: 3,
  147. headRows: 2,
  148. headRowHeight: [25, 25],
  149. defaultRowHeight: 21,
  150. headerFont: '12px 微软雅黑',
  151. font: '12px 微软雅黑',
  152. localCache: { key: 'budget', colWidth: true },
  153. };
  154. if (!needGcl) {
  155. spreadSetting.cols = spreadSetting.cols.filter(x => {
  156. return ['b_code', 'quantity', 'unit_price'].indexOf(x.field) < 0;
  157. });
  158. }
  159. return spreadSetting;
  160. }
  161. _getRelaService(type) {
  162. switch(type) {
  163. case 'gu': return this.ctx.service.budgetGu;
  164. case 'gai': return this.ctx.service.budgetGai;
  165. case 'yu': return this.ctx.service.budgetYu;
  166. case 'zb': return this.ctx.service.budgetZb;
  167. default: return null;
  168. }
  169. }
  170. async _getNeedGcl() {
  171. if (!this.ctx.params.btype) throw '参数错误';
  172. const funRela = await this.ctx.service.project.getFunRela(this.ctx.session.sessionProject.id);
  173. return ['yu', 'zb'].indexOf(this.ctx.params.btype) >= 0 && !!funRela.needGcl;
  174. }
  175. async detail(ctx) {
  176. try {
  177. const needGcl = await this._getNeedGcl();
  178. const renderData = {
  179. spreadSetting: this._getSpreadSetting(needGcl),
  180. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.detail),
  181. needGcl,
  182. };
  183. [renderData.stdBills, renderData.stdChapters] = await ctx.service.budgetStd.getStdList(ctx.budget.std_id, ctx.params.btype);
  184. await this.layout('budget/detail.ejs', renderData, 'budget/detail_modal.ejs');
  185. } catch (err) {
  186. ctx.log(err);
  187. }
  188. }
  189. async detailLoad(ctx) {
  190. try {
  191. const relaService = this._getRelaService(ctx.params.btype);
  192. ctx.body = {
  193. err: 0, msg: '',
  194. data: await relaService.getData(ctx.budget.id),
  195. }
  196. } catch (err) {
  197. ctx.log(err);
  198. ctx.ajaxErrorBody(err, '获取数据错误');
  199. }
  200. }
  201. async _billsBase(relaService, type, data) {
  202. if (isNaN(data.id) || data.id <= 0) throw '数据错误';
  203. if (type !== 'add') {
  204. if (isNaN(data.count) || data.count <= 0) data.count = 1;
  205. }
  206. switch (type) {
  207. case 'add':
  208. return await relaService.addNodeBatch(this.ctx.budget.id, data.id, {}, data.count);
  209. case 'delete':
  210. return await relaService.delete(this.ctx.budget.id, data.id, data.count);
  211. case 'up-move':
  212. return await relaService.upMoveNode(this.ctx.budget.id, data.id, data.count);
  213. case 'down-move':
  214. return await relaService.downMoveNode(this.ctx.budget.id, data.id, data.count);
  215. case 'up-level':
  216. return await relaService.upLevelNode(this.ctx.budget.id, data.id, data.count);
  217. case 'down-level':
  218. return await relaService.downLevelNode(this.ctx.budget.id, data.id, data.count);
  219. }
  220. }
  221. async _addStd(relaService, data) {
  222. if ((isNaN(data.id) || data.id <= 0) || !data.stdType || !data.stdNode) throw '参数错误';
  223. let stdLib, addType;
  224. switch (data.stdType) {
  225. case 'xmj':
  226. stdLib = this.ctx.service.stdXmj;
  227. addType = stdDataAddType.withParent;
  228. break;
  229. case 'gcl':
  230. stdLib = this.ctx.service.stdGcl;
  231. const selectNode = await relaService.getDataByKid(this.ctx.budget.id, data.id);
  232. addType = selectNode.b_code ? stdDataAddType.next : stdDataAddType.child;
  233. break;
  234. default:
  235. throw '未知标准库';
  236. }
  237. const stdData = await stdLib.getDataByDataId(data.stdLibId, data.stdNode);
  238. switch (addType) {
  239. case stdDataAddType.child:
  240. return await relaService.addStdNodeAsChild(this.ctx.budget.id, data.id, stdData);
  241. case stdDataAddType.next:
  242. return await relaService.addStdNode(this.ctx.budget.id, data.id, stdData);
  243. case stdDataAddType.withParent:
  244. return await relaService.addStdNodeWithParent(this.ctx.budget.id, stdData, stdLib);
  245. default:
  246. throw '未知添加方式';
  247. }
  248. }
  249. async _pasteBlock(relaService, data) {
  250. if ((isNaN(data.id) || data.id <= 0) ||
  251. (!data.tid && data.tid <= 0) ||
  252. (!data.block || data.block.length <= 0)) throw '参数错误';
  253. return await relaService.pasteBlockData(this.ctx.budget.id, data.id, data.block);
  254. }
  255. async detailUpdate(ctx) {
  256. try {
  257. if (!ctx.budget) throw '项目数据错误';
  258. if (ctx.budget.readOnly) throw '你无权修改数据';
  259. const relaService = this._getRelaService(ctx.params.btype);
  260. const data = JSON.parse(ctx.request.body.data);
  261. if (!data.postType || !data.postData) throw '数据错误';
  262. const responseData = { err: 0, msg: '', data: {} };
  263. switch (data.postType) {
  264. case 'add':
  265. case 'delete':
  266. case 'up-move':
  267. case 'down-move':
  268. case 'up-level':
  269. case 'down-level':
  270. responseData.data = await this._billsBase(relaService, data.postType, data.postData);
  271. break;
  272. case 'update':
  273. ctx.helper.checkDgnQtyPrecision(data.postData);
  274. responseData.data = await relaService.updateCalc(ctx.budget.id, data.postData);
  275. break;
  276. case 'add-std':
  277. responseData.data = await this._addStd(relaService, data.postData);
  278. break;
  279. case 'paste-block':
  280. responseData.data = await this._pasteBlock(relaService, data.postData);
  281. break;
  282. default:
  283. throw '未知操作';
  284. }
  285. ctx.body = responseData;
  286. } catch (err) {
  287. this.log(err);
  288. ctx.body = this.ajaxErrorBody(err, '数据错误');
  289. }
  290. }
  291. async detailUploadExcel(ctx) {
  292. try {
  293. if (!ctx.budget) throw '项目数据错误';
  294. if (ctx.budget.readOnly) throw '你无权导入数据';
  295. const relaService = this._getRelaService(ctx.params.btype);
  296. const needGcl = await this._getNeedGcl();
  297. const ueType = ctx.params.ueType;
  298. const compressData = ctx.request.body.data;
  299. const data = JSON.parse(LzString.decompressFromUTF16(compressData));
  300. const responseData = { err: 0, msg: '', data: {} };
  301. switch (ueType) {
  302. case 'tz':
  303. const templateId = await this.ctx.service.budgetStd.getTemplateId(this.ctx.budget.std_id, ctx.params.btype);
  304. responseData.data = await relaService.importExcel(templateId, data.sheet, needGcl, data.filter);
  305. break;
  306. case 'gcl2xmj':
  307. responseData.data = await relaService.importGclExcel(data.id, data.sheet);
  308. break;
  309. default:
  310. throw '数据错误';
  311. }
  312. ctx.body = responseData;
  313. } catch (err) {
  314. this.log(err);
  315. ctx.body = { err: 1, msg: err.toString(), data: null };
  316. }
  317. }
  318. async decimal(ctx) {
  319. try {
  320. if (!ctx.budget) throw '项目数据错误';
  321. if (ctx.budget.readOnly) throw '你无权修改小数位数';
  322. const data = JSON.parse(ctx.request.body.data);
  323. if (!data || !data.decimal || !data.page) throw '缺少必要参数';
  324. const refreshData = await ctx.service.budget.saveDecimal(data.decimal, data.page);
  325. ctx.body = { err: 0, msg: '', data: refreshData };
  326. } catch (err) {
  327. this.log(err);
  328. this.ajaxErrorBody(err, '保存小数位数失败');
  329. }
  330. }
  331. }
  332. return BudgetController;
  333. };