revise_controller.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date
  7. * @version
  8. */
  9. const stdDataAddType = {
  10. withParent: 1,
  11. child: 2,
  12. next: 3,
  13. };
  14. const audit = require('../const/audit');
  15. const tenderMenu = require('../../config/menu').tenderMenu;
  16. const measureType = require('../const/tender').measureType;
  17. const spreadConst = require('../const/spread');
  18. const fs = require('fs');
  19. const LzString = require('lz-string');
  20. const accountGroup = require('../const/account_group').group;
  21. module.exports = app => {
  22. class ReviseController extends app.BaseController {
  23. /**
  24. * 构造函数
  25. *
  26. * @param {Object} ctx - egg全局变量
  27. * @return {void}
  28. */
  29. constructor(ctx) {
  30. super(ctx);
  31. }
  32. /**
  33. * 是否可以新增修订
  34. * @param ctx
  35. * @returns {Promise<boolean>}
  36. * @private
  37. */
  38. async _getAddReviseValid(ctx) {
  39. const stage = await ctx.service.stage.getLastestStage(ctx.tender.id, true);
  40. const revise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);
  41. return (ctx.tender.data.user_id === ctx.session.sessionUser.accountId) &&
  42. (ctx.tender.data.ledger_status === audit.ledger.status.checked) &&
  43. (!stage || stage.status === audit.stage.status.checked) &&
  44. (!revise || !revise.valid || revise.status === audit.revise.status.checked);
  45. }
  46. /**
  47. * 台账修订页面 (Get)
  48. *
  49. * @param {object} ctx - egg全局变量
  50. * @return {void}
  51. */
  52. async index(ctx) {
  53. try {
  54. // 分页相关
  55. const count = await ctx.service.ledgerRevise.count({tid: ctx.tender.id});
  56. const ledgerRevise = await ctx.service.ledgerRevise.getReviseList(ctx.tender.id);
  57. const addValid = await this._getAddReviseValid(ctx);
  58. const [stdBills, stdChapters] = await this.ctx.service.valuation.getValuationStdList(ctx.tender.data.valuation);
  59. const renderData = {
  60. tender: ctx.tender.data,
  61. tenderMenu: this.menu.tenderMenu,
  62. preUrl: '/tender/' + ctx.tender.id,
  63. pageInfo: {
  64. page: ctx.page,
  65. total: Math.ceil(count/app.config.pageSize),
  66. queryData: JSON.stringify(ctx.urlInfo.query),
  67. },
  68. ledgerRevise,
  69. addValid,
  70. auditConst: audit.revise,
  71. stdBills,
  72. stdChapters,
  73. };
  74. await this.layout('revise/index.ejs', renderData, 'revise/modal.ejs');
  75. } catch (err) {
  76. this.log(err);
  77. ctx.redirect(ctx.request.header.referer);
  78. }
  79. }
  80. /**
  81. * 新增 修订 (Post)
  82. * @param ctx
  83. * @returns {Promise<void>}
  84. */
  85. async add(ctx) {
  86. try {
  87. const addValid = await this._getAddReviseValid(ctx);
  88. if (!addValid) {
  89. throw '无法新增修订';
  90. }
  91. const revise = await ctx.service.ledgerRevise.add(ctx.tender.id, ctx.session.sessionUser.accountId);
  92. ctx.redirect('/tender/' + ctx.tender.id + '/revise/info');
  93. } catch(err) {
  94. this.log(err);
  95. ctx.redirect(ctx.request.header.referer);
  96. }
  97. }
  98. /**
  99. * 作废 修订 (Post)
  100. * @param ctx
  101. * @returns {Promise<void>}
  102. */
  103. async cancel(ctx) {
  104. try {
  105. const revise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);
  106. if (revise.uid !== ctx.session.sessionUser.accountId) {
  107. throw '您无权作废该修订';
  108. }
  109. if (revise.status === audit.revise.checking) {
  110. throw '修订审批中,不可作废';
  111. } else if (revise.status === audit.revise.checked) {
  112. throw '修订已审批通过,不可作废';
  113. }
  114. if (revise.valid) {
  115. const result = await ctx.service.ledgerRevise.cancelRevise(revise.id);
  116. }
  117. ctx.redirect('/tender/' + ctx.tender.id + '/revise/');
  118. } catch(err) {
  119. this.log(err);
  120. ctx.redirect(ctx.request.header.referer);
  121. }
  122. }
  123. /**
  124. * 保存 修订详情 (Ajax-post)
  125. * @param ctx
  126. * @returns {Promise<void>}
  127. */
  128. async save(ctx) {
  129. try {
  130. const revise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);
  131. if (!revise.valid) {
  132. throw '该修订已作废';
  133. } else if (revise.status === audit.revise.status.uncheck || revise.status === audit.revise.status.checkNo) {
  134. if (ctx.session.sessionUser.accountId !== revise.uid) {
  135. throw '您无权修改';
  136. }
  137. } else if (revise.status === audit.revise.status.checking) {
  138. throw '修订在审批中,不可修改修订数据';
  139. } else if (revise.staut === audit.revise.status.checked) {
  140. throw '修订已经审批通过,不可修改修订数据,请新增下一修订';
  141. }
  142. const data = JSON.parse(ctx.request.body.data);
  143. if (data.content === undefined) {
  144. throw '保存数据有误'
  145. }
  146. const result = await ctx.service.ledgerRevise.defaultUpdate({id: revise.id, content: data.content});
  147. if (result.affectedRows !== 1) {
  148. throw '保存数据失败,请重试';
  149. }
  150. ctx.body = {err: 0, msg: '', data: null};
  151. } catch(err) {
  152. this.log(err);
  153. ctx.body = this.ajaxErrorBody(err, '保存数据失败,请重试');
  154. }
  155. }
  156. /**
  157. * 获取SpreadSetting
  158. * @private
  159. */
  160. _getSpreadSetting(revise) {
  161. const _ = this.app._;
  162. function removeFieldCols(setting, cols) {
  163. _.remove(setting.cols, function(c) {
  164. return cols.indexOf(c.field) > -1;
  165. });
  166. }
  167. const tender = this.ctx.tender;
  168. const setting = tender.info.display.ledger.clQty ? spreadConst.withCl : spreadConst.withoutCl;
  169. const ledger = JSON.parse(JSON.stringify(setting.ledger));
  170. const pos = JSON.parse(JSON.stringify(setting.pos));
  171. if (revise.status === audit.revise.status.checking || revise.status === audit.revise.status.checked) {
  172. ledger.readOnly = true;
  173. pos.readOnly = true;
  174. }
  175. if (tender.data.measure_type === measureType.tz.value) {
  176. removeFieldCols(ledger, spreadConst.filterCols.tzWithoutCols);
  177. }
  178. if (!tender.info.display.ledger.dgnQty) {
  179. removeFieldCols(ledger, spreadConst.filterCols.dgnCols);
  180. }
  181. return [ledger, pos];
  182. }
  183. async _getDefaultReviseInfoData(ctx, revise) {
  184. const reviseBills = await ctx.service.reviseBills.getData(ctx.tender.id);
  185. const revisePos = await ctx.service.revisePos.getData(ctx.tender.id, revise.id);
  186. const [ledgerSpread, posSpread] = this._getSpreadSetting(revise);
  187. const [stdBills, stdChapters] = await this.ctx.service.valuation.getValuationStdList(ctx.tender.data.valuation);
  188. return {
  189. revise: revise, tender: ctx.tender.data,
  190. reviseBills, revisePos, ledgerSpread, posSpread, tenderMenu, measureType,
  191. preUrl: '/tender/' + ctx.tender.id,
  192. audit: audit.revise,
  193. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.ledger.revise),
  194. stdBills,
  195. stdChapters,
  196. };
  197. }
  198. /**
  199. * 历史修订页面(修订完成后,用于查看)
  200. * @param ctx
  201. * @param revise
  202. * @returns {Promise<void>}
  203. * @private
  204. */
  205. async _getHistoryReviseInfo(ctx, revise) {
  206. const renderData = await this._getDefaultReviseInfoData(ctx, revise);
  207. renderData.ledgerSpread.readOnly = true;
  208. renderData.posSpread.readOnly = true;
  209. renderData.readOnly = true;
  210. renderData.history = true;
  211. renderData.historyRevise = await ctx.service.ledgerRevise.getReviseList(ctx.tender.id);
  212. await this.layout('revise/info.ejs', renderData, 'revise/info_modal.ejs');
  213. }
  214. /**
  215. * 修订页面(草稿,审批中)
  216. * @param ctx
  217. * @param revise
  218. * @returns {Promise<void>}
  219. * @private
  220. */
  221. async _getReviseInfo(ctx, revise) {
  222. const renderData = await this._getDefaultReviseInfoData(ctx, revise);
  223. const readOnly = (revise.status === audit.revise.status.checking || revise.status === audit.revise.status.checked || revise.uid !== ctx.session.sessionUser.accountId);
  224. if (readOnly) {
  225. renderData.ledgerSpread.readOnly = true;
  226. renderData.posSpread.readOnly = true;
  227. }
  228. renderData.readOnly = readOnly;
  229. renderData.history = false;
  230. renderData.historyRevise = [];
  231. await this.layout('revise/info.ejs', renderData, 'revise/info_modal.ejs');
  232. }
  233. /**
  234. * 修订 详细页面(Get)
  235. * @param ctx
  236. * @returns {Promise<void>}
  237. */
  238. async info(ctx) {
  239. try {
  240. const revise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);
  241. if (revise.status === audit.revise.status.checked) {
  242. await this._getHistoryReviseInfo(ctx);
  243. } else {
  244. await this._getReviseInfo(ctx, revise);
  245. }
  246. } catch (err) {
  247. this.log(err);
  248. ctx.redirect(ctx.request.header.referer);
  249. }
  250. }
  251. async _checkRevise() {
  252. const revise = await this.ctx.service.ledgerRevise.getLastestRevise(this.ctx.tender.id);
  253. if (revise.uid !== this.ctx.session.sessionUser.accountId) {
  254. throw '修订数据错误';
  255. } else if (revise.status === audit.revise.status.checking) {
  256. throw '修订审批中,不可修改';
  257. } else if (revise.status === audit.revise.status.checked) {
  258. throw '修订已完成,如需修订台账,请新增修订';
  259. }
  260. this.ctx.revise = revise;
  261. }
  262. /**
  263. * 增、删、上下移、升降级
  264. * @param ctx
  265. * @returns {Promise<void>}
  266. */
  267. async _billsBase(ctx, data) {
  268. if (isNaN(data.id) || data.id <= 0) throw '数据错误';
  269. if (type !== 'add') {
  270. if (isNaN(data.count) || data.count <= 0) data.count = 1;
  271. }
  272. switch (type) {
  273. case 'add':
  274. return await ctx.service.reviseBills.addNode(ctx.tender.id, ctx.revise.id, data.id);
  275. case 'delete':
  276. return await ctx.service.reviseBills.delete(ctx.tender.id, data.id, data.count);
  277. case 'up-move':
  278. return await ctx.service.reviseBills.upMoveNode(ctx.tender.id, data.id, data.count);
  279. case 'down-move':
  280. return await ctx.service.reviseBills.downMoveNode(ctx.tender.id, data.id, data.count);
  281. case 'up-level':
  282. return await ctx.service.reviseBills.upLevelNode(ctx.tender.id, data.id, data.count);
  283. case 'down-level':
  284. return await ctx.service.reviseBills.downLevelNode(ctx.tender.id, data.id, data.count);
  285. }
  286. }
  287. /**
  288. * 批量插入数据 (Ajax)
  289. *
  290. * data = {id, batchData, batchType}
  291. * data.batchType = 'batchInsertChild'/'batchInsertNext'
  292. * data.batchData = [{name, children}] -- 项目节列表
  293. * data.batchData.children = [{code, name, unit, unit_price, quantity}] -- 工程量清单列表
  294. *
  295. * @param ctx
  296. * @return {Promise<void>}
  297. */
  298. async _batchInsert(ctx, data) {
  299. if ((isNaN(data.id) || data.id <= 0) || !data.batchType) {
  300. throw '参数错误';
  301. }
  302. switch (data.batchType) {
  303. case 'child':
  304. return await ctx.service.reviseBills.batchInsertChild(ctx.tender.id, ctx.revise.id, data.id, data.batchData);
  305. case 'next':
  306. return await ctx.service.reviseBills.batchInsertNext(ctx.tender.id, ctx.revise.id, data.id, data.batchData);
  307. default:
  308. throw '参数错误';
  309. }
  310. }
  311. async _pasteBlock(ctx, data) {
  312. if ((isNaN(data.id) || data.id <= 0) ||
  313. (!data.tid && data.tid <= 0) ||
  314. (!data.block || data.block.length <= 0)) throw '参数错误';
  315. return await ctx.service.revise.pasteBlock(data.tid, data.id, data.block);
  316. }
  317. async _addStd(ctx, data) {
  318. if ((isNaN(data.id) || data.id <= 0) || !data.stdType || !data.stdNode) throw '参数错误';
  319. // todo 校验项目是否使用该库的权限
  320. let stdLib, addType;
  321. switch (data.stdType) {
  322. case 'xmj':
  323. stdLib = ctx.service.stdXmj;
  324. addType = stdDataAddType.withParent;
  325. break;
  326. case 'gcl':
  327. stdLib = ctx.service.stdGcl;
  328. const selectNode = await ctx.service.reviseBills.getDataByKid(ctx.tender.id, data.id);
  329. if (selectNode.b_code) {
  330. addType = stdDataAddType.next;
  331. } else {
  332. addType = stdDataAddType.child;
  333. }
  334. break;
  335. default:
  336. throw '未知标准库';
  337. }
  338. const stdData = await stdLib.getDataByDataId(data.stdLibId, data.stdNode);
  339. switch (addType) {
  340. case stdDataAddType.child:
  341. return await ctx.service.reviseBills.addStdNodeAsChild(ctx.tender.id, data.id, stdData, ctx.revise.id);
  342. break;
  343. case stdDataAddType.next:
  344. return await ctx.service.reviseBills.addStdNode(ctx.tender.id, data.id, stdData, ctx.revise.id);
  345. case stdDataAddType.withParent:
  346. return await ctx.service.reviseBills.addStdNodeWithParent(ctx.tender.id, stdData, stdLib, ctx.revise.id);
  347. default:
  348. throw '未知添加方式';
  349. }
  350. }
  351. async _addDeal(ctx, data) {
  352. if (!data.type || !data.dealBills) throw '数据错误';
  353. if (data.type === 'child') {
  354. return await ctx.service.reviseBills.addChild(ctx.tender.id, data.id, data.dealBills, ctx.revise.id);
  355. } else if (data.type === 'next') {
  356. return await ctx.service.reviseBills.addBillsNode(ctx.tender.id, data.id, data.dealBills, ctx.revise.id);
  357. } else {
  358. throw '数据错误';
  359. }
  360. }
  361. async update(ctx) {
  362. try {
  363. if (!ctx.tender.data) throw '标段数据错误';
  364. await this._checkRevise();
  365. const data = JSON.parse(ctx.request.body.data);
  366. if (!data.postType || !data.postData) throw '数据错误';
  367. const responseData = {err: 0, msg: '', data: {}};
  368. switch (data.postType) {
  369. case 'add':
  370. // case 'delete':
  371. case 'up-move':
  372. case 'down-move':
  373. // case 'up-level':
  374. // case 'down-level':
  375. responseData.data = await this._billsBase(ctx, data.postData);
  376. break;
  377. case 'batch-insert':
  378. responseData.data = await this._batchInsert(ctx, data.postData);
  379. break;
  380. case 'add-deal':
  381. responseData.data = await this._addDeal(ctx, data.postData);
  382. break;
  383. case 'add-std':
  384. responseData.data = await this._addStd(ctx, data.postData);
  385. break;
  386. case 'paste-block':
  387. responseData.data = await this._pasteBlock(ctx, data.postData);
  388. break;
  389. default:
  390. throw '未知操作';
  391. }
  392. ctx.body = responseData;
  393. } catch (err) {
  394. this.log(err);
  395. ctx.body = this.ajaxErrorBody(err, '数据错误');
  396. }
  397. }
  398. }
  399. return ReviseController;
  400. };