budget_controller.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  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 LzString = require('lz-string');
  16. module.exports = app => {
  17. class BudgetController extends app.BaseController {
  18. /**
  19. * 概算投资
  20. *
  21. * @param ctx
  22. * @returns {Promise<void>}
  23. */
  24. async list(ctx) {
  25. try {
  26. const renderData = {
  27. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.list),
  28. auditConst,
  29. };
  30. renderData.budgetList = await ctx.service.budget.getAllDataByCondition({
  31. where: { pid: ctx.session.sessionProject.id },
  32. orders: [['name', 'asc']],
  33. });
  34. renderData.budgetStd = await ctx.service.budgetStd.getDataByProjectId(ctx.session.sessionProject.id);
  35. renderData.tenderList = await ctx.service.tender.getList4Select('stage');
  36. await this.layout('budget/list.ejs', renderData, 'budget/list_modal.ejs');
  37. } catch (err) {
  38. ctx.log(err);
  39. }
  40. }
  41. async add(ctx) {
  42. try {
  43. const data = JSON.parse(ctx.request.body.data);
  44. if (!data.name || data.name.length > 100) throw '项目名称有误';
  45. if (!data.std_id) throw '概预算标准有误';
  46. const result = await ctx.service.budget.add(data);
  47. ctx.body = { err: 0, msg: '', data: result };
  48. } catch (err) {
  49. ctx.log(err);
  50. ctx.ajaxErrorBody(err, '新建项目失败');
  51. }
  52. }
  53. async del(ctx) {
  54. try {
  55. const data = JSON.parse(ctx.request.body.data);
  56. if (!data.id) throw '参数有误';
  57. const result = await ctx.service.budget.deleteBudgetNoBackup(data.id);
  58. ctx.body = { err: 0, msg: '', data: result };
  59. } catch(err) {
  60. ctx.log(err);
  61. ctx.ajaxErrorBody(err, '删除项目失败');
  62. }
  63. }
  64. async save(ctx) {
  65. try {
  66. const data = JSON.parse(ctx.request.body.data);
  67. if (!data.id) throw '参数有误';
  68. let result = null;
  69. if (data.name !== undefined) {
  70. if (!data.name || data.name.length > 100) throw '项目名称有误';
  71. result = await ctx.service.budget.save({ id: data.id, name: data.name });
  72. } else if (data.rela_tender !== undefined) {
  73. result = await ctx.service.budget.save({ id: data.id, rela_tender: data.rela_tender });
  74. }
  75. ctx.body = { err: 0, msg: '', data: result };
  76. } catch(err) {
  77. ctx.log(err);
  78. ctx.ajaxErrorBody(err, '保存数据失败');
  79. }
  80. }
  81. async compare(ctx) {
  82. try {
  83. const renderData = {
  84. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.compare),
  85. auditConst,
  86. };
  87. renderData.tenderList = await ctx.service.tender.getList4Select('stage');
  88. await this.layout('budget/compare.ejs', renderData, 'budget/compare_modal.ejs');
  89. } catch (err) {
  90. ctx.log(err);
  91. }
  92. }
  93. async compareLoad(ctx) {
  94. try {
  95. const gu = await ctx.service.budgetGu.getData(ctx.budget.id);
  96. const gai = await ctx.service.budgetGai.getData(ctx.budget.id);
  97. const yu = await ctx.service.budgetYu.getData(ctx.budget.id);
  98. ctx.body = { err: 0, msg: '', data: { gu, gai, yu } }
  99. } catch (err) {
  100. ctx.log(err);
  101. ctx.ajaxErrorBody(err, '获取数据错误');
  102. }
  103. }
  104. async compareFinal(ctx) {
  105. try {
  106. const data = JSON.parse(ctx.request.body.data);
  107. const final = [], helper = this.ctx.helper;
  108. for (const id of data.id) {
  109. const bills = await ctx.service.ledger.getFinalData(id, ['id', 'ledger_id', 'ledger_pid', 'level', 'order', 'full_path', 'is_leaf',
  110. 'code', 'b_code', 'name', 'unit', 'dgn_qty1', 'dgn_qty2', 'total_price']);
  111. // 使用完成数据对比
  112. // const stage = ctx.service.stage.getLastestCompleteStage(id);
  113. // const finalBills = ctx.service.stageBillsFinal.getFinalData(id, stage.id);
  114. // ctx.helper.assignRelaData(bills, [
  115. // { data: finalBills, fields: ['contract_tp', 'qc_tp', 'used'], prefix: 'end_', relaId: 'lid' },
  116. // ]);
  117. // 使用最新一期对比
  118. const stage = await ctx.service.stage.getLastestStage(id);
  119. if (stage.status === auditConst.stage.status.checked) {
  120. const finalBills = await ctx.service.stageBillsFinal.getFinalData({id}, stage.order);
  121. ctx.helper.assignRelaData(bills, [
  122. { data: finalBills, fields: ['contract_tp', 'qc_tp'], prefix: 'end_', relaId: 'lid' },
  123. ]);
  124. bills.forEach(b => {
  125. b.end_gather_tp = helper.add(b.end_qc_tp, b.end_contract_tp);
  126. delete b.end_contract_tp;
  127. delete b.end_qc_tp;
  128. });
  129. } else {
  130. await ctx.service.stage.doCheckStage(stage);
  131. const curBills = stage.readOnly
  132. ? await ctx.service.stageBills.getAuditorStageData2(id, stage.id, stage.curTimes, stage.curOrder)
  133. : await ctx.service.stageBills.getLastestStageData2(id, stage.id);
  134. const preBills = stage.order > 1 ? await ctx.service.stageBillsFinal.getFinalData({id}, stage.order - 1) : [];
  135. ctx.helper.assignRelaData(bills, [
  136. { data: curBills, fields: ['contract_tp', 'qc_tp'], prefix: '', relaId: 'lid' },
  137. { data: preBills, fields: ['contract_tp', 'qc_tp'], prefix: 'pre_', relaId: 'lid' },
  138. ]);
  139. bills.forEach(b => {
  140. b.end_gather_tp = helper.sum([b.qc_tp, b.contract_tp, b.pre_qc_tp, b.pre_contract_tp]);
  141. delete b.contract_tp;
  142. delete b.qc_tp;
  143. delete b.pre_contract_tp;
  144. delete b.pre_qc_tp;
  145. });
  146. }
  147. final.push(bills);
  148. }
  149. ctx.body = { err: 0, msg: '', data: final }
  150. } catch (err) {
  151. ctx.log(err);
  152. ctx.ajaxErrorBody(err, '获取决算数据错误');
  153. }
  154. }
  155. _getSpreadSetting(type) {
  156. const spreadSetting = {
  157. cols: [
  158. {title: '项目节编号', colSpan: '1', rowSpan: '2', field: 'code', hAlign: 0, width: 180, formatter: '@', cellType: 'tree'},
  159. {title: '清单编号', colSpan: '1', rowSpan: '2', field: 'b_code', hAlign: 0, width: 120, formatter: '@'},
  160. {title: '名称', colSpan: '1', rowSpan: '2', field: 'name', hAlign: 0, width: 230, formatter: '@'},
  161. {title: '单位', colSpan: '1', rowSpan: '2', field: 'unit', hAlign: 1, width: 50, formatter: '@', cellType: 'unit'},
  162. {title: '设计数量|数量1', colSpan: '2|1', rowSpan: '1|1', field: 'dgn_qty1', hAlign: 2, width: 80, type: 'Number'},
  163. {title: '|数量2', colSpan: '|1', rowSpan: '|1', field: 'dgn_qty2', hAlign: 2, width: 80, type: 'Number'},
  164. {title: '经济指标', colSpan: '1', rowSpan: '2', field: 'dgn_price', hAlign: 2, width: 80, type: 'Number', readOnly: true},
  165. {title: '清单数量', colSpan: '1', rowSpan: '2', field: 'quantity', hAlign: 2, width: 80, type: 'Number'},
  166. {title: '单价', colSpan: '1', rowSpan: '2', field: 'unit_price', hAlign: 2, width: 80, type: 'Number'},
  167. {title: '金额', colSpan: '1', rowSpan: '2', field: 'total_price', hAlign: 2, width: 80, type: 'Number'},
  168. {title: '图册号', colSpan: '1', rowSpan: '2', field: 'drawing_code', hAlign: 0, width: 100, formatter: '@'},
  169. {title: '备注', colSpan: '1', rowSpan: '2', field: 'memo', hAlign: 0, width: 100, formatter: '@'},
  170. ],
  171. emptyRows: 3,
  172. headRows: 2,
  173. headRowHeight: [25, 25],
  174. defaultRowHeight: 21,
  175. headerFont: '12px 微软雅黑',
  176. font: '12px 微软雅黑',
  177. };
  178. // todo 根据设置判断预算是否需要清单
  179. if (type !== 'yu') {
  180. spreadSetting.cols = spreadSetting.cols.filter(x => {
  181. return ['b_code', 'quantity', 'unit_price'].indexOf(x.field) < 0;
  182. });
  183. }
  184. return spreadSetting;
  185. }
  186. _getRelaService(type) {
  187. switch(type) {
  188. case 'gu': return this.ctx.service.budgetGu;
  189. case 'gai': return this.ctx.service.budgetGai;
  190. case 'yu': return this.ctx.service.budgetYu;
  191. default: return null;
  192. }
  193. }
  194. async _getNeedGcl() {
  195. if (!this.ctx.params.btype) throw '参数错误';
  196. const funRela = this.ctx.service.project.getFunRela(this.ctx.session.sessionProject.id);
  197. return this.ctx.params.btype === 'yu' && !!funRela.needGcl;
  198. }
  199. async detail(ctx) {
  200. try {
  201. const renderData = {
  202. spreadSetting: this._getSpreadSetting(ctx.params.btype),
  203. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.budget.detail),
  204. needGcl: await this._getNeedGcl(),
  205. };
  206. [renderData.stdBills, renderData.stdChapters] = await ctx.service.budgetStd.getStdList(ctx.budget.std_id, ctx.params.btype);
  207. await this.layout('budget/detail.ejs', renderData, 'budget/detail_modal.ejs');
  208. } catch (err) {
  209. ctx.log(err);
  210. }
  211. }
  212. async detailLoad(ctx) {
  213. try {
  214. const relaService = this._getRelaService(ctx.params.btype);
  215. ctx.body = {
  216. err: 0, msg: '',
  217. data: await relaService.getData(ctx.budget.id),
  218. }
  219. } catch (err) {
  220. ctx.log(err);
  221. ctx.ajaxErrorBody(err, '获取数据错误');
  222. }
  223. }
  224. async _billsBase(relaService, type, data) {
  225. if (isNaN(data.id) || data.id <= 0) throw '数据错误';
  226. if (type !== 'add') {
  227. if (isNaN(data.count) || data.count <= 0) data.count = 1;
  228. }
  229. switch (type) {
  230. case 'add':
  231. return await relaService.addNodeBatch(this.ctx.budget.id, data.id, {}, data.count);
  232. case 'delete':
  233. return await relaService.delete(this.ctx.budget.id, data.id, data.count);
  234. case 'up-move':
  235. return await relaService.upMoveNode(this.ctx.budget.id, data.id, data.count);
  236. case 'down-move':
  237. return await relaService.downMoveNode(this.ctx.budget.id, data.id, data.count);
  238. case 'up-level':
  239. return await relaService.upLevelNode(this.ctx.budget.id, data.id, data.count);
  240. case 'down-level':
  241. return await relaService.downLevelNode(this.ctx.budget.id, data.id, data.count);
  242. }
  243. }
  244. async _addStd(relaService, data) {
  245. if ((isNaN(data.id) || data.id <= 0) || !data.stdType || !data.stdNode) throw '参数错误';
  246. let stdLib, addType;
  247. switch (data.stdType) {
  248. case 'xmj':
  249. stdLib = this.ctx.service.stdXmj;
  250. addType = stdDataAddType.withParent;
  251. break;
  252. case 'gcl':
  253. stdLib = this.ctx.service.stdGcl;
  254. const selectNode = await relaService.getDataByKid(this.ctx.budget.id, data.id);
  255. addType = selectNode.b_code ? stdDataAddType.next : stdDataAddType.child;
  256. break;
  257. default:
  258. throw '未知标准库';
  259. }
  260. const stdData = await stdLib.getDataByDataId(data.stdLibId, data.stdNode);
  261. switch (addType) {
  262. case stdDataAddType.child:
  263. return await relaService.addStdNodeAsChild(this.ctx.budget.id, data.id, stdData);
  264. case stdDataAddType.next:
  265. return await relaService.addStdNode(this.ctx.budget.id, data.id, stdData);
  266. case stdDataAddType.withParent:
  267. return await relaService.addStdNodeWithParent(this.ctx.budget.id, stdData, stdLib);
  268. default:
  269. throw '未知添加方式';
  270. }
  271. }
  272. async _pasteBlock(relaService, data) {
  273. if ((isNaN(data.id) || data.id <= 0) ||
  274. (!data.tid && data.tid <= 0) ||
  275. (!data.block || data.block.length <= 0)) throw '参数错误';
  276. return await relaService.pasteBlockData(this.ctx.budget.id, data.id, data.block);
  277. }
  278. async detailUpdate(ctx) {
  279. try {
  280. const relaService = this._getRelaService(ctx.params.btype);
  281. if (!ctx.budget) throw '项目数据错误';
  282. const data = JSON.parse(ctx.request.body.data);
  283. if (!data.postType || !data.postData) throw '数据错误';
  284. const responseData = { err: 0, msg: '', data: {} };
  285. switch (data.postType) {
  286. case 'add':
  287. case 'delete':
  288. case 'up-move':
  289. case 'down-move':
  290. case 'up-level':
  291. case 'down-level':
  292. responseData.data = await this._billsBase(relaService, data.postType, data.postData);
  293. break;
  294. case 'update':
  295. ctx.helper.checkDgnQtyPrecision(data.postData);
  296. responseData.data = await relaService.updateCalc(ctx.budget.id, data.postData);
  297. break;
  298. case 'add-std':
  299. responseData.data = await this._addStd(relaService, data.postData);
  300. break;
  301. case 'paste-block':
  302. responseData.data = await this._pasteBlock(relaService, data.postData);
  303. break;
  304. default:
  305. throw '未知操作';
  306. }
  307. ctx.body = responseData;
  308. } catch (err) {
  309. this.log(err);
  310. ctx.body = this.ajaxErrorBody(err, '数据错误');
  311. }
  312. }
  313. async detailUploadExcel(ctx) {
  314. try {
  315. const relaService = this._getRelaService(ctx.params.btype);
  316. const needGcl = await this._getNeedGcl();
  317. const ueType = ctx.params.ueType;
  318. const compressData = ctx.request.body.data;
  319. const data = JSON.parse(LzString.decompressFromUTF16(compressData));
  320. const responseData = { err: 0, msg: '', data: {} };
  321. switch (ueType) {
  322. case 'tz':
  323. const templateId = await this.ctx.service.budgetStd.getTemplateId(this.ctx.budget.std_id, ctx.params.btype);
  324. responseData.data = await relaService.importExcel(templateId, data.sheet, needGcl, data.filter);
  325. break;
  326. case 'gcl2xmj':
  327. responseData.data = await relaService.importGclExcel(data.id, data.sheet);
  328. break;
  329. default:
  330. throw '数据错误';
  331. }
  332. ctx.body = responseData;
  333. } catch (err) {
  334. this.log(err);
  335. ctx.body = { err: 1, msg: err.toString(), data: null };
  336. }
  337. }
  338. }
  339. return BudgetController;
  340. };