ledger_controller.js 17 KB

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