budget_controller.js 20 KB

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