stage_controller.js 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date 2018/6/20
  7. * @version
  8. */
  9. const moment = require('moment');
  10. const auditConst = require('../const/audit').stage;
  11. const spreadConst = require('../const/spread');
  12. const tenderConst = require('../const/tender');
  13. const payConst = require('../const/deal_pay.js');
  14. const changeConst = require('../const/change');
  15. const measureType = tenderConst.measureType;
  16. const path = require('path');
  17. const PayCalculator = require('../lib/pay_calc');
  18. const accountGroup = require('../const/account_group').group;
  19. module.exports = app => {
  20. class StageController extends app.BaseController {
  21. /**
  22. * 构造函数
  23. *
  24. * @param {Object} ctx - egg全局变量
  25. * @return {void}
  26. */
  27. constructor(ctx) {
  28. super(ctx);
  29. ctx.showProject = true;
  30. ctx.showTender = true;
  31. ctx.showTitle = true;
  32. }
  33. /**
  34. * 获取通用的renderData(用于layout, Menu, subMenu部分)
  35. * @param ctx
  36. * @returns {{tender, tenderMenu, auditConst}}
  37. * @private
  38. */
  39. _getDefaultRenderData (ctx) {
  40. const data = {
  41. tender: ctx.tender.data,
  42. tenderMenu: JSON.parse(JSON.stringify(this.menu.stageMenu)),
  43. auditConst,
  44. measureType,
  45. preUrl: '/tender/' + ctx.tender.id + '/measure/stage/' + ctx.params.order,
  46. stage: ctx.stage,
  47. };
  48. data.tenderMenu.back.children[0].url = '/tender/' + ctx.tender.id + '/measure/stage';
  49. return data;
  50. }
  51. _stageReadOnly(stage) {
  52. return stage.status === auditConst.status.checked;
  53. }
  54. /**
  55. * 获取SpreadSetting
  56. * @private
  57. */
  58. _getSpreadSetting() {
  59. const _ = this.app._;
  60. function removeFieldCols(setting, cols) {
  61. _.remove(setting.cols, function (c) {
  62. return cols.indexOf(c.field) > -1;
  63. });
  64. }
  65. function setColFormat(cols, field, formatter) {
  66. const filterCols = cols.filter(function (c) {
  67. return c.field.search(field) !== -1;
  68. });
  69. for (const col of filterCols) {
  70. col.formatter = formatter;
  71. }
  72. }
  73. const tpFormatter = this.ctx.helper.getNumberFormatter(this.ctx.tender.info.decimal.tp);
  74. const upFormatter = this.ctx.helper.getNumberFormatter(2);
  75. const ledger = JSON.parse(JSON.stringify(spreadConst.stage.ledger));
  76. setColFormat(ledger.cols, 'unit_price', upFormatter);
  77. setColFormat(ledger.cols, 'total_price', tpFormatter);
  78. setColFormat(ledger.cols, 'tp', tpFormatter);
  79. const pos = JSON.parse(JSON.stringify(spreadConst.stage.pos));
  80. const tender = this.ctx.tender, stage = this.ctx.stage;
  81. if (this._stageReadOnly(stage)) {
  82. ledger.readOnly = true;
  83. pos.readOnly = true;
  84. }
  85. if (tender.data.measure_type === measureType.tz.value) {
  86. removeFieldCols(ledger, spreadConst.filterCols.tzWithoutCols);
  87. removeFieldCols(pos, spreadConst.filterCols.posTzWithoutCols);
  88. }
  89. if (!tender.info.display.ledger.dgnQty) {
  90. removeFieldCols(ledger, spreadConst.filterCols.dgnCols);
  91. }
  92. return [ledger, pos];
  93. }
  94. /**
  95. * 获取期数据
  96. * @param ctx
  97. * @returns {Promise<void>}
  98. * @private
  99. */
  100. async _getStage(ctx) {
  101. const stageOrder = parseInt(ctx.params.order);
  102. ctx.stage = await ctx.service.stage.getDataByCondition({
  103. tid: ctx.tender.id,
  104. order: stageOrder,
  105. });
  106. if (ctx.stage.status === auditConst.status.uncheck && ctx.stage.user_id !== ctx.session.sessionUser.accountId) {
  107. throw '数据错误';
  108. }
  109. ctx.stage.auditors = await ctx.service.stageAudit.getAuditors(ctx.stage.id, ctx.stage.times);
  110. ctx.stage.curAuditor = await ctx.service.stageAudit.getCurAuditor(ctx.stage.id, ctx.stage.times);
  111. ctx.stage.user = await ctx.service.projectAccount.getAccountInfoById(ctx.stage.user_id);
  112. ctx.stage.auditHistory = [];
  113. if (ctx.stage.times > 1) {
  114. for (let i = 1; i < ctx.stage.times; i++) {
  115. ctx.stage.auditHistory.push(await ctx.service.stageAudit.getAuditors(ctx.stage.id, i));
  116. }
  117. }
  118. }
  119. /**
  120. * 期计量页面 (Get)
  121. * @param {Object} ctx - egg全局变量
  122. * @returns {Promise<void>}
  123. */
  124. async index(ctx) {
  125. try {
  126. await this._getStage(ctx);
  127. const renderData = this._getDefaultRenderData(ctx);
  128. [renderData.ledgerSpread, renderData.posSpread] = this._getSpreadSetting();
  129. renderData.changeConst = changeConst;
  130. renderData.ledgerData = await ctx.service.ledger.getDataByTenderId(ctx.tender.id, -1);
  131. renderData.curStageData = await ctx.service.stageBills.getAuditorStageData(ctx.tender.id, ctx.stage.id, ctx.stage.times, 0);
  132. renderData.jsFiles = this.app.jsFiles.common.concat(this.app.jsFiles.stage.index);
  133. // 查询截止上期数据
  134. // if (ctx.stage.order > 0) {
  135. // renderData.preStageData
  136. // }
  137. if (ctx.stage.status === auditConst.status.uncheck && ctx.session.sessionUser.accountId === ctx.stage.user_id) {
  138. renderData.accountGroup = accountGroup;
  139. // 获取所有项目参与者
  140. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  141. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  142. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group'],
  143. });
  144. renderData.accountList = accountList;
  145. }
  146. await this.layout('stage/index.ejs', renderData, 'stage/modal.ejs');
  147. } catch (err) {
  148. this.log(err);
  149. ctx.redirect('/tender/' + ctx.tender.id + '/measure/stage');
  150. }
  151. }
  152. /**
  153. * 获取期数据(截止上期 & 本期) (Ajax)
  154. * @param ctx
  155. * @returns {Promise<void>}
  156. */
  157. async getStagePosData(ctx) {
  158. try {
  159. await this._getStage(ctx);
  160. const condition = JSON.parse(ctx.request.body.data) || {};
  161. condition.tid = ctx.tender.id;
  162. const responseData = {err: 0, msg: '', data: {}};
  163. responseData.data.pos = await ctx.service.pos.getPosData(condition);
  164. // todo 根据当前人,或指定对象查询数据
  165. responseData.data.curStageData = await ctx.service.stagePos.getAuditorStageData(ctx.tender.id,
  166. ctx.stage.id, ctx.stage.times, ctx.stage.curAuditor ? ctx.stage.curAuditor.order : 0,
  167. this.app._.map(responseData.data.pos, 'id'));
  168. ctx.body = responseData;
  169. } catch (err) {
  170. this.log(err);
  171. ctx.body = {err: 1, msg: err.toString(), data: null};
  172. }
  173. }
  174. /**
  175. * 保存数据 (Ajax)
  176. * @param ctx
  177. * @returns {Promise<void>}
  178. */
  179. async updateStageData(ctx) {
  180. try {
  181. await this._getStage(ctx);
  182. // 检查登录用户,是否可操作
  183. if (ctx.session.sessionUser.accountId === ctx.stage.user_id) {
  184. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  185. throw '该计量期当前您无权操作';
  186. }
  187. } else {
  188. // 检查是否可修改
  189. //const curAuditor = await ctx.service.stageAudit.get
  190. }
  191. const data = JSON.parse(ctx.request.body.data);
  192. const responseData = { err: 0, msg: '', data: {}, };
  193. if (data.pos) {
  194. responseData.data = await ctx.service.stagePos.updateStageData(data.pos);
  195. } else if (data.bills) {
  196. responseData.data.bills = await ctx.service.stageBills.updateStageData(data.bills);
  197. }
  198. ctx.body = responseData;
  199. } catch (err) {
  200. this.log(err);
  201. ctx.body = {err: 1, msg: err.toString(), data: null};
  202. }
  203. }
  204. /**
  205. * 中间计量 (Get)
  206. * @param ctx
  207. * @returns {Promise<void>}
  208. */
  209. async detail(ctx) {
  210. try {
  211. await this._getStage(ctx);
  212. const renderData = this._getDefaultRenderData(ctx);
  213. renderData.jsFiles = this.app.jsFiles.common.concat(this.app.jsFiles.stage.detail);
  214. renderData.imType = tenderConst.imType;
  215. await this.layout('stage/detail.ejs', renderData, 'stage/detail_modal.ejs');
  216. } catch (err) {
  217. this.log(err);
  218. ctx.redirect('/tender/' + ctx.tender.id + '/measure/stage/' + ctx.params.order);
  219. }
  220. }
  221. /**
  222. * 设置中间计量生成规则,并生成中间计量数据 (Ajax)
  223. * @param ctx
  224. * @returns {Promise<void>}
  225. */
  226. async buildDetailData (ctx) {
  227. try {
  228. await this._getStage(ctx);
  229. // 检查登录用户,是否可操作
  230. if (ctx.session.sessionUser.accountId === ctx.stage.user_id) {
  231. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  232. throw '该计量期当前您无权操作';
  233. }
  234. }
  235. const data = JSON.parse(ctx.request.body.data);
  236. await ctx.service.stage.buildDetailData(ctx.tender.id, ctx.stage.order, data);
  237. ctx.body = {err: 0, msg: '', data: null};
  238. } catch (err) {
  239. this.log(err);
  240. ctx.body = {err: 1, msg: err.toString(), data: null};
  241. }
  242. }
  243. /**
  244. * 加载数据 (Ajax)
  245. * @param ctx
  246. * @returns {Promise<void>}
  247. */
  248. async loadDetailRelaData(ctx) {
  249. try {
  250. await this._getStage(ctx);
  251. const data = JSON.parse(ctx.request.body.data);
  252. // 加载台账数据
  253. if (data.loadType === 'ledger') {
  254. const ledgerData = await ctx.service.ledger.getDataByTenderId(ctx.tender.id, -1);
  255. ctx.body = {err: 0, msg: '', data: ledgerData };
  256. } else if (data.loadType === 'all') {
  257. const ledger = await ctx.service.ledger.getDataByTenderId(ctx.tender.id, -1);
  258. const curStage = await ctx.service.stageBills.getLastestStageData(ctx.tender.id, ctx.stage.id);
  259. const stageDetail = await ctx.service.stageDetail.getLastestStageData(ctx.tender.id, ctx.stage.id);
  260. ctx.body = {
  261. err: 0,
  262. msg: '',
  263. data: {
  264. ledger,
  265. curStage,
  266. stageDetail,
  267. }
  268. };
  269. }
  270. } catch (err) {
  271. this.log(err);
  272. ctx.body = {err: 1, msg: err.toString(), data: null};
  273. }
  274. }
  275. /**
  276. * 中间计量,生成规则,高级设置 (Ajax)
  277. * @param ctx
  278. * @returns {Promise<void>}
  279. */
  280. async setAdvancedConfig(ctx) {
  281. try {
  282. await this._getStage(ctx);
  283. // 检查登录用户,是否可操作
  284. if (ctx.session.sessionUser.accountId === ctx.stage.user_id) {
  285. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  286. throw '该计量期当前您无权操作';
  287. }
  288. }
  289. const data = JSON.parse(ctx.request.body.data);
  290. await ctx.service.stage.update(data, { id: ctx.stage.id });
  291. ctx.body = {err: 0, msg: '', data: this.ctx.stage};
  292. } catch (err) {
  293. this.log(err);
  294. ctx.body = {err: 1, msg: err.toString(), data: null};
  295. }
  296. }
  297. /**
  298. * 中间计量,编辑中间计量数据(Ajax)
  299. * @param ctx
  300. * @returns {Promise<void>}
  301. */
  302. async saveDetailData(ctx) {
  303. try {
  304. await this._getStage(ctx);
  305. // 检查登录用户,是否可操作
  306. if (ctx.session.sessionUser.accountId === ctx.stage.user_id) {
  307. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  308. throw '该计量期当前您无权操作';
  309. }
  310. } else {
  311. // 检查是否可修改
  312. //const curAuditor = await ctx.service.stageAudit.get
  313. }
  314. const data = JSON.parse(ctx.request.body.data);
  315. const responseData = { err: 0, msg: '', data: {}, };
  316. responseData.data = await ctx.service.stageDetail.saveDetailData(data);
  317. ctx.body = responseData;
  318. } catch (err) {
  319. this.log(err);
  320. ctx.body = {err: 1, msg: err.toString(), data: null};
  321. }
  322. }
  323. /**
  324. * 中间计量,添加草图,上传图片 (Ajax)
  325. * @param ctx
  326. * @returns {Promise<void>}
  327. */
  328. async addCalcImage(ctx) {
  329. try {
  330. const stream = await ctx.getFileStream();
  331. const create_time = Date.parse(new Date()) / 1000;
  332. const fileInfo = path.parse(stream.filename);
  333. const fileName = path.join('public/upload', this.ctx.tender.id.toString(), 'im', 'calcImg_' + create_time + fileInfo.ext);
  334. await ctx.helper.saveStreamFile(stream, path.join(this.app.baseDir, 'app', fileName));
  335. ctx.body = {err: 0, msg: '', data: fileName};
  336. } catch(err) {
  337. this.log(err);
  338. ctx.body = {err: 1, msg: err.toString(), data: null};
  339. }
  340. }
  341. /**
  342. * 中间计量,设置草图 (Ajax)
  343. * @param ctx
  344. * @returns {Promise<void>}
  345. */
  346. async mergeCalcImage(ctx) {
  347. try {
  348. await this._getStage(ctx);
  349. // 检查登录用户,是否可操作
  350. if (ctx.session.sessionUser.accountId === ctx.stage.user_id) {
  351. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  352. throw '该计量期当前您无权操作';
  353. }
  354. } else {
  355. // 检查是否可修改
  356. //const curAuditor = await ctx.service.stageAudit.get
  357. }
  358. const data = JSON.parse(ctx.request.body.data);
  359. if (data.updateType === 'update') {
  360. const create_time = Date.parse(new Date()) / 1000;
  361. const fileName = path.join('public/upload', this.ctx.tender.id.toString(), 'im', 'calcImg_' + create_time + '.jpg');
  362. const base64Data = data.img.replace(/^data:image\/\w+;base64,/, "");
  363. const dataBuffer = new Buffer(base64Data, 'base64');
  364. await this.ctx.helper.saveBufferFile(dataBuffer, path.join(this.app.baseDir, 'app', fileName));
  365. data.calc_img = fileName;
  366. data.calc_img_org = JSON.stringify(data.imgInfo);
  367. delete data.updateType;
  368. delete data.img;
  369. delete data.imgInfo;
  370. } else if (data.updateType === 'clear') {
  371. data.calc_img = null;
  372. data.calc_img_org = null;
  373. delete data.updateType;
  374. }
  375. await this.ctx.service.stageDetail.saveDetailData(data);
  376. const imData = await ctx.service.stageDetail.getLastestImStageData(this.ctx.tender.id, this.ctx.stage.id, data.lid, data.uuid);
  377. const responseData = {err: 0, msg: '', data: imData};
  378. ctx.body = responseData;
  379. } catch (err) {
  380. this.log(err);
  381. ctx.body = {err: 1, msg: err.toString(), data: null};
  382. }
  383. }
  384. /**
  385. * 合同支付 (Get)
  386. * @param ctx
  387. * @returns {Promise<void>}
  388. */
  389. async pay(ctx) {
  390. try {
  391. await this._getStage(ctx);
  392. const renderData = this._getDefaultRenderData(ctx);
  393. const dealPay = await ctx.service.stagePay.getStagePays(ctx.stage);
  394. if (dealPay && dealPay.length > 0) {
  395. renderData.dealPay = dealPay;
  396. } else {
  397. await ctx.service.pay.addDefaultPayData(this.ctx.tender.id);
  398. await ctx.service.stagePay.addInitialStageData(ctx.stage);
  399. renderData.dealPay = await ctx.service.stagePay.getStagePays(ctx.stage);
  400. }
  401. renderData.calcBase = await ctx.service.stage.getStagePayCalcBase();
  402. renderData.jsFiles = this.app.jsFiles.common.concat(this.app.jsFiles.stage.pay);
  403. // 计算 本期金额
  404. const payCalculator = new PayCalculator(this.ctx, this.ctx.tender.info.decimal);
  405. await payCalculator.calculateAll(renderData.dealPay);
  406. await this.layout('stage/pay.ejs', renderData, 'stage/pay_modal.ejs');
  407. } catch (err) {
  408. this.log(err);
  409. ctx.redirect('/tender/' + ctx.tender.id + '/measure/stage');
  410. }
  411. }
  412. /**
  413. * 合同支付 - 编辑合同支付项 (Ajax)
  414. * @param ctx
  415. * @returns {Promise<void>}
  416. */
  417. async savePayData(ctx) {
  418. try {
  419. await this._getStage(ctx);
  420. // 检查登录用户,是否可操作
  421. if (ctx.session.sessionUser.accountId === ctx.stage.user_id) {
  422. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  423. throw '该计量期当前您无权操作';
  424. }
  425. } else {
  426. if (!ctx.stage.curAuditor || ctx.stage.curAuditor.id !== ctx.session.sessionUser.accountId) {
  427. throw '该计量期当前您无权操作';
  428. }
  429. }
  430. const data = JSON.parse(ctx.request.body.data);
  431. const responseData = { err: 0, msg: '', data: {}, };
  432. const payCalculator = new PayCalculator(this.ctx, this.ctx.tender.info.decimal);
  433. switch (data.type) {
  434. case 'add':
  435. responseData.data = await ctx.service.pay.add();
  436. responseData.data = await ctx.service.stagePay.getStagePay(ctx.stage, responseData.data.pid);
  437. break;
  438. case 'del':
  439. await ctx.service.pay.del(data.id);
  440. responseData.data = await ctx.service.stagePay.getStagePays(ctx.stage);
  441. await payCalculator.calculate(responseData.data);
  442. break;
  443. case 'changeOrder':
  444. responseData.data = await ctx.service.pay.changeOrder(data.id1, data.id2);
  445. break;
  446. case 'info':
  447. responseData.data = await ctx.service.pay.save(data.updateData);
  448. if (data.updateData.sexpr !== undefined || data.updateData.sprice !== undefined
  449. || data.updateData.rexpr !== undefined || data.updateData.rprice !== undefined
  450. || data.updateData.minus !== undefined || data.updateData.is_yf !== undefined
  451. || data.updateData.dl_type !== undefined) {
  452. responseData.data = await ctx.service.stagePay.getStagePays(ctx.stage);
  453. await payCalculator.calculateAll(responseData.data);
  454. //console.log(responseData.data[responseData.data.length - 1]);
  455. } else {
  456. responseData.data = await ctx.service.stagePay.getStagePay(ctx.stage, responseData.data.id);
  457. }
  458. break;
  459. case 'stage':
  460. await ctx.service.stagePay.save(data.updateData);
  461. responseData.data = await ctx.service.stagePay.getStagePays(ctx.stage);
  462. await payCalculator.calculateAll(responseData.data);
  463. break;
  464. }
  465. ctx.body = responseData;
  466. } catch(err) {
  467. this.log(err);
  468. ctx.body = {err: 1, msg: err.toString(), data: null};
  469. }
  470. }
  471. /**
  472. * 合同支付 - 章节明细 (Ajax-Post)
  473. * @param ctx
  474. * @returns {Promise<void>}
  475. */
  476. async chapterDetail(ctx) {
  477. const _ = this.app._;
  478. const fields = ['contract_tp', 'qc_tp', 'gather_tp', 'end_gather_tp'];
  479. function assignStageData(chapter, curStage, preStage) {
  480. chapter.contract_tp = curStage.contract_tp;
  481. chapter.qc_tp = curStage.qc_tp;
  482. chapter.gather_tp = _.add(curStage.contract_tp, curStage.qc_tp);
  483. if (preStage) {
  484. chapter.pre_contract_tp = preStage.contract_tp;
  485. chapter.pre_qc_tp = preStage.qc_tp;
  486. chapter.pre_gather_tp = _.add(preStage.contract_tp, preStage.qc_tp);
  487. chapter.end_contract_tp = _.add(curStage.contract_tp, preStage.contract_tp);
  488. chapter.end_qc_tp = _.add(curStage.qc_tp, preStage.qc_tp);
  489. chapter.end_gather_tp = _.add(chapter.gather_tp, chapter.pre_gather_tp);
  490. } else {
  491. chapter.end_contract_tp = curStage.contract_tp;
  492. chapter.end_qc_tp = curStage.qc_tp;
  493. chapter.end_gather_tp = chapter.gather_tp;
  494. }
  495. }
  496. try {
  497. await this._getStage(ctx);
  498. const chapterDetail = JSON.parse(JSON.stringify(payConst.chapterDetail));
  499. const calcDetail = _.sortBy(chapterDetail, ['cType']);
  500. for (const cd of calcDetail) {
  501. switch (cd.cType) {
  502. case 1:
  503. const tp = await ctx.service.stageBills.getSumTotalPriceGcl(ctx.stage, cd.filter);
  504. assignStageData(cd, tp);
  505. break;
  506. case 11:
  507. assignStageData(cd, await ctx.service.stageBills.getSumTotalPriceGcl(ctx.stage));
  508. break;
  509. case 21:
  510. const sum = _.find(calcDetail, {cType: 11});
  511. const chapters = _.filter(calcDetail, {cType: 1});
  512. for (const f of fields) {
  513. cd[f] = _.subtract(sum[f], _.sumBy(chapters, f));
  514. }
  515. break;
  516. case 31:
  517. assignStageData(cd, await ctx.service.stageBills.getSumTotalPriceNotGcl(ctx.stage));
  518. break;
  519. case 41:
  520. const sum1 = _.find(calcDetail, {cType: 11});
  521. const sum2 = _.find(calcDetail, {cType: 31});
  522. for (const f of fields) {
  523. cd[f] = _.add(sum1.value, sum2.value);
  524. }
  525. break;
  526. }
  527. }
  528. ctx.body = {err: 0, msg: '', data: {detail: chapterDetail, decimal: ctx.tender.info.decimal}};
  529. } catch (err) {
  530. this.log(err);
  531. ctx.body = {err: 1, msg: err.toString(), data: null};
  532. }
  533. }
  534. /**
  535. * 变更令 (Get)
  536. * @param ctx
  537. * @returns {Promise<void>}
  538. */
  539. async change(ctx) {
  540. try {
  541. await this._getStage(ctx);
  542. const renderData = this._getDefaultRenderData(ctx);
  543. await this.layout('stage/change.ejs', renderData);
  544. } catch (err) {
  545. this.log(err);
  546. ctx.redirect('/tender/' + ctx.tender.id + '/measure/stage');
  547. }
  548. }
  549. /**
  550. * 查询可用变更令 (Ajax-Post)
  551. * @param ctx
  552. * @returns {Promise<void>}
  553. */
  554. async searchValidChange(ctx) {
  555. try {
  556. const data = JSON.parse(ctx.request.body.data);
  557. if (!data.bills && !data.pos) {
  558. throw '数据错误';
  559. }
  560. const bills = data.bills ? data.bills : await ctx.service.ledger.getDataById(data.pos.lid);
  561. const pos = data.pos;
  562. const changes = await ctx.service.change.getValidChanges(bills, pos);
  563. ctx.body = {err: 0, msg: '', data: changes};
  564. } catch(err) {
  565. this.log(err);
  566. ctx.body = {err: 1, msg: err.toString(), data: null};
  567. }
  568. }
  569. // 审批相关
  570. /**
  571. * 添加审批人
  572. * @param ctx
  573. * @returns {Promise<void>}
  574. */
  575. async addAudit(ctx) {
  576. try {
  577. await this._getStage(ctx);
  578. const data = JSON.parse(ctx.request.body.data);
  579. const id = this.app._.toInteger(data.auditorId);
  580. if (isNaN(id) || id <= 0) {
  581. throw '参数错误';
  582. }
  583. // 检查权限等
  584. if (ctx.stage.user_id !== ctx.session.sessionUser.accountId) {
  585. throw '您无权添加审核人';
  586. }
  587. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  588. throw '当前不允许添加审核人';
  589. }
  590. // 检查审核人是否已存在
  591. const exist = this.app._.find(ctx.stage.auditors, {aid: id});
  592. if (exist) {
  593. throw '该审核人已存在,请勿重复添加';
  594. }
  595. const result = await ctx.service.stageAudit.addAuditor(ctx.stage.id, id);
  596. if (!result) {
  597. throw '添加审核人失败';
  598. }
  599. const audit = await ctx.service.stageAudit.getAuditor(ctx.stage.id, id);
  600. ctx.body = {err: 0, msg: '', data: audit};
  601. } catch (err) {
  602. this.log(err);
  603. ctx.body = {err: 1, msg: err.toString(), data: null};
  604. }
  605. }
  606. /**
  607. * 移除审批人
  608. * @param ctx
  609. * @returns {Promise<void>}
  610. */
  611. async deleteAudit(ctx) {
  612. try {
  613. await this._getStage(ctx);
  614. const data = JSON.parse(ctx.request.body.data);
  615. const id = data.auditorId instanceof Number ? data.auditorId : this.app._.toNumber(data.auditorId);
  616. if (isNaN(id) || id <= 0) {
  617. throw '参数错误';
  618. }
  619. const result = await ctx.service.stageAudit.deleteAuditor(ctx.stage.id, id, ctx.stage.times);
  620. if (!result) {
  621. throw '移除审核人失败';
  622. }
  623. const auditors = await ctx.service.stageAudit.getAuditors(ctx.stage.id);
  624. ctx.body = {err: 0, msg: '', data: auditors};
  625. } catch (err) {
  626. ctx.body = {err: 1, msg: err.toString(), data: null};
  627. }
  628. }
  629. /**
  630. * 上报
  631. * @param ctx
  632. * @returns {Promise<void>}
  633. */
  634. async startAudit(ctx) {
  635. try {
  636. await this._getStage(ctx);
  637. // 检查权限等
  638. if (!ctx.stage) {
  639. throw '数据错误';
  640. }
  641. if (ctx.stage.user_id !== ctx.session.sessionUser.accountId) {
  642. throw '您无权上报该期数据';
  643. }
  644. if (ctx.stage.status === auditConst.status.checking || ctx.stage.status === auditConst.status.checked) {
  645. throw '该期数据当前无法上报';
  646. }
  647. await ctx.service.stageAudit.start(ctx.stage.id, ctx.stage.times);
  648. ctx.redirect(ctx.request.header.referer);
  649. } catch (err) {
  650. this.log(err);
  651. ctx.session.postError = err.toString();
  652. ctx.redirect(ctx.request.header.referer);
  653. }
  654. }
  655. /**
  656. * 审批
  657. * @param ctx
  658. * @returns {Promise<void>}
  659. */
  660. async checkAudit(ctx) {
  661. try {
  662. await this._getStage(ctx);
  663. if (!this.ctx.stage || this.ctx.stage.status !== auditConst.status.checking) {
  664. throw '当前期数据有误';
  665. }
  666. if (!this.ctx.stage.curAuditor || this.ctx.stage.curAuditor.aid !== ctx.session.sessionUser.accountId) {
  667. throw '您无权进行该操作';
  668. }
  669. const data = {
  670. checkType: parseInt(ctx.request.body.checkType),
  671. opinion: ctx.request.body.opinion,
  672. };
  673. if (!data.checkType || isNaN(data.checkType)) {
  674. throw '提交数据错误';
  675. }
  676. if (data.checkType === auditConst.status.checkNo) {
  677. if (!data.checkType || isNaN(data.checkType)) {
  678. throw '提交数据错误';
  679. }
  680. }
  681. await ctx.service.stageAudit.check(ctx.stage.id, data, ctx.stage.times);
  682. console.log('success');
  683. ctx.redirect(ctx.request.header.referer);
  684. } catch (err) {
  685. console.log(err);
  686. this.log(err);
  687. ctx.session.postError = err.toString();
  688. ctx.redirect(ctx.request.header.referer);
  689. }
  690. }
  691. _getGatherSpreadSetting() {
  692. const _ = this.app._;
  693. function removeFieldCols(setting, cols) {
  694. _.remove(setting.cols, function (c) {
  695. return cols.indexOf(c.field) > -1;
  696. });
  697. }
  698. function setColFormat(cols, field, formatter) {
  699. const filterCols = cols.filter(function (c) {
  700. return c.field.search(field) !== -1;
  701. });
  702. for (const col of filterCols) {
  703. col.formatter = formatter;
  704. }
  705. }
  706. const tpFormatter = this.ctx.helper.getNumberFormatter(this.ctx.tender.info.decimal.tp);
  707. const upFormatter = this.ctx.helper.getNumberFormatter(2);
  708. const gcl = JSON.parse(JSON.stringify(spreadConst.stageGather.gcl));
  709. setColFormat(gcl.cols, 'unit_price', upFormatter);
  710. setColFormat(gcl.cols, 'total_price', tpFormatter);
  711. setColFormat(gcl.cols, 'tp', tpFormatter);
  712. const leafXmj = JSON.parse(JSON.stringify(spreadConst.stageGather.leafXmj));
  713. const tender = this.ctx.tender;
  714. if (tender.data.measure_type === measureType.tz.value) {
  715. removeFieldCols(gcl, spreadConst.filterCols.tzWithoutCols);
  716. }
  717. return [gcl, leafXmj];
  718. }
  719. /**
  720. * 清单汇总
  721. * @param ctx
  722. * @returns {Promise<void>}
  723. */
  724. async gather(ctx) {
  725. try {
  726. await this._getStage(ctx);
  727. const renderData = this._getDefaultRenderData(ctx);
  728. [renderData.gclSpread, renderData.leafXmjSpread] = this._getGatherSpreadSetting();
  729. renderData.ledger = await ctx.service.ledger.getDataByTenderId(ctx.tender.id, -1);
  730. renderData.curLedgerData = await ctx.service.stageBills.getAuditorStageData(ctx.tender.id, ctx.stage.id, ctx.stage.times, 0);
  731. renderData.pos = await ctx.service.pos.getPosData({tid: ctx.tender.id});
  732. // todo 根据当前人,或指定对象查询数据
  733. renderData.curPosData = await ctx.service.stagePos.getAuditorStageData(ctx.tender.id,
  734. ctx.stage.id, ctx.stage.times, ctx.stage.curAuditor ? ctx.stage.curAuditor.order : 0);
  735. //renderData.gcl = await this.ctx.service.ledger.getAllLeafGclBills(this.ctx.tender.id);
  736. renderData.jsFiles = this.app.jsFiles.common.concat(this.app.jsFiles.stage.gather);
  737. await this.layout('stage/gather.ejs', renderData);
  738. } catch (err) {
  739. this.log(err);
  740. ctx.redirect('/tender/' + ctx.tender.id + '/measure/stage/' + ctx.params.order);
  741. }
  742. }
  743. /**
  744. * 审核比较
  745. * @param ctx
  746. * @returns {Promise<void>}
  747. */
  748. async compare(ctx) {
  749. function getCompareSpreadSetting () {
  750. function setColFormat(cols, field, formatter) {
  751. const filterCols = cols.filter(function (c) {
  752. return c.field.search(field) !== -1;
  753. });
  754. for (const col of filterCols) {
  755. col.formatter = formatter;
  756. }
  757. }
  758. const tpFormatter = ctx.helper.getNumberFormatter(ctx.tender.info.decimal.tp);
  759. const upFormatter = ctx.helper.getNumberFormatter(2);
  760. const ledger = JSON.parse(JSON.stringify(spreadConst.stageCompare.ledger));
  761. setColFormat(ledger.baseCols, 'unit_price', upFormatter);
  762. setColFormat(ledger.extraCols, 'total_price', tpFormatter);
  763. setColFormat(ledger.extraCols, 'tp', tpFormatter);
  764. const pos = JSON.parse(JSON.stringify(spreadConst.stageCompare.pos));
  765. return [ledger, pos];
  766. };
  767. try {
  768. await this._getStage(ctx);
  769. const renderData = this._getDefaultRenderData(ctx);
  770. [renderData.ledgerSpread, renderData.posSpread] = getCompareSpreadSetting();
  771. renderData.ledger = await ctx.service.ledger.getDataByTenderId(ctx.tender.id, -1);
  772. renderData.orgStageLedger = await ctx.service.stageBills.getAuditorStageData(ctx.tender.id, ctx.stage.id, ctx.stage.times, 0);
  773. renderData.pos = await ctx.service.pos.getPosData({tid: ctx.tender.id});
  774. renderData.orgStagePos = await ctx.service.stagePos.getAuditorStageData(ctx.tender.id,
  775. ctx.stage.id, ctx.stage.times, ctx.stage.curAuditor ? ctx.stage.curAuditor.order : 0);
  776. renderData.jsFiles = this.app.jsFiles.common.concat(this.app.jsFiles.stage.compare);
  777. await this.layout('stage/compare.ejs', renderData, 'stage/compare_modal.ejs');
  778. } catch (err) {
  779. this.log(err);
  780. ctx.redirect('/tender/' + ctx.tender.id + '/measure/stage/' + ctx.params.order);
  781. }
  782. }
  783. async compareAuditor(ctx) {
  784. try {
  785. await this._getStage(ctx);
  786. const data = JSON.parse(ctx.request.body.data);
  787. const result = [];
  788. for (const order of data.auditors) {
  789. const data = { order: order, bills: [], pos: [] };
  790. data.bills = await ctx.service.stageBills.getAuditorStageData(ctx.tender.id, ctx.stage.id, ctx.stage.times, order);
  791. data.pos = await ctx.service.stagePos.getAuditorStageData(ctx.tender.id, ctx.stage.id, ctx.stage.times, order);
  792. result.push(data);
  793. }
  794. ctx.body = {err: 0, msg: '', data: result};
  795. } catch (err) {
  796. this.log(err);
  797. ctx.body = {err: 1, msg: err.toString(), data: null};
  798. }
  799. }
  800. /**
  801. * 报表
  802. * @param ctx
  803. * @returns {Promise<void>}
  804. */
  805. async report(ctx) {
  806. try {
  807. await this._getStage(ctx);
  808. const renderData = this._getDefaultRenderData(ctx);
  809. await this.layout('stage/report.ejs', renderData, 'stage/report_modal.ejs');
  810. } catch (err) {
  811. this.log(err);
  812. ctx.redirect('/tender/' + ctx.tender.id + '/measure/stage');
  813. }
  814. }
  815. }
  816. return StageController;
  817. };