ledger_controller.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503
  1. 'use strict';
  2. /**
  3. * 台账相关控制器
  4. *
  5. * @author CaiAoLin
  6. * @date 2017/11/30
  7. * @version
  8. */
  9. const stdDataAddType = {
  10. self: 1,
  11. withParent: 2
  12. }
  13. const auditConst = require('../const/audit').flow;
  14. const spreadConst = require('../const/spread');
  15. const tenderMenu = require('../../config/menu').tenderMenu;
  16. module.exports = app => {
  17. class LedgerController extends app.BaseController {
  18. /**
  19. * 构造函数
  20. *
  21. * @param {Object} ctx - egg全局变量
  22. * @return {void}
  23. */
  24. constructor(ctx) {
  25. super(ctx);
  26. ctx.showProject = true;
  27. ctx.showTitle = true;
  28. ctx.showTender = true;
  29. }
  30. /**
  31. * 台账分解页面 (Get)
  32. *
  33. * @param {Object} ctx - egg全局变量
  34. * @return {void}
  35. */
  36. async explode(ctx) {
  37. try {
  38. const tenderId = ctx.params.id;
  39. const tender = await this.service.tender.getDataById(tenderId);
  40. if (!tender.ledger_status) {
  41. tender.ledger_status = auditConst.status.uncheck;
  42. }
  43. if (!tender.ledger_times) {
  44. tender.ledger_times = 1;
  45. }
  46. const curAuditor = await ctx.service.ledgerAudit.getCurAuditor(tenderId, tender.ledger_times);
  47. const times = tender.ledger_status === auditConst.status.checkNo ? tender.ledger_times - 1 : tender.ledger_times;
  48. const auditors = await ctx.service.ledgerAudit.getAuditors(tenderId, times);
  49. const content = auditors.length > 0 ? await ctx.service.ledgerAuditContent.getAllDataByCondition({
  50. where: {tender_id: tenderId, times: times, audit_id: auditors[0].audit_id}
  51. }) : null;
  52. const ledgerData = await ctx.service.ledger.getDataByTenderId(tenderId, 4);
  53. const renderData = {
  54. tender,
  55. auditConst,
  56. auditors,
  57. curAuditor,
  58. content,
  59. ledger: JSON.stringify(ledgerData),
  60. ledgerSpreadSetting: JSON.stringify(spreadConst.ledgerSpread),
  61. posSpreadSetting: JSON.stringify(spreadConst.ledgerPosSpread),
  62. tenderMenu,
  63. preUrl: '/tender/' + tenderId,
  64. };
  65. await this.layout('ledger/explode.ejs', renderData, 'ledger/explode_modal.ejs');
  66. } catch (err) {
  67. this.log(err);
  68. await this.layout('/dashboard');
  69. }
  70. }
  71. /**
  72. * 检查标段是否只读(审核中,审核完成)
  73. * @param {Object} tenderData
  74. * @returns {boolean}
  75. * @private
  76. */
  77. _ledgerReadOnly (tenderData) {
  78. return tenderData.ledger_status === auditConst.status.checking || tenderData.ledger_status === auditConst.status.checked;
  79. }
  80. /**
  81. * 获取子节点 (Ajax)
  82. * @param ctx
  83. * @return {Promise<void>}
  84. */
  85. async getChildren(ctx) {
  86. const responseData = {
  87. err: 0,
  88. msg: '',
  89. data: [],
  90. };
  91. try {
  92. const tenderId = ctx.params.id;
  93. if (!tenderId) {
  94. throw '当前未打开标段';
  95. }
  96. const data = JSON.parse(ctx.request.body.data);
  97. const id = data.ledger_id;
  98. if (isNaN(id) || id <= 0) {
  99. throw '参数错误';
  100. }
  101. responseData.data = await ctx.service.ledger.getChildrenByParentId(tenderId, id);
  102. } catch (err) {
  103. responseData.err = 1;
  104. responseData.msg = err;
  105. }
  106. ctx.body = responseData;
  107. }
  108. /**
  109. * 树结构基本操作(增、删、上下移、升降级)(Ajax)
  110. * @param {Object} ctx - egg全局变量
  111. * @return {Promise<void>}
  112. */
  113. async baseOperation(ctx) {
  114. const responseData = {
  115. err: 0,
  116. msg: '',
  117. data: [],
  118. };
  119. try {
  120. const tenderId = ctx.params.id;
  121. if (!tenderId) {
  122. throw '当前未打开标段';
  123. }
  124. const tenderData = await ctx.service.tender.getDataById(tenderId);
  125. if (!tenderData || tenderData.user_id !== ctx.session.sessionUser.accountId || this._ledgerReadOnly(tenderData)) {
  126. throw '标段数据错误';
  127. }
  128. const data = JSON.parse(ctx.request.body.data);
  129. if ((isNaN(data.id) || data.id <= 0) || (!data.postType)) {
  130. throw '参数错误';
  131. }
  132. switch (data.postType) {
  133. case 'add':
  134. responseData.data = await ctx.service.ledger.addNode(tenderId, data.id);
  135. break;
  136. case 'delete':
  137. responseData.data = await ctx.service.ledger.deleteNode(tenderId, data.id);
  138. break;
  139. case 'up-move':
  140. responseData.data = await ctx.service.ledger.upMoveNode(tenderId, data.id);
  141. break;
  142. case 'down-move':
  143. responseData.data = await ctx.service.ledger.downMoveNode(tenderId, data.id);
  144. break;
  145. case 'up-level':
  146. responseData.data = await ctx.service.ledger.upLevelNode(tenderId, data.id);
  147. break;
  148. case 'down-level':
  149. responseData.data = await ctx.service.ledger.downLevelNode(tenderId, data.id);
  150. break;
  151. default:
  152. throw '未知操作';
  153. }
  154. } catch (err) {
  155. responseData.err = 1;
  156. responseData.msg = err;
  157. this.log(err);
  158. }
  159. ctx.body = responseData;
  160. }
  161. /**
  162. * 提交更新数据 (Ajax)
  163. * @param ctx
  164. * @return {Promise<void>}
  165. */
  166. async updateInfo(ctx) {
  167. const responseData = {
  168. err: 0,
  169. msg: '',
  170. data: [],
  171. };
  172. try {
  173. const tenderId = ctx.params.id;
  174. if (!tenderId) {
  175. throw '当前未打开标段';
  176. }
  177. const tenderData = await ctx.service.tender.getDataById(tenderId);
  178. if (!tenderData || tenderData.user_id !== ctx.session.sessionUser.accountId || this._ledgerReadOnly(tenderData)) {
  179. throw '标段数据错误';
  180. }
  181. const data = JSON.parse(ctx.request.body.data);
  182. if (data instanceof Array) {
  183. responseData.data = await ctx.service.ledger.updateInfos(tenderId, data);
  184. } else {
  185. responseData.data = await ctx.service.ledger.updateInfo(tenderId, data);
  186. }
  187. } catch (err) {
  188. responseData.err = 1;
  189. responseData.msg = err;
  190. }
  191. ctx.body = responseData;
  192. }
  193. async update(ctx) {
  194. console.log(ctx.body);
  195. const responseData = {
  196. err: 0,
  197. msg: '',
  198. data: [],
  199. };
  200. try {
  201. const tenderId = ctx.params.id;
  202. if (!tenderId) {
  203. throw '当前未打开标段';
  204. }
  205. const tenderData = await ctx.service.tender.getDataById(tenderId);
  206. if (!tenderData || tenderData.user_id !== ctx.session.sessionUser.accountId || this._ledgerReadOnly(tenderData)) {
  207. throw '标段数据错误';
  208. }
  209. const data = JSON.parse(ctx.request.body.data);
  210. responseData.data = await ctx.service.ledger.updateCalc(tenderId, data);
  211. } catch (err) {
  212. responseData.err = 1;
  213. responseData.msg = err;
  214. }
  215. ctx.body = responseData;
  216. }
  217. /**
  218. * 复制粘贴整块 (Ajax)
  219. *
  220. * @param ctx
  221. * @return {Promise<void>}
  222. */
  223. async pasteBlock(ctx) {
  224. const responseData = {
  225. err: 0,
  226. msg: '',
  227. data: [],
  228. };
  229. try {
  230. const tenderId = ctx.params.id;
  231. if (!tenderId) {
  232. throw '当前未打开标段';
  233. }
  234. const tenderData = await ctx.service.tender.getDataById(tenderId);
  235. if (!tenderData || tenderData.user_id !== ctx.session.sessionUser.accountId || this._ledgerReadOnly(tenderData)) {
  236. throw '标段数据错误';
  237. }
  238. const data = JSON.parse(ctx.request.body.data);
  239. if ((isNaN(data.id) || data.id <= 0) || (!data.block || data.block.length <= 0)) {
  240. throw '参数错误';
  241. }
  242. responseData.data = await ctx.service.ledger.pasteBlock(tenderId, data.id, data.block);
  243. } catch (err) {
  244. responseData.err = 1;
  245. responseData.msg = err;
  246. }
  247. ctx.body = responseData;
  248. }
  249. /**
  250. * 从标准项目表添加数据 (Ajax)
  251. * @param ctx
  252. * @returns {Promise<void>}
  253. */
  254. async addFromStandardLib(ctx) {
  255. const responseData = {
  256. err: 0,
  257. msg: '',
  258. data: [],
  259. };
  260. try {
  261. const tenderId = ctx.params.id;
  262. if (!tenderId) {
  263. throw '当前未打开标段';
  264. }
  265. const tenderData = await ctx.service.tender.getDataById(tenderId);
  266. if (!tenderData || tenderData.user_id !== ctx.session.sessionUser.accountId || this._ledgerReadOnly(tenderData)) {
  267. throw '标段数据错误';
  268. }
  269. const data = JSON.parse(ctx.request.body.data);
  270. if ((isNaN(data.id) || data.id <= 0) || !data.stdType || !data.stdNode) {
  271. throw '参数错误';
  272. }
  273. //todo 校验项目是否使用该库的权限
  274. let stdLib;
  275. switch (data.stdType) {
  276. case 'chapter':
  277. stdLib = ctx.service.stdChapter;
  278. break;
  279. case 'bills':
  280. stdLib = ctx.service.stdBills;
  281. break;
  282. default:
  283. throw '未知标准库';
  284. }
  285. const stdData = await stdLib.getDataByDataId(data.stdLibId, data.stdNode);
  286. const addType = stdDataAddType.withParent;
  287. switch (addType) {
  288. case stdDataAddType.self:
  289. responseData.data = await ctx.service.ledger.addStdNode(tenderId, data.id, stdData);
  290. break;
  291. case stdDataAddType.withParent:
  292. responseData.data = await ctx.service.ledger.addStdNodeWithParent(tenderId, stdData, stdLib);
  293. break;
  294. default:
  295. throw '未知添加方法';
  296. }
  297. } catch (err) {
  298. responseData.err = 1;
  299. responseData.msg = err;
  300. }
  301. ctx.body = responseData;
  302. }
  303. /**
  304. * 批量插入数据 (Ajax)
  305. *
  306. * data = {id, batchData, batchType}
  307. * data.batchType = 'batchInsertChild'/'batchInsertNext'
  308. * data.batchData = [{name, children}] -- 项目节列表
  309. * data.batchData.children = [{code, name, unit, unit_price, quantity}] -- 工程量清单列表
  310. *
  311. * @param ctx
  312. * @returns {Promise<void>}
  313. */
  314. async batchInsert(ctx) {
  315. const responseData = {
  316. err: 0,
  317. msg: '',
  318. data: [],
  319. };
  320. try {
  321. const tenderId = ctx.params.id;
  322. if (!tenderId) {
  323. throw '当前未打开标段';
  324. }
  325. const tenderData = await ctx.service.tender.getDataById(tenderId);
  326. if (!tenderData || tenderData.user_id !== ctx.session.sessionUser.accountId || this._ledgerReadOnly(tenderData)) {
  327. throw '标段数据错误';
  328. }
  329. const data = JSON.parse(ctx.request.body.data);
  330. if ((isNaN(data.id) || data.id <= 0) || !data.batchType) {
  331. throw '参数错误';
  332. }
  333. switch (data.batchType) {
  334. case 'batchInsertChild':
  335. responseData.data = await ctx.service.ledger.batchInsertChild(tenderId, data.id, data.batchData);
  336. break;
  337. case 'batchInsertNext':
  338. responseData.data = await ctx.service.ledger.batchInsertNext(tenderId, data.id, data.batchData);
  339. break;
  340. default:
  341. throw '参数错误';
  342. }
  343. } catch (err) {
  344. responseData.err = 1;
  345. responseData.msg = err;
  346. }
  347. ctx.body = responseData;
  348. }
  349. /**
  350. * 查询
  351. *
  352. * @param ctx
  353. * @returns {Promise<void>}
  354. */
  355. async search(ctx) {
  356. const responseData = {
  357. err: 0,
  358. msg: '',
  359. data: [],
  360. };
  361. try {
  362. const tenderId = ctx.params.id;
  363. if (!tenderId) {
  364. throw '当前未打开标段';
  365. }
  366. const data = JSON.parse(ctx.request.body.data);
  367. if (!data.keyword || data.keyword === '') {
  368. throw '参数错误';
  369. }
  370. responseData.data = await ctx.service.ledger.search(tenderId, {
  371. value: ctx.app.mysql.escape('%' + data.keyword + '%'),
  372. operate: 'Like',
  373. fields: ['code', 'b_code', 'name'],
  374. });
  375. } catch (err) {
  376. this.log(err);
  377. responseData.err = 1;
  378. responseData.msg = err;
  379. }
  380. ctx.body = responseData;
  381. }
  382. /**
  383. * 定位
  384. * @param ctx
  385. * @returns {Promise<void>}
  386. */
  387. async locate(ctx) {
  388. const responseData = {
  389. err: 0,
  390. msg: '',
  391. data: [],
  392. };
  393. try {
  394. const tenderId = ctx.params.id;
  395. if (!tenderId) {
  396. throw '当前未打开标段';
  397. }
  398. const data = JSON.parse(ctx.request.body.data);
  399. if ((isNaN(data.id) || data.id <= 0)) {
  400. throw '参数错误';
  401. }
  402. responseData.data = await ctx.service.ledger.locateNode(tenderId, data.id);
  403. } catch (err) {
  404. this.log(err);
  405. responseData.err = 1;
  406. responseData.msg = err;
  407. }
  408. ctx.body = responseData;
  409. }
  410. /**
  411. * 获取全部子节点
  412. *
  413. * @param ctx
  414. * @returns {Promise<void>}
  415. */
  416. async posterity(ctx) {
  417. const responseData = {
  418. err: 0,
  419. msg: '',
  420. data: [],
  421. };
  422. try {
  423. const tenderId = ctx.params.id;
  424. if (!tenderId) {
  425. throw '当前未打开标段';
  426. }
  427. const data = JSON.parse(ctx.request.body.data);
  428. if ((isNaN(data.id) || data.id <= 0)) {
  429. throw '参数错误';
  430. }
  431. const expandData = await ctx.service.ledger.getPosterityByParentId(tenderId, data.id);
  432. responseData.data = { expand: expandData };
  433. } catch (err) {
  434. this.log(err);
  435. responseData.err = 1;
  436. responseData.msg = err;
  437. }
  438. ctx.body = responseData;
  439. }
  440. /**
  441. * 台账变更页面 (Get)
  442. *
  443. * @param {object} ctx - egg全局变量
  444. * @return {void}
  445. */
  446. async change(ctx) {
  447. try {
  448. const renderData = await this.getCommonRenderData(ctx);
  449. await this.layout('ledger/change.ejs', renderData, 'ledger/change_modal.ejs');
  450. } catch(err) {
  451. this.log(err);
  452. ctx.redirect(ctx.request.header.referer);
  453. }
  454. }
  455. /**
  456. * 计量台账页面 (Get)
  457. *
  458. * @param {object} ctx - egg全局变量
  459. * @return {void}
  460. */
  461. async index(ctx) {
  462. const renderData = {};
  463. await this.layout('ledger/index.ejs', renderData);
  464. }
  465. }
  466. return LedgerController;
  467. };