tender_controller.js 74 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510
  1. 'use strict';
  2. /**
  3. * 标段管理控制器
  4. *
  5. * @author CaiAoLin
  6. * @date 2018/2/5
  7. * @version
  8. */
  9. const tenderConst = require('../const/tender');
  10. const codeRuleConst = require('../const/code_rule');
  11. const settingConst = require('../const/setting.js');
  12. const tenderMenu = require('../../config/menu').tenderMenu;
  13. const auditConst = require('../const/audit');
  14. const shenpiConst = require('../const/shenpi');
  15. const accountGroup = require('../const/account_group').group;
  16. const accountPermission = require('../const/account_permission');
  17. const measureType = require('../const/tender').measureType;
  18. const billsPosConvert = require('../lib/bills_pos_convert');
  19. const path = require('path');
  20. const sendToWormhole = require('stream-wormhole');
  21. const scheduleConst = require('../const/schedule');
  22. const changeConst = require('../const/change');
  23. const tenderInfoModel = require('../lib/tender_info');
  24. const mapConst = require('../const/map');
  25. const advanceConst = require('../const/advance');
  26. module.exports = app => {
  27. class TenderController extends app.BaseController {
  28. /**
  29. * 构造函数
  30. *
  31. * @param {Object} ctx - egg全局变量
  32. * @return {void}
  33. */
  34. constructor(ctx) {
  35. super(ctx);
  36. ctx.showProject = true;
  37. ctx.showTitle = true;
  38. }
  39. async _getLedgerAuditInfo(tender) {
  40. tender.cur_flow = {
  41. title: '台账',
  42. status: auditConst.ledger.tiStatusString[tender.ledger_status],
  43. status_class: auditConst.ledger.tiStatusStringClass[tender.ledger_status],
  44. };
  45. if (tender.ledger_status === auditConst.ledger.status.uncheck) {
  46. tender.cur_flow.name = tender.user_name;
  47. } else {
  48. const cur = tender.ledger_status === auditConst.ledger.status.checkNo
  49. ? await this.ctx.service.ledgerAudit.getLastestAuditor(tender.id, tender.ledger_times - 1, auditConst.ledger.status.checkNo)
  50. : await this.ctx.service.ledgerAudit.getLastestAuditor(tender.id, tender.ledger_times, tender.ledger_status);
  51. if (cur) {
  52. tender.cur_flow.name = cur.name;
  53. if (cur.audit_order === 1) {
  54. tender.pre_flow = { name: tender.user_name, time: cur.begin_time };
  55. } else {
  56. const pre = await this.ctx.service.ledgerAudit.getAuditorByOrder(tender.id, cur.audit_order - 1, cur.times);
  57. if (pre) tender.pre_flow = { name: pre.name, time: pre.end_time };
  58. }
  59. } else {
  60. tender.cur_flow.name = '';
  61. }
  62. }
  63. }
  64. async _getStageAuditInfo(tender, stage) {
  65. tender.cur_flow = {
  66. title: '第' + stage.order + '期',
  67. status: auditConst.stage.tiStatusString[stage.status],
  68. status_class: auditConst.stage.tiStatusStringClass[stage.status],
  69. };
  70. if (stage.status === auditConst.stage.status.uncheck) {
  71. if (tender.user_id === stage.user_id) {
  72. tender.cur_flow.name = tender.user_name;
  73. } else {
  74. const user = await this.ctx.service.projectAccount.getDataById(stage.user_id);
  75. tender.cur_flow.name = user.name;
  76. }
  77. if (stage.order > 1) {
  78. const preStage = await this.ctx.service.stage.getDataByCondition({ tid: tender.id, order: stage.order - 1 });
  79. if (!preStage) return;
  80. const pre = await this.ctx.service.stageAudit.getLastestAuditor(preStage.id, preStage.times, auditConst.stage.status.checked);
  81. if (pre) tender.pre_flow = { name: pre.name, time: pre.end_time };
  82. }
  83. } else {
  84. let cur;
  85. if (stage.status === auditConst.stage.status.checkNo) {
  86. cur = await this.ctx.service.stageAudit.getLastestAuditor(stage.id, stage.times - 1, auditConst.stage.status.checkNo);
  87. } else if (stage.status === auditConst.stage.status.checked) {
  88. cur = await this.ctx.service.stageAudit.getLastestAuditor(stage.id, stage.times, auditConst.stage.status.checked);
  89. } else {
  90. cur = await this.ctx.service.stageAudit.getCurAuditor(stage.id, stage.times);
  91. }
  92. if (cur) {
  93. tender.cur_flow.name = cur.name;
  94. if (cur.order === 1) {
  95. tender.pre_flow = {};
  96. if (tender.user_id === stage.user_id) {
  97. tender.pre_flow.name = tender.user_name;
  98. } else {
  99. const user = await this.ctx.service.projectAccount.getDataById(stage.user_id);
  100. tender.pre_flow.name = user.name;
  101. }
  102. tender.pre_flow.time = cur.begin_time;
  103. } else {
  104. const pre = await this.ctx.service.stageAudit.getAuditorByOrder(stage.id, cur.order - 1, cur.times);
  105. if (pre) tender.pre_flow = { name: pre.name, time: pre.end_time };
  106. }
  107. } else {
  108. tender.cur_flow.name = '';
  109. }
  110. }
  111. }
  112. async _listDetail(view, modal = '') {
  113. try {
  114. // 获取用户新建标段权利
  115. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  116. const userPermission = accountInfo !== undefined && accountInfo.permission !== ''
  117. ? JSON.parse(accountInfo.permission) : null;
  118. const tenderList = await this.ctx.service.tender.getList('', userPermission, this.ctx.session.sessionUser.is_admin);
  119. for (const t of tenderList) {
  120. const tenderInfo = await this.ctx.service.tenderInfo.getTenderInfo(t.id);
  121. t.contract_price = tenderInfo.deal_param.contractPrice;
  122. let bCalcTp = t.user_id === this.ctx.session.sessionUser.accountId && (
  123. t.ledger_status === auditConst.ledger.status.checkNo || t.ledger_status === auditConst.ledger.status.uncheck);
  124. t.advance_tp = await this.ctx.service.advance.getSumAdvance(t.id);
  125. if (t.ledger_status === auditConst.ledger.status.checked) {
  126. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, true);
  127. if (t.lastStage && t.lastStage.status === auditConst.stage.status.uncheck &&
  128. t.lastStage.user_id !== this.ctx.session.sessionUser.accountId) {
  129. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id);
  130. }
  131. if (t.lastStage) await this.ctx.service.stage.checkStageGatherData(t.lastStage, this.ctx.session.sessionUser.is_admin);
  132. t.completeStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  133. if ((!bCalcTp) && t.measure_type === measureType.gcl.value) {
  134. bCalcTp = t.lastStage && t.lastStage.status !== auditConst.stage.status.checked && !t.lastStage.readOnly;
  135. }
  136. }
  137. if (bCalcTp) {
  138. const sum = await this.ctx.service.ledger.addUp({ tender_id: t.id/* , is_leaf: true*/ });
  139. t.total_price = sum.total_price;
  140. t.deal_tp = sum.deal_tp;
  141. }
  142. if (t.lastStage) {
  143. await this._getStageAuditInfo(t, t.lastStage);
  144. } else {
  145. await this._getLedgerAuditInfo(t);
  146. }
  147. }
  148. const categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  149. const valuations = await this.ctx.service.valuation.getProjectValidValuation(this.ctx.session.sessionProject.id);
  150. const renderData = {
  151. tenderList,
  152. tenderConst,
  153. settingConst,
  154. categoryData,
  155. measureType: tenderConst.measureType,
  156. jsFiles: this.app.jsFiles.common.concat(this.jsFiles),
  157. auditConst,
  158. userPermission,
  159. valuations,
  160. uid: this.ctx.session.sessionUser.accountId,
  161. pid: this.ctx.session.sessionProject.id,
  162. };
  163. await this.layout(view, renderData, modal);
  164. } catch (err) {
  165. this.log(err);
  166. this.ctx.redirect('/dashboard');
  167. }
  168. }
  169. async _list(view, renderData, modal = '', list_status = '') {
  170. try {
  171. renderData.tenderList = await this.ctx.service.tender.getList(list_status, renderData.userPermission, this.ctx.session.sessionUser.is_admin);
  172. for (const t of renderData.tenderList) {
  173. if (t.ledger_status === auditConst.ledger.status.checked) {
  174. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, true);
  175. t.completeStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  176. }
  177. if (t.lastStage) {
  178. if (t.lastStage.status === auditConst.stage.status.uncheck) {
  179. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(t.lastStage.user_id);
  180. t.status_users = status_name ? status_name.name : '';
  181. // const status_name = await this.ctx.service.stageAudit.getStatusName(t.lastStage.id, t.lastStage.times - 1);
  182. // t.status_users = status_name ? status_name.name : '';
  183. } else {
  184. const status_name = await this.ctx.service.stageAudit.getAuditorByStatus(t.lastStage.id, t.lastStage.status, t.lastStage.times);
  185. t.status_users = status_name ? status_name.name : '';
  186. }
  187. } else {
  188. if (t.ledger_status !== auditConst.ledger.status.uncheck) {
  189. const status_name = await this.ctx.service.ledgerAudit.getStatusName(t.id, t.ledger_times);
  190. t.status_users = status_name ? status_name.name : '';
  191. }
  192. }
  193. }
  194. renderData.categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  195. renderData.valuations = await this.ctx.service.valuation.getProjectValidValuation(this.ctx.session.sessionProject.id);
  196. renderData.tenderConst = tenderConst;
  197. renderData.settingConst = settingConst;
  198. renderData.measureType = tenderConst.measureType;
  199. renderData.jsFiles = this.app.jsFiles.common.concat(this.jsFiles);
  200. renderData.auditConst = auditConst;
  201. renderData.uid = this.ctx.session.sessionUser.accountId;
  202. renderData.pid = this.ctx.session.sessionProject.id;
  203. await this.layout(view, renderData, modal);
  204. } catch (err) {
  205. this.log(err);
  206. this.ctx.redirect('/dashboard');
  207. }
  208. }
  209. async listDefault(ctx) {
  210. this.jsFiles = this.app.jsFiles.tender.list;
  211. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  212. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  213. const renderData = {
  214. accountInfo,
  215. userPermission,
  216. };
  217. await this._list('tender/index.ejs', renderData, 'tender/modal.ejs');
  218. }
  219. /**
  220. * 标段概况(Get)
  221. *
  222. * @param {object} ctx - egg全局变量
  223. * @return {void}
  224. */
  225. async listInfo(ctx) {
  226. this.jsFiles = this.app.jsFiles.tender.info;
  227. await this._listDetail('tender/info.ejs', 'tender/modal.ejs');
  228. }
  229. /**
  230. * 计量进度(Get)
  231. *
  232. * @param ctx
  233. * @return {Promise<void>}
  234. */
  235. async listProgress(ctx) {
  236. this.jsFiles = this.app.jsFiles.tender.progress;
  237. await this._listDetail('tender/progress.ejs', 'tender/modal.ejs');
  238. }
  239. /**
  240. * 标段管理(Get)
  241. *
  242. * @param ctx
  243. * @return {Promise<void>}
  244. */
  245. async listManage(ctx) {
  246. this.jsFiles = this.app.jsFiles.tender.manage;
  247. // 先判断权限
  248. // 获取用户新建标段权利
  249. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  250. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  251. if (userPermission !== null && userPermission.tender !== undefined && userPermission.tender.indexOf('1') !== -1) {
  252. const renderData = {
  253. accountInfo,
  254. userPermission,
  255. };
  256. await this._list('tender/manage.ejs', renderData, 'tender/manage_modal.ejs', 'manage');
  257. // 获取用户新建标段权利
  258. // const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  259. // const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  260. //
  261. // const tenderList = await this.ctx.service.tender.getList('manage', userPermission);
  262. // for (const t of tenderList) {
  263. // t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, true);
  264. // t.completeStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  265. // }
  266. // const categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  267. // const valuations = await this.ctx.service.valuation.getProjectValidValuation(this.ctx.session.sessionProject.id);
  268. // const renderData = {
  269. // tenderList,
  270. // tenderConst,
  271. // settingConst,
  272. // categoryData,
  273. // measureType: tenderConst.measureType,
  274. // jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.manage),
  275. // auditConst,
  276. // userPermission,
  277. // valuations,
  278. // };
  279. // await this.layout('tender/manage.ejs', renderData, 'tender/manage_modal.ejs');
  280. } else {
  281. this.ctx.redirect(ctx.request.header.referer);
  282. }
  283. }
  284. /**
  285. * 新增标段(Ajax)
  286. *
  287. * @param ctx
  288. * @return {Promise<void>}
  289. */
  290. async addTender(ctx) {
  291. try {
  292. const responseData = {
  293. err: 0, msg: '', data: null,
  294. };
  295. // 获取用户新建标段权利
  296. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  297. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  298. if (userPermission === null || userPermission.tender === undefined || userPermission.tender.indexOf('1') === -1) {
  299. throw '当前用户没有创建标段的权限';
  300. }
  301. const data = JSON.parse(ctx.request.body.data);
  302. if (!data.name || data.name === '' || !data.valuation) {
  303. throw '标段信息不完整';
  304. }
  305. responseData.data = await ctx.service.tender.add(data);
  306. ctx.body = responseData;
  307. } catch (error) {
  308. this.log(error);
  309. ctx.body = { err: 1, msg: error.toString(), data: null };
  310. }
  311. }
  312. /**
  313. * 编辑标段(Ajax)
  314. *
  315. * @param ctx
  316. * @return {Promise<void>}
  317. */
  318. async updateTender(ctx) {
  319. try {
  320. const responseData = {
  321. err: 0, msg: '', data: null,
  322. };
  323. const data = JSON.parse(ctx.request.body.data);
  324. if (!data.id) {
  325. throw '提交信息错误';
  326. }
  327. if (!data.name || data.name === '') {
  328. throw '标段信息不完整';
  329. }
  330. if (await ctx.service.tender.save(data, data.id)) {
  331. responseData.data = await ctx.service.tender.getTender(data.id);
  332. }
  333. ctx.body = responseData;
  334. } catch (error) {
  335. this.log(error);
  336. ctx.body = { err: 1, msg: error.toString(), data: null };
  337. }
  338. }
  339. /**
  340. * 删除标段(Ajax)
  341. *
  342. * @param ctx
  343. * @return {Promise<void>}
  344. */
  345. async deleteTender(ctx) {
  346. try {
  347. const data = JSON.parse(ctx.request.body.data),
  348. result = [];
  349. if (!(data instanceof Array) && (data.length === 0)) {
  350. throw '提交数据有误';
  351. }
  352. for (const id of data) {
  353. if (await ctx.service.tender.deleteTenderNoBackup(id)) {
  354. result.push(id);
  355. }
  356. }
  357. ctx.body = { err: 0, msg: '', data: result };
  358. } catch (err) {
  359. ctx.body = { err: 1, msg: err.toString(), data: [] };
  360. }
  361. }
  362. /**
  363. * 标段概况(Get)
  364. *
  365. * @param ctx
  366. * @return {Promise<void>}
  367. */
  368. async tenderInfo(ctx) {
  369. try {
  370. const tender = ctx.tender.data;
  371. let bCalcTp = tender.user_id === this.ctx.session.sessionUser.accountId && (
  372. tender.ledger_status === auditConst.ledger.status.checkNo || tender.ledger_status === auditConst.ledger.status.uncheck);
  373. const stages = await ctx.service.stage.getValidStages(ctx.tender.id);
  374. const lastStage = stages.length > 0 ? stages[0] : null; // await ctx.service.stage.getLastestStage(ctx.tender.id);
  375. const [change_tp, change_p_tp, change_n_tp] = await ctx.service.change.getChangeTp(tender.id);
  376. tender.change_tp = change_tp;
  377. tender.change_p_tp = change_p_tp;
  378. tender.change_n_tp = change_n_tp;
  379. if (lastStage) {
  380. await this.ctx.service.stage.checkStageGatherData(lastStage, this.ctx.session.sessionUser.is_admin);
  381. if ((!bCalcTp) && tender.measure_type === measureType.gcl.value) {
  382. bCalcTp = lastStage.status !== auditConst.stage.status.checked && !lastStage.readOnly;
  383. }
  384. if (bCalcTp) {
  385. const sum = await this.ctx.service.ledger.addUp({ tender_id: tender.id/* , is_leaf: true*/ });
  386. tender.total_price = sum.total_price;
  387. tender.deal_tp = sum.deal_tp;
  388. }
  389. tender.sum = ctx.helper.add(tender.total_price, tender.change_tp);
  390. tender.gather_tp = ctx.helper.sum([lastStage.contract_tp, lastStage.qc_tp, lastStage.pc_tp]);
  391. tender.end_contract_tp = ctx.helper.sum([lastStage.contract_tp, lastStage.pre_contract_tp, lastStage.contract_pc_tp]);
  392. tender.end_qc_tp = ctx.helper.sum([lastStage.qc_tp, lastStage.pre_qc_tp, lastStage.qc_pc_tp]);
  393. tender.end_positive_qc_tp = ctx.helper.sum([lastStage.positive_qc_tp, lastStage.pre_positive_qc_tp, lastStage.positive_qc_pc_tp]);
  394. tender.end_negative_qc_tp = ctx.helper.sum([lastStage.negative_qc_tp, lastStage.pre_negative_qc_tp, lastStage.negative_qc_pc_tp]);
  395. tender.end_gather_tp = ctx.helper.add(tender.end_contract_tp, tender.end_qc_tp);
  396. tender.pre_gather_tp = ctx.helper.add(lastStage.pre_contract_tp, lastStage.pre_qc_tp);
  397. tender.yf_tp = lastStage.yf_tp;
  398. tender.qc_ratio = ctx.helper.mul(ctx.helper.div(tender.end_qc_tp, ctx.tender.info.deal_param.contractPrice, 2), 100);
  399. tender.pre_ratio = ctx.helper.mul(ctx.helper.div(tender.pre_gather_tp, tender.sum, 2), 100);
  400. tender.cur_ratio = ctx.helper.mul(ctx.helper.div(tender.gather_tp, tender.sum, 2), 100);
  401. tender.other_tp = ctx.helper.sub(ctx.helper.sub(tender.sum, tender.pre_gather_tp), tender.gather_tp);
  402. tender.other_ratio = Math.max(0, 100 - tender.pre_ratio - tender.cur_ratio);
  403. tender.end_yf_tp = ctx.helper.add(lastStage.yf_tp, lastStage.pre_yf_tp);
  404. tender.end_sf_tp = ctx.helper.add(lastStage.sf_tp, lastStage.pre_sf_tp);
  405. tender.undone_tp = ctx.helper.sub(ctx.helper.sub(ctx.helper.add(tender.total_price, change_tp), tender.end_contract_tp), tender.end_qc_tp);
  406. if (lastStage.status === auditConst.stage.status.uncheck) {
  407. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(lastStage.user_id);
  408. lastStage.status_users = status_name ? status_name.name : '';
  409. lastStage.auditors = [];
  410. } else {
  411. lastStage.status = lastStage.status === auditConst.stage.status.checkNoPre ? auditConst.stage.status.checking : lastStage.status;
  412. let cur;
  413. if (lastStage.status === auditConst.stage.status.checked) {
  414. cur = await this.ctx.service.stageAudit.getLastestAuditor(lastStage.id, lastStage.times, auditConst.stage.status.checked);
  415. } else if (lastStage.status === auditConst.stage.status.checking) {
  416. cur = await this.ctx.service.stageAudit.getCurAuditor(lastStage.id, lastStage.times);
  417. } else {
  418. cur = await this.ctx.service.stageAudit.getAuditorByStatus(lastStage.id, lastStage.status, lastStage.times);
  419. }
  420. // const status_name = await this.ctx.service.stageAudit.getAuditorByStatus(lastStage.id, lastStage.status, lastStage.times);
  421. lastStage.status_users = cur ? cur.name : '';
  422. const times = lastStage.status === auditConst.stage.status.checkNo ? lastStage.times - 1 : lastStage.times;
  423. lastStage.auditors = await ctx.service.stageAudit.getFinalAuditGroup(lastStage.id, times);
  424. }
  425. } else {
  426. if (bCalcTp) {
  427. const sum = await this.ctx.service.ledger.addUp({ tender_id: tender.id/* , is_leaf: true*/ });
  428. tender.total_price = sum.total_price;
  429. tender.deal_tp = sum.deal_tp;
  430. }
  431. tender.sum = ctx.helper.add(tender.total_price, tender.change_tp);
  432. if (tender.ledger_status !== auditConst.ledger.status.uncheck) {
  433. const status_name = await this.ctx.service.ledgerAudit.getStatusName(tender.id, tender.ledger_times);
  434. tender.status_users = status_name ? status_name.name : '';
  435. const times = tender.ledger_status === auditConst.ledger.status.checkNo ? tender.ledger_times - 1 : tender.ledger_times;
  436. tender.auditors = await ctx.service.ledgerAudit.getFinalAuditGroup(tender.id, times);
  437. } else {
  438. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(tender.user_id);
  439. tender.status_users = status_name ? status_name.name : '';
  440. }
  441. }
  442. const tiModel = new tenderInfoModel(ctx);
  443. const gclChapter = await tiModel.gatherChapter();
  444. const monthProgress = [];
  445. for (const s of stages) {
  446. if (s.s_time) {
  447. let progress = monthProgress.find(function(x) {
  448. return x.month === s.s_time;
  449. });
  450. if (!progress) {
  451. progress = { month: s.s_time };
  452. monthProgress.push(progress);
  453. }
  454. progress.tp = ctx.helper.add(ctx.helper.add(progress.tp, s.contract_tp), s.qc_tp);
  455. }
  456. }
  457. monthProgress.sort(function(x, y) {
  458. return Date.parse(x.month) - Date.parse(y.month);
  459. });
  460. let sum = 0;
  461. for (const p of monthProgress) {
  462. p.ratio = ctx.helper.mul(ctx.helper.div(p.tp, tender.sum, 4), 100);
  463. sum = ctx.helper.add(sum, p.tp);
  464. p.end_tp = sum;
  465. p.end_ratio = ctx.helper.mul(ctx.helper.div(p.end_tp, tender.sum, 4), 100);
  466. }
  467. const revise = await ctx.service.ledgerRevise.getLastestRevise(tender.id);
  468. const tenders = await ctx.service.tender.getList('', null, 1);
  469. const categoryData = await ctx.service.category.getAllCategory(ctx.session.sessionProject.id);
  470. // 变更图表数据
  471. const change_done_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.checked);
  472. const change_doing_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.checking);
  473. const change_uncheck_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.uncheck);
  474. const change_status_total = [
  475. { num: change_uncheck_total, name: '待上报' },
  476. { num: change_doing_total, name: '审批中' },
  477. { num: change_done_total, name: '已完成' },
  478. ];
  479. const change_common_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.common.value);
  480. const change_more_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.more.value);
  481. const change_great_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.great.value);
  482. const change_quality_total = [
  483. { num: change_common_total, name: '一般变更' },
  484. { num: change_more_total, name: '较大变更' },
  485. { num: change_great_total, name: '重大变更' },
  486. ];
  487. // 调差最新期数据
  488. const materials = await ctx.service.material.getValidMaterials(ctx.tender.id);
  489. let materialData = null;
  490. if (materials && materials.length > 0) {
  491. materialData = materials[0];
  492. materialData.curAuditor = await ctx.service.materialAudit.getAuditorByStatus(materialData.id, materialData.status, materialData.times);
  493. const times = materialData.status === auditConst.material.status.checkNo ? materialData.times - 1 : materialData.times;
  494. materialData.auditors = materialData.status === auditConst.material.status.uncheck ? [] : await ctx.service.materialAudit.getFinalAuditGroup(materialData.id, times);
  495. }
  496. // 修订完成数目
  497. // const reviseNum = await ctx.service.ledgerRevise.count({ tid: tender.id, status: auditConst.revise.status.checked });
  498. // 计量完成概况
  499. // tender.total_price
  500. const stage_total = [
  501. { num: tender.end_contract_tp ? tender.end_contract_tp : 0, name: '合同完成' },
  502. { num: tender.end_qc_tp ? tender.end_qc_tp : 0, name: '变更完成' },
  503. { num: tender.undone_tp ? tender.undone_tp : 0, name: '未完成' },
  504. ];
  505. // 地图
  506. const tenderALLInfo = await ctx.service.tender.getDataById(tender.id);
  507. const hadMap = tenderALLInfo.had_map === 0 ? 1 : tenderALLInfo.had_map;// 0为初始值,因为默认可能会变化,所以暂时把0都默认为1。共三种模式坐标模式1,图片模式2,无图模式3。
  508. const tenderMapList = await ctx.service.tenderMap.getAllDataByCondition({ where: { tid: tender.id } });
  509. // 默认坐标,否则则取办事处坐标
  510. const projectData = await ctx.service.project.getDataById(ctx.session.sessionProject.id);
  511. let map_json = {
  512. province: mapConst.map[0].province,
  513. lng: mapConst.map[0].lng,
  514. lat: mapConst.map[0].lat,
  515. level: 15,
  516. };
  517. if (projectData.map_json) {
  518. map_json = JSON.parse(projectData.map_json);
  519. } else {
  520. const mapInfo = ctx.helper._.find(mapConst.map, { office: projectData.manager_office });
  521. if (mapInfo) {
  522. map_json.province = mapInfo.province;
  523. map_json.lng = mapInfo.lng;
  524. map_json.lat = mapInfo.lat;
  525. }
  526. }
  527. const renderData = {
  528. tenders,
  529. categoryData,
  530. tender,
  531. revise,
  532. tenderInfo: ctx.tender.info,
  533. tenderMenu: this.menu.tenderMenu,
  534. preUrl: '/tender/' + ctx.tender.id,
  535. cooperation: ctx.session.sessionUser.cooperation,
  536. lastStage,
  537. stages: stages.reverse(),
  538. monthProgress,
  539. audit: auditConst,
  540. change_status_total,
  541. change_quality_total,
  542. materialData,
  543. // reviseNum,
  544. stage_total,
  545. hadMap,
  546. map_pic: tenderALLInfo.map_pic,
  547. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.tenderInfo),
  548. gclChapter,
  549. tenderMapList,
  550. map_json,
  551. fujianOssPath: ctx.app.config.fujianOssPath,
  552. };
  553. if (ctx.session.sessionUser.is_admin) {
  554. renderData.tourists = await ctx.service.tenderTourist.getTourists(tender.id);
  555. for (const t of renderData.tourists) {
  556. t.permission = await ctx.service.tenderTourist.getTouristPermission(t);
  557. }
  558. // 获取所有项目参与者
  559. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  560. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  561. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  562. });
  563. const accountGroupList = accountGroup.map((item, idx) => {
  564. const groupList = accountList.filter(item => item.account_group === idx);
  565. return { groupName: item, groupList };
  566. });
  567. renderData.accountList = accountList;
  568. renderData.accountGroup = accountGroupList;
  569. }
  570. if (ctx.session.sessionProject.page_show.xxjd && ctx.session.sessionUser.is_admin) {
  571. // 投资进度内容
  572. renderData.scheduleAuditList = await ctx.service.scheduleAudit.getAllDataByCondition({ where: { tid: tender.id } });
  573. renderData.scPermission = scheduleConst.permission;
  574. }
  575. await this.layout('tender/detail.ejs', renderData, 'tender/detail_modal.ejs');
  576. } catch (error) {
  577. this.log(error);
  578. this.ctx.redirect('/list');
  579. }
  580. }
  581. /**
  582. * 保存标段属性等(Ajax)
  583. *
  584. * @param ctx
  585. * @return {Promise<void>}
  586. */
  587. async saveTenderInfo(ctx) {
  588. try {
  589. const data = JSON.parse(ctx.request.body.data);
  590. if (!data) {
  591. throw '提交数据错误';
  592. }
  593. // 针对查阅所有标段者但非原报和审批人提示
  594. const times = ctx.tender.data.ledger_status === auditConst.ledger.status.checkNo ? ctx.tender.data.ledger_times - 1 : ctx.tender.data.ledger_times;
  595. const auditors = await this.service.ledgerAudit.getAuditors(ctx.tender.id, times);
  596. const auditorsId = ctx.helper._.map(auditors, 'audit_id');
  597. const stageAuditors = await this.service.stageAudit.getAllAuditors(ctx.tender.id);
  598. const stageAUditorsId = ctx.helper._.map(stageAuditors, 'aid');
  599. const accountId = ctx.session.sessionUser.accountId;
  600. if (auditorsId.indexOf(accountId) === -1 && ctx.tender.data.user_id !== accountId &&
  601. stageAUditorsId.indexOf(accountId) === -1) {
  602. throw '您无权修改标段设置内容';
  603. }
  604. if (ctx.tender.data.ledger_status === auditConst.ledger.status.checked) {
  605. if (data.deal_param) {
  606. const lastStage = await this.ctx.service.stage.getLastestStage(ctx.tender.id, true);
  607. if (lastStage) {
  608. if (lastStage.order > 1 || (lastStage.status === auditConst.stage.status.checked || lastStage.status === auditConst.stage.status.checking)) throw '第一期上报后不可修改合同参数';
  609. if (lastStage.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改合同参数';
  610. }
  611. }
  612. }
  613. if (data.decimal) {
  614. if (ctx.tender.data.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改小数位数';
  615. await ctx.service.tenderInfo.saveDecimal(ctx.tender.id, data.decimal, ctx.tender.info.decimal);
  616. } else if (data.precision) {
  617. if (ctx.tender.data.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改清单精度';
  618. await ctx.service.tenderInfo.savePrecision(ctx.tender.id,
  619. data.precision, ctx.tender.info.precision, ctx.tender.info.decimal);
  620. } else {
  621. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, data);
  622. }
  623. ctx.body = { err: 0, msg: '', data: JSON.parse(ctx.request.body.data) };
  624. } catch (err) {
  625. this.log(err);
  626. ctx.body = this.ajaxErrorBody(err, '保存标段设置失败');
  627. }
  628. }
  629. async saveTenderInfo2(ctx) {
  630. try {
  631. const data = JSON.parse(ctx.request.body.data);
  632. if (!data || (!data.ledger_check && !data.fun_rela)) throw '提交数据错误';
  633. if (!ctx.session.sessionUser.is_admin) throw '您无权修改该内容';
  634. const updateData = {};
  635. if (data.ledger_check) updateData.ledger_check = data.ledger_check;
  636. if (data.fun_rela) updateData.fun_rela = data.fun_rela;
  637. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, data);
  638. ctx.body = { err: 0, msg: '', data: JSON.parse(ctx.request.body.data) };
  639. } catch (err) {
  640. this.log(err);
  641. ctx.body = this.ajaxErrorBody(err, '保存失败');
  642. }
  643. }
  644. /**
  645. * 设置标段计量类型并调整到标段概况(Get)
  646. *
  647. * @param ctx
  648. * @return {Promise<void>}
  649. */
  650. async tenderType(ctx) {
  651. try {
  652. const tenderId = ctx.params.id,
  653. type = ctx.query.type;
  654. if (!tenderId) {
  655. throw '当前未打开标段';
  656. }
  657. await ctx.service.tender.checkTender(tenderId);
  658. if (!ctx.tender) {
  659. throw '标段数据错误';
  660. }
  661. if (!ctx.tender.measure_type) {
  662. await ctx.service.tender.setTenderType(ctx.tender, parseInt(type));
  663. }
  664. ctx.redirect('/tender/' + tenderId);
  665. } catch (error) {
  666. ctx.helper.log(error);
  667. this.postError(error, '设置标段类型错误');
  668. ctx.redirect('/list');
  669. }
  670. }
  671. /**
  672. * 标段协作办公
  673. *
  674. * @param {Object} ctx - egg全局变量
  675. * @return {void}
  676. */
  677. async tenderCooperation(ctx) {
  678. const tenderId = ctx.params.id;
  679. try {
  680. if (!ctx.session.sessionUser.cooperation) {
  681. throw '权限不足';
  682. }
  683. const tender = await ctx.service.tender.getDataById(tenderId);
  684. const user = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);
  685. // 获取已参与协作用户列表
  686. const cooperationArray = [];
  687. if (tender.cooperation !== null && tender.cooperation !== '') {
  688. const cooperationList = JSON.parse(tender.cooperation);
  689. for (const cl in cooperationList) {
  690. const clArray = [];
  691. for (const audit of cooperationList[cl]) {
  692. const userInfo = await ctx.service.projectAccount.getDataById(audit);
  693. clArray.push(userInfo);
  694. }
  695. cooperationArray[cl] = clArray;
  696. }
  697. }
  698. const renderData = {
  699. user,
  700. tender,
  701. tenderMenu: this.menu.tenderMenu,
  702. preUrl: '/tender/' + tenderId,
  703. tenderPermissionList: accountPermission.tenderPermissionList,
  704. cooperationArray,
  705. };
  706. await this.layout('tender/cooperation.ejs', renderData, 'tender/cooperationModal.ejs');
  707. } catch (error) {
  708. ctx.helper.log(error);
  709. this.ctx.redirect('/tender/' + tenderId);
  710. }
  711. }
  712. /**
  713. * 添加标段操作
  714. *
  715. * @param {Object} ctx - egg全局变量
  716. * @return {void}
  717. */
  718. async add(ctx) {
  719. try {
  720. const rule = ctx.service.tender.rule('add');
  721. ctx.helper.validate(rule);
  722. const result = ctx.service.tender.add(ctx.request.body);
  723. if (!result) {
  724. throw '新增标段失败';
  725. }
  726. } catch (error) {
  727. ctx.helper.log(error);
  728. this.setMessage(error.toString(), this.messageType.ERROR);
  729. }
  730. ctx.redirect(ctx.request.header.referer);
  731. }
  732. /**
  733. * 保存标段操作
  734. *
  735. * @param {Object} ctx - egg全局变量
  736. * @return {void}
  737. */
  738. async save(ctx) {
  739. let id = ctx.request.body.tenderId;
  740. id = parseInt(id);
  741. try {
  742. if (isNaN(id) || id < 0) {
  743. throw '参数错误';
  744. }
  745. // 获取数据规则
  746. const rule = ctx.service.tender.rule('save');
  747. ctx.validate(rule);
  748. const result = await ctx.service.tender.save(ctx.request.body, id);
  749. if (!result) {
  750. throw '保存标段数据失败';
  751. }
  752. this.setMessage('保存标段数据成功', this.messageType.SUCCESS);
  753. } catch (error) {
  754. this.postError(error, '保存标段数据失败');
  755. }
  756. ctx.redirect(ctx.request.header.referer);
  757. }
  758. /**
  759. * 删除标段
  760. *
  761. * @param {Object} ctx -egg全局变量
  762. * @return {void}
  763. */
  764. async delete(ctx) {
  765. let id = ctx.request.body.tenderId;
  766. id = parseInt(id);
  767. try {
  768. if (isNaN(id) || id <= 0) {
  769. throw '参数错误';
  770. }
  771. const result = ctx.service.tender.deleteTenderById(id);
  772. if (!result) {
  773. throw '删除标段失败';
  774. }
  775. this.setMessage('删除标段成功', this.messageType.SUCCESS);
  776. } catch (error) {
  777. this.postError(error, '删除标段失败');
  778. }
  779. ctx.redirect(ctx.request.header.referer);
  780. }
  781. async rule(ctx) {
  782. const responseData = {
  783. err: 0,
  784. msg: '',
  785. data: {},
  786. };
  787. try {
  788. const tenderId = ctx.session.sessionUser.tenderId;
  789. if (!tenderId) {
  790. throw '当前未打开标段';
  791. }
  792. const data = JSON.parse(ctx.request.body.data);
  793. if (!data.rule || !JSON.parse(data.data).length || !codeRuleConst.ruleField[data.rule]) {
  794. throw '请选择组件再添加';
  795. }
  796. if (!data.connector) {
  797. throw '请选择连接符';
  798. }
  799. const updateData = {
  800. id: tenderId,
  801. };
  802. if (data.type) {
  803. const tenderData = await ctx.service.tender.getDataById(tenderId);
  804. const c_code_rules = tenderData.c_code_rules ? JSON.parse(tenderData.c_code_rules) : {};
  805. c_code_rules[data.type + '_rule'] = JSON.parse(data.data);
  806. c_code_rules[data.type + '_rule_first'] = 0;
  807. c_code_rules[data.type + '_connector'] = data.connector;
  808. updateData.c_code_rules = JSON.stringify(c_code_rules);
  809. } else {
  810. updateData[codeRuleConst.ruleField[data.rule]] = data.data;
  811. updateData.c_connector = data.connector;
  812. updateData.c_rule_first = 0;
  813. }
  814. const result = await ctx.service.tender.db.update(ctx.service.tender.tableName, updateData);
  815. if (result.affectedRows !== 1) {
  816. throw '更新规则失败';
  817. }
  818. } catch (err) {
  819. ctx.helper.log(err);
  820. responseData.err = 1;
  821. responseData.msg = err.toString();
  822. }
  823. ctx.body = responseData;
  824. }
  825. async ruleFirst(ctx) {
  826. const responseData = {
  827. err: 0,
  828. msg: '',
  829. data: {},
  830. };
  831. try {
  832. const tenderId = ctx.session.sessionUser.tenderId;
  833. if (!tenderId) {
  834. throw '当前未打开标段';
  835. }
  836. const updateData = {
  837. id: tenderId,
  838. };
  839. const data = JSON.parse(ctx.request.body.data);
  840. if (data && data.type) {
  841. const tenderData = await ctx.service.tender.getDataById(tenderId);
  842. const c_code_rules = tenderData.c_code_rules ? JSON.parse(tenderData.c_code_rules) : {};
  843. c_code_rules[data.type + '_rule_first'] = 0;
  844. updateData.c_code_rules = JSON.stringify(c_code_rules);
  845. } else {
  846. updateData.c_rule_first = 0;
  847. }
  848. const result = await ctx.service.tender.db.update(ctx.service.tender.tableName, updateData);
  849. if (result.affectedRows !== 1) {
  850. throw '更新规则失败';
  851. }
  852. } catch (err) {
  853. ctx.helper.log(err);
  854. responseData.err = 1;
  855. responseData.msg = err.toString();
  856. }
  857. ctx.body = responseData;
  858. }
  859. async shenpiSet(ctx) {
  860. if (ctx.session.sessionUser.is_admin === 0) {
  861. ctx.request.headers.referer ? ctx.redirect(ctx.request.headers.referer) : ctx.redirect('/list');
  862. }
  863. // 获取所有项目参与者
  864. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  865. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  866. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  867. });
  868. const accountGroupList = accountGroup.map((item, idx) => {
  869. const groupList = accountList.filter(item => item.account_group === idx);
  870. return { groupName: item, groupList };
  871. });
  872. // 获取固定审批流 or 固定终审
  873. for (const sp of shenpiConst.sp_lc) {
  874. sp.status = ctx.tender.info.shenpi ? ctx.tender.info.shenpi[sp.code] : shenpiConst.sp_status.sqspr;
  875. if (sp.status === shenpiConst.sp_status.gdspl) {
  876. sp.auditList = await ctx.service.shenpiAudit.getAuditList(ctx.tender.id, sp.type, sp.status);
  877. } else if (sp.status === shenpiConst.sp_status.gdzs) {
  878. sp.audit = await ctx.service.shenpiAudit.getAudit(ctx.tender.id, sp.type, sp.status);
  879. }
  880. }
  881. const tenders = await ctx.service.tender.getList('', null, 1);
  882. const removeTenders = [];
  883. for (const tender of tenders) {
  884. const shenpiInfo = await ctx.service.tenderInfo.getTenderShenpiInfo(tender.id);
  885. if (!shenpiInfo) {
  886. removeTenders.push(tender.id);
  887. } else {
  888. tender.shenpiInfo = shenpiInfo;
  889. // 获取所有的固定审批流或固定终审
  890. const shenpiauditList = {};
  891. for (const shenpi in tender.shenpiInfo) {
  892. if (tender.shenpiInfo[shenpi] === shenpiConst.sp_status.gdspl) {
  893. const shenpiList = await ctx.service.shenpiAudit.getAllDataByCondition({ where: { tid: tender.id, sp_type: shenpiConst.sp_type[shenpi], sp_status: tender.shenpiInfo[shenpi] } });
  894. const shenpiIdList = ctx.helper._.map(shenpiList, 'audit_id');
  895. shenpiauditList[shenpi] = shenpiIdList.length ? shenpiIdList : null;
  896. } else if (tender.shenpiInfo[shenpi] === shenpiConst.sp_status.gdzs) {
  897. const shenpiInfo = await ctx.service.shenpiAudit.getDataByCondition({ tid: tender.id, sp_type: shenpiConst.sp_type[shenpi], sp_status: tender.shenpiInfo[shenpi] });
  898. shenpiauditList[shenpi] = shenpiInfo && shenpiInfo.audit_id ? [shenpiInfo.audit_id] : null;
  899. }
  900. }
  901. tender.shenpiauditList = shenpiauditList;
  902. }
  903. }
  904. if (removeTenders.length > 0) {
  905. ctx.helper._.remove(tenders, function(n) {
  906. return removeTenders.indexOf(n.id) !== -1;
  907. });
  908. }
  909. const categoryData = await ctx.service.category.getAllCategory(ctx.session.sessionProject.id);
  910. // 是否修订中
  911. const lastRevise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);
  912. const revising = (lastRevise && lastRevise.status !== auditConst.revise.status.checked) || false;
  913. const cooperationNum = await ctx.service.ledgerCooperation.count({ tid: ctx.tender.id, status: 1 });
  914. const renderData = {
  915. shenpi: shenpiConst,
  916. accountList,
  917. accountGroup: accountGroupList,
  918. tenders,
  919. categoryData,
  920. auditConst,
  921. revising,
  922. measureType,
  923. cooperationNum,
  924. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.shenpi),
  925. };
  926. await this.layout('tender/shenpi.ejs', renderData, 'tender/shenpi_modal.ejs');
  927. }
  928. async saveTenderInfoShenpi(ctx) {
  929. try {
  930. const data = JSON.parse(ctx.request.body.data);
  931. if (!data) {
  932. throw '提交数据错误';
  933. }
  934. // 判断修改权限
  935. if (ctx.session.sessionUser.is_admin === 0) {
  936. throw '你没有权限修改审批流程';
  937. }
  938. // let postData = {};
  939. // if (!ctx.tender.info.shenpi) {
  940. // for (const sp of shenpiConst.sp_lc) {
  941. // if (sp.code === data.code) {
  942. // postData[sp.code] = data.status;
  943. // } else {
  944. // postData[sp.code] = shenpiConst.sp_status.sqspr;
  945. // }
  946. // }
  947. // } else {
  948. const postData = ctx.tender.info.shenpi;
  949. postData[data.code] = data.status;
  950. if (data.code === shenpiConst.sp_lc[shenpiConst.sp_type.stage - 1].code) {
  951. const status = parseInt(data.status) === shenpiConst.sp_status.gdspl ? 1 : 0;
  952. await ctx.service.ledgerCooperation.changeAllStatus(status);
  953. }
  954. // }
  955. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, { shenpi: postData });
  956. let auditList = [];
  957. if (data.status === shenpiConst.sp_status.gdspl) {
  958. auditList = await ctx.service.shenpiAudit.getAuditList(ctx.tender.id, shenpiConst.sp_type[data.code], data.status);
  959. } else if (data.status === shenpiConst.sp_status.gdzs) {
  960. auditList = await ctx.service.shenpiAudit.getAudit(ctx.tender.id, shenpiConst.sp_type[data.code], data.status);
  961. }
  962. ctx.body = { err: 0, msg: '', data: auditList };
  963. } catch (err) {
  964. this.log(err);
  965. ctx.body = this.ajaxErrorBody(err, '保存审批流程设置失败');
  966. }
  967. }
  968. async saveShenpiAudit(ctx) {
  969. try {
  970. const data = JSON.parse(ctx.request.body.data);
  971. if (!data) {
  972. throw '提交数据错误';
  973. }
  974. // 判断修改权限
  975. if (ctx.session.sessionUser.is_admin === 0) {
  976. throw '你没有权限修改审批流程';
  977. }
  978. let info = '';
  979. switch (data.type) {
  980. case 'add':
  981. const result = await ctx.service.shenpiAudit.addAudit(data);
  982. if (result) {
  983. throw '添加审批人失败';
  984. }
  985. break;
  986. case 'del':
  987. await ctx.service.shenpiAudit.removeAudit(data);
  988. break;
  989. case 'copy2ot':
  990. await ctx.service.shenpiAudit.copyAudit2otherTender(data);
  991. break;
  992. case 'copy2os':
  993. await ctx.service.shenpiAudit.copyAudit2otherShenpi(data);
  994. break;
  995. case 'pwd':
  996. info = await ctx.service.ledgerCooperation.save(data);
  997. break;
  998. case 'company':
  999. info = await ctx.service.ledgerCooperation.saveCompany(data);
  1000. break;
  1001. default:break;
  1002. }
  1003. ctx.body = { err: 0, msg: '', data: info };
  1004. } catch (err) {
  1005. this.log(err);
  1006. ctx.body = this.ajaxErrorBody(err, '保存审批流程设置失败');
  1007. }
  1008. }
  1009. /**
  1010. * 签名,上传图片 (Ajax)
  1011. * @param ctx
  1012. * @return {Promise<void>}
  1013. */
  1014. async saveCooperateSign(ctx) {
  1015. try {
  1016. const stream = await ctx.getFileStream();
  1017. const create_time = Date.parse(new Date()) / 1000;
  1018. const id = stream.fields.id;
  1019. const fileInfo = path.parse(stream.filename);
  1020. const fileName = path.join('public/upload', ctx.tender.id.toString(), 'sign', 'signImg_' + create_time + fileInfo.ext);
  1021. await ctx.helper.saveStreamFile(stream, path.join(this.app.baseDir, 'app', fileName));
  1022. if (stream) {
  1023. await sendToWormhole(stream);
  1024. }
  1025. await ctx.service.ledgerCooperation.saveSign(id, fileName);
  1026. ctx.body = { err: 0, msg: '', data: fileName };
  1027. } catch (err) {
  1028. this.log(err);
  1029. ctx.body = { err: 1, msg: err.toString(), data: null };
  1030. }
  1031. }
  1032. /**
  1033. * 拷贝设置
  1034. * @param {object} ctx - 上下文
  1035. */
  1036. async copyTender(ctx) {
  1037. try {
  1038. const id = ctx.tender.data.id;
  1039. const { id: copy_id = '', type: typeArr = [] } = JSON.parse(ctx.request.body.data);
  1040. await ctx.service.tenderInfo.copyTenderHandler(id, copy_id, typeArr);
  1041. ctx.body = { err: 0, msg: '' };
  1042. } catch (error) {
  1043. this.log(error);
  1044. ctx.body = this.ajaxErrorBody(error, '保存审批流程设置失败');
  1045. }
  1046. }
  1047. /**
  1048. * 游客账号设置
  1049. * @param {object} ctx - 上下文
  1050. */
  1051. async saveTourist(ctx) {
  1052. try {
  1053. const data = JSON.parse(ctx.request.body.data);
  1054. if (!data) {
  1055. throw '提交数据错误';
  1056. }
  1057. // 判断修改权限
  1058. if (ctx.session.sessionUser.is_admin === 0) {
  1059. throw '你没有权限修改游客账号';
  1060. }
  1061. let info = '';
  1062. switch (data.type) {
  1063. case 'add':
  1064. const result = await ctx.service.tenderTourist.addAudit(data);
  1065. if (!result) {
  1066. throw '添加审批人失败';
  1067. }
  1068. info = result;
  1069. break;
  1070. case 'del':
  1071. await ctx.service.tenderTourist.removeAudit(data);
  1072. break;
  1073. case 'permission':
  1074. await ctx.service.tenderTourist.setPermission(data);
  1075. break;
  1076. default:break;
  1077. }
  1078. ctx.body = { err: 0, msg: '', data: info };
  1079. } catch (err) {
  1080. this.log(err);
  1081. ctx.body = this.ajaxErrorBody(err, '保存游客账号设置失败');
  1082. }
  1083. }
  1084. /**
  1085. * 获取部位明细数据(Ajax)
  1086. *
  1087. * @param ctx
  1088. * @return {Promise<void>}
  1089. */
  1090. async loadLedgerData(ctx) {
  1091. try {
  1092. const ledgerData = await ctx.service.ledger.getData(ctx.tender.id);
  1093. const posData = ctx.tender.data.measure_type === measureType.tz.value
  1094. ? await ctx.service.pos.getPosData({ tid: ctx.tender.id }) : [];
  1095. const convert = new billsPosConvert(ctx);
  1096. convert.loadData(ledgerData, posData, []);
  1097. const result = await convert.convert();
  1098. const ledgerCooperationList = await ctx.service.ledgerCooperation.getAllDataByCondition({ where: { tid: ctx.tender.id, status: 1 } });
  1099. ctx.body = { err: 0, msg: '', data: { ledgerList: result, ledgerCooperationList } };
  1100. } catch (err) {
  1101. this.log(err);
  1102. ctx.body = { err: 1, msg: err.toString(), data: [] };
  1103. }
  1104. }
  1105. async billsTag(ctx) {
  1106. try {
  1107. const isValidTourist = ctx.tender.isTourist && ctx.tender.touristPermission.tag;
  1108. if (ctx.stage) {
  1109. const isAuditor = ctx.stage.users.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1110. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1111. } else if (ctx.revise) {
  1112. const isAuditor = ctx.revise.reviseUsers.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1113. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1114. } else {
  1115. const isAuditor = ctx.tender.ledgerUsers.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1116. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1117. }
  1118. const data = JSON.parse(ctx.request.body.data);
  1119. const result = await ctx.service.ledgerTag.update(data);
  1120. ctx.body = { err: 0, msg: '', data: result };
  1121. } catch (err) {
  1122. this.log(err);
  1123. ctx.body = this.ajaxErrorBody(err, '书签数据错误');
  1124. }
  1125. }
  1126. async listLoad(ctx) {
  1127. try {
  1128. const data = JSON.parse(ctx.request.body.data);
  1129. if (!data.tid || !data.lid || !data.type) throw '数据错误';
  1130. const responseData = {
  1131. err: 0,
  1132. msg: '',
  1133. data: { ledgerAuditConst: auditConst.ledger, stageAuditConst: auditConst.stage },
  1134. };
  1135. responseData.data.category = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  1136. // 获取用户权限
  1137. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  1138. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  1139. const tenderList = await this.ctx.service.tender.getList('', userPermission);
  1140. for (const t of tenderList) {
  1141. if (t.ledger_status === auditConst.ledger.status.checked) {
  1142. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, false);
  1143. }
  1144. }
  1145. if (data.type === 'ledger') {
  1146. responseData.data.tenders = tenderList.filter(x => {
  1147. return x.ledger_status === auditConst.ledger.status.checked;
  1148. });
  1149. const history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1150. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1151. } else if (data.type === 'stage') {
  1152. responseData.data.tenders = tenderList.filter(x => {
  1153. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastStage;
  1154. });
  1155. let history = await this.ctx.service.sumLoadHistory.getStageHistory(data.tid, data.lid);
  1156. if (history) {
  1157. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'stage' };
  1158. } else {
  1159. history = await this.ctx.service.sumLoadHistory.getReviseHistory(data.tid, data.lid);
  1160. if (history) {
  1161. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'revise' };
  1162. } else {
  1163. history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1164. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1165. }
  1166. }
  1167. } else if (data.type === 'revise') {
  1168. responseData.data.tenders = tenderList.filter(x => {
  1169. return x.ledger_status === auditConst.ledger.status.checked;
  1170. });
  1171. let history = await this.ctx.service.sumLoadHistory.getReviseHistory(data.tid, data.lid);
  1172. if (history) {
  1173. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'revise' };
  1174. } else {
  1175. history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1176. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1177. }
  1178. }
  1179. ctx.body = responseData;
  1180. } catch(err) {
  1181. this.log(err);
  1182. this.ajaxErrorBody(err, '获取标段列表错误');
  1183. }
  1184. }
  1185. async listLoad2(ctx) {
  1186. try {
  1187. const data = JSON.parse(ctx.request.body.data);
  1188. if (!data.type) throw '数据错误';
  1189. const responseData = {
  1190. err: 0,
  1191. msg: '',
  1192. data: { ledgerAuditConst: auditConst.ledger, stageAuditConst: auditConst.stage },
  1193. };
  1194. responseData.data.category = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  1195. // 获取用户权限
  1196. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  1197. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  1198. const tenderList = await this.ctx.service.tender.getList('', userPermission);
  1199. for (const t of tenderList) {
  1200. if (t.ledger_status === auditConst.ledger.status.checked) {
  1201. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, false);
  1202. t.lastCheckedStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  1203. }
  1204. }
  1205. if (data.type === 'all') {
  1206. responseData.data.tenders = tenderList;
  1207. } else if (data.type === 'ledger-checked') {
  1208. responseData.data.tenders = tenderList.filter(x => {
  1209. return x.ledger_status === auditConst.ledger.status.checked;
  1210. });
  1211. } else if (data.type === 'stage') {
  1212. responseData.data.tenders = tenderList.filter(x => {
  1213. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastStage;
  1214. });
  1215. } else if (data.type === 'stage-checked') {
  1216. responseData.data.tenders = tenderList.filter(x => {
  1217. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastCheckedStage;
  1218. });
  1219. }
  1220. ctx.body = responseData;
  1221. } catch(err) {
  1222. this.log(err);
  1223. this.ajaxErrorBody(err, '获取标段列表错误');
  1224. }
  1225. }
  1226. async sumLoad(ctx) {
  1227. try {
  1228. const data = JSON.parse(ctx.request.body.data);
  1229. if (!data.lid || !data.type || !data.tenders) throw '数据错误';
  1230. switch (data.type) {
  1231. case 'ledger':
  1232. const refreshData = await this.ctx.service.ledger.sumLoad(data.lid, data.tenders);
  1233. ctx.body = {err: 0, msg: '', data: refreshData};
  1234. break;
  1235. case 'revise':
  1236. const reviseData = await this.ctx.service.reviseBills.sumLoad(data.lid, this.ctx.revise.id, data.tenders);
  1237. ctx.body = {err: 0, msg: '', data: reviseData};
  1238. break;
  1239. case 'stage':
  1240. const stageData = await this.ctx.service.stageBills.sumLoad(data.lid, data.tenders, data.cover);
  1241. stageData.import_change = { target: { import_lid: data.lid } };
  1242. stageData.import_change.data = await this.ctx.service.stageImportChange.getLeafXmjStageImportData(ctx.stage, data.lid);
  1243. await ctx.service.stage.updateCheckCalcFlag(ctx.stage, true);
  1244. await ctx.service.stage.updateCacheTime(ctx.stage.id);
  1245. ctx.body = {err: 0, msg: '', data: stageData};
  1246. break;
  1247. default:
  1248. throw '数据错误';
  1249. }
  1250. } catch (err) {
  1251. this.log(err);
  1252. ctx.ajaxErrorBody(err, '导入数据失败');
  1253. }
  1254. }
  1255. /**
  1256. * 地图数据设置(Ajax)
  1257. *
  1258. * @param ctx
  1259. * @return {Promise<void>}
  1260. */
  1261. async saveMap(ctx) {
  1262. try {
  1263. const data = JSON.parse(ctx.request.body.data);
  1264. if (!data) {
  1265. throw '提交数据错误';
  1266. }
  1267. // 判断修改权限
  1268. if (ctx.session.sessionUser.is_admin === 0) {
  1269. throw '你没有权限修改概况设置';
  1270. }
  1271. let info = '';
  1272. let result = '';
  1273. switch (data.type) {
  1274. case 'select-map':
  1275. result = await ctx.service.tender.saveTenderData(ctx.tender.id, { had_map: data.value });
  1276. if (!result) {
  1277. throw '修改概况设置失败';
  1278. }
  1279. info = result;
  1280. break;
  1281. case 'add-map':
  1282. result = await ctx.service.tenderMap.addMap(ctx.tender.id, data.name);
  1283. info = await ctx.service.tenderMap.getDataById(result.insertId);
  1284. break;
  1285. case 'del-map':
  1286. await ctx.service.tenderMap.deleteById(data.id);
  1287. break;
  1288. case 'save-map':
  1289. await ctx.service.tenderMap.saveMap(data.mapData);
  1290. break;
  1291. default:break;
  1292. }
  1293. ctx.body = { err: 0, msg: '', data: info };
  1294. } catch (err) {
  1295. this.log(err);
  1296. ctx.body = this.ajaxErrorBody(err, '保存概况设置失败');
  1297. }
  1298. }
  1299. /**
  1300. * 上传静态图
  1301. *
  1302. * @param {object} ctx - egg全局变量
  1303. * @return {void}
  1304. */
  1305. async uploadMap(ctx) {
  1306. const responseData = {
  1307. err: 0, msg: '', data: null,
  1308. };
  1309. try {
  1310. const stream = await ctx.getFileStream();
  1311. const create_time = Date.parse(new Date()) / 1000;
  1312. const fileInfo = path.parse(stream.filename);
  1313. const filepath = `app/public/upload/${ctx.tender.id}/map/${create_time}${fileInfo.ext}`;
  1314. await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);
  1315. await sendToWormhole(stream);
  1316. const tenderInfo = await ctx.service.tender.getDataById(ctx.tender.id);
  1317. const oldMapPic = tenderInfo && tenderInfo.map_pic ? tenderInfo.map_pic : null;
  1318. const result = await ctx.service.tender.update({ map_pic: filepath }, { id: ctx.tender.id });
  1319. if (result) {
  1320. // 移除旧的map_pic oss
  1321. if (oldMapPic) {
  1322. await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + oldMapPic);
  1323. }
  1324. responseData.data = { map_pic: filepath };
  1325. } else {
  1326. throw '添加数据库失败';
  1327. }
  1328. } catch (err) {
  1329. this.log(err);
  1330. responseData.err = 1;
  1331. responseData.msg = err;
  1332. }
  1333. ctx.body = responseData;
  1334. }
  1335. async loadData(ctx) {
  1336. try {
  1337. const data = JSON.parse(ctx.request.body.data);
  1338. const filter = data.filter.split(';');
  1339. const responseData = { err: 0, msg: '', data: {} };
  1340. const where = { tid: this.ctx.tender.id };
  1341. for (const f of filter) {
  1342. switch (f) {
  1343. case 'change':
  1344. where.valid = 1;
  1345. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.flow.status.checked;
  1346. responseData.data[f] = await ctx.service.change.getAllDataByCondition({
  1347. columns: [ 'cid', 'code', 'name', 'selected' ],
  1348. where,
  1349. orders: [['in_time', 'desc']],
  1350. });
  1351. break;
  1352. case 'change_project':
  1353. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeProject.status.checked;
  1354. responseData.data[f] = await ctx.service.changeProject.getAllDataByCondition({
  1355. columns: [ 'id', 'code', 'name', 'selected' ],
  1356. where,
  1357. orders: [['in_time', 'desc']],
  1358. });
  1359. break;
  1360. case 'change_apply':
  1361. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeApply.status.checked;
  1362. responseData.data[f] = await ctx.service.changeApply.getAllDataByCondition({
  1363. columns: [ 'id', 'code', 'name', 'selected' ],
  1364. where,
  1365. orders: [['in_time', 'desc']],
  1366. });
  1367. break;
  1368. case 'change_plan':
  1369. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeApply.status.checked;
  1370. responseData.data[f] = await ctx.service.changePlan.getAllDataByCondition({
  1371. columns: [ 'id', 'code', 'name', 'selected' ],
  1372. where,
  1373. orders: [['in_time', 'desc']],
  1374. });
  1375. break;
  1376. case 'advance':
  1377. const advance = await ctx.service.advance.getAllDataByCondition({
  1378. columns: [ 'id', 'order', 'status', 'selected', 'type' ],
  1379. where,
  1380. orders: [['type', 'asc'], ['create_time', 'desc']],
  1381. });
  1382. advance.forEach(x => {
  1383. x.orderStr = `第${x.order}期`;
  1384. x.statusStr = auditConst.advance.statusString[x.status];
  1385. x.typeStr = advanceConst.typeColMap[x.type].text;
  1386. });
  1387. responseData.data[f] = advance;
  1388. break;
  1389. case 'pm_deal':
  1390. if (accountPermission.PermissionCheck.viewPmData(this.ctx.session.sessionUser.permission)) {
  1391. const selects = await this.ctx.service.project.getPmDealCache(this.ctx.session.sessionProject.id);
  1392. const pm = require('../lib/pm');
  1393. const tenders = await pm.dealCatagory(ctx, this.ctx.session.sessionProject.code);
  1394. tenders.forEach(x => {
  1395. x.selected = selects.indexOf(x.bidsectionId + '') >= 0;
  1396. });
  1397. responseData.data[f] = tenders;
  1398. } else {
  1399. throw '您无权查看该数据';
  1400. }
  1401. break;
  1402. default:
  1403. throw '未知数据类型';
  1404. }
  1405. }
  1406. ctx.body = responseData;
  1407. } catch (err) {
  1408. ctx.log(err);
  1409. ctx.ajaxErrorBody(err, '加载数据错误');
  1410. }
  1411. }
  1412. async saveRelaData(ctx) {
  1413. try {
  1414. const data = JSON.parse(ctx.request.body.data);
  1415. const responseData = { err: 0, msg: '', data: {} };
  1416. if (data.change) await this.ctx.service.change.updateChangeSelect(data.change);
  1417. if (data.change_apply) await this.ctx.service.changeApply.defaultUpdateRows(data.change_apply);
  1418. if (data.change_project) await this.ctx.service.changeProject.defaultUpdateRows(data.change_project);
  1419. if (data.change_plan) await this.ctx.service.changePlan.defaultUpdateRows(data.change_plan);
  1420. if (data.advance) await this.ctx.service.advance.defaultUpdateRows(data.advance);
  1421. if (data.pm_deal) await this.ctx.service.project.setPmDealCache(this.ctx.session.sessionProject.id, data.pm_deal);
  1422. ctx.body = responseData;
  1423. } catch (err) {
  1424. ctx.log(err);
  1425. ctx.ajaxErrorBody(err, '保存数据错误');
  1426. }
  1427. }
  1428. }
  1429. return TenderController;
  1430. };