budget_controller.js 18 KB

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