wap_controller.js 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841
  1. 'use strict';
  2. /**
  3. * 登录页面控制器
  4. *
  5. * @author CaiAoLin
  6. * @date 2017/11/15
  7. * @version
  8. */
  9. const URL = require('url');
  10. const maintainConst = require('../const/maintain');
  11. const auditConst = require('../const/audit');
  12. const changeConst = require('../const/change');
  13. const advanceConst = require('../const/advance');
  14. const fs = require('fs');
  15. const path = require('path');
  16. const sendToWormhole = require('stream-wormhole');
  17. const moment = require('moment');
  18. const auditType = require('../const/audit').auditType;
  19. module.exports = app => {
  20. class WapController extends app.BaseController {
  21. /**
  22. * 登录页面
  23. *
  24. * @param {Object} ctx - egg全局页面
  25. * @return {void}
  26. */
  27. async index(ctx) {
  28. if (!ctx.helper.isMobile(ctx.request.header['user-agent'])) {
  29. ctx.redirect('/');
  30. return;
  31. }
  32. const errorMessage = ctx.session.loginError;
  33. // 显示完删除
  34. ctx.session.loginError = null;
  35. // 获取系统维护信息
  36. const maintainData = await ctx.service.maintain.getDataById(1);
  37. if (!ctx.app.config.is_debug) {
  38. await ctx.service.maintain.syncMaintainData();
  39. }
  40. let projectData = '';
  41. if (ctx.session.wapTenderID) {
  42. const tenderData = await ctx.service.tender.getDataById(ctx.session.wapTenderID);
  43. if (tenderData) projectData = await ctx.service.project.getDataById(tenderData.project_id);
  44. }
  45. const renderData = {
  46. maintainData,
  47. maintainConst,
  48. errorMessage,
  49. projectData,
  50. };
  51. await ctx.render('wap/login.ejs', renderData);
  52. }
  53. /**
  54. * 登录操作
  55. *
  56. * @param {Object} ctx - egg全局变量
  57. * @return {void}
  58. */
  59. async login(ctx) {
  60. let loginType = ctx.request.body.type;
  61. try {
  62. loginType = parseInt(loginType);
  63. const result = await ctx.service.projectAccount.accountLogin(ctx.request.body, loginType);
  64. if (!result) {
  65. throw '用户名或密码错误';
  66. }
  67. if (result === 2) {
  68. // 判断是否有设置停用提示,有则展示
  69. const msg = await ctx.service.projectStopmsg.getMsg(ctx.session.sessionProject.id);
  70. throw msg;
  71. }
  72. // 调用 rotateCsrfSecret 刷新用户的 CSRF token
  73. ctx.rotateCsrfSecret();
  74. // 判断是否已经有对应用户信息,没有则跳转初始化页面
  75. const needBoot = await ctx.service.customer.isNeedBoot(ctx.request.body);
  76. const url = needBoot ? '/boot' : '/wap/dashboard';
  77. const query = URL.parse(ctx.request.header.referer, true).query;
  78. let referer = '';
  79. if (ctx.session.wapTenderID) {
  80. referer = query.referer ? query.referer + '#shenpi' : url;
  81. ctx.session.wapTenderID = null;
  82. }
  83. ctx.redirect(referer ? referer : url);
  84. } catch (error) {
  85. this.log(error);
  86. ctx.session.loginError = error;
  87. ctx.redirect('/wap');
  88. }
  89. }
  90. /**
  91. * 退出登录
  92. *
  93. * @param {Object} ctx - egg全局变量
  94. * @return {void}
  95. */
  96. async logout(ctx) {
  97. // 删除session并跳转
  98. ctx.session = null;
  99. ctx.redirect('/wap');
  100. }
  101. /**
  102. * 待办页
  103. *
  104. * @param {Object} ctx - egg全局变量
  105. * @return {void}
  106. */
  107. async dashboard(ctx) {
  108. // 获取待审批的期
  109. const auditStages = await ctx.service.stageAudit.getAuditStageByWap(ctx.session.sessionUser.accountId);
  110. for (const audit of auditStages) {
  111. if (audit.status !== auditConst.stage.status.uncheck) {
  112. const isLastage = await ctx.service.stage.isLastStage(audit.tid, audit.id);
  113. if (isLastage) await this.ctx.service.stage.checkStageGatherData(audit);
  114. }
  115. audit.gather_tp = ctx.helper.sum([audit.contract_tp, audit.qc_tp, audit.pc_tp]);
  116. audit.end_contract_tp = ctx.helper.sum([audit.contract_tp, audit.pre_contract_tp, audit.contract_pc_tp]);
  117. audit.end_qc_tp = ctx.helper.sum([audit.qc_tp, audit.pre_qc_tp, audit.qc_pc_tp]);
  118. audit.pre_gather_tp = ctx.helper.add(audit.pre_contract_tp, audit.pre_qc_tp);
  119. audit.end_gather_tp = ctx.helper.add(audit.gather_tp, audit.pre_gather_tp);
  120. }
  121. // 获取待审批的变更期
  122. const auditChanges = await ctx.service.changeAudit.getAuditChangeByWap(ctx.session.sessionUser.accountId);
  123. // 获取待审批的变更方案
  124. let auditChangePlans = [];
  125. if (ctx.session.sessionProject.page_show.openChangePlan) {
  126. auditChangePlans = await ctx.service.changePlanAudit.getAuditChangePlanByWap(ctx.session.sessionUser.accountId);
  127. }
  128. // 获取待审批的台账修订
  129. const auditRevise = await ctx.service.reviseAudit.getAuditReviseByWap(ctx.session.sessionUser.accountId);
  130. for (const revise of auditRevise) {
  131. const yb_audit = await ctx.service.projectAccount.getAccountInfoById(revise.audit_id);
  132. revise.yb_name = yb_audit.name;
  133. }
  134. // 获取待审批的预付款
  135. const auditAdvance = await ctx.service.advanceAudit.getAuditAdvanceByWap(ctx.session.sessionUser.accountId);
  136. const renderData = {
  137. auditStages,
  138. auditChanges,
  139. auditChangePlans,
  140. auditRevise,
  141. auditAdvance,
  142. changeConst,
  143. advanceConst,
  144. tpUnit: 2,
  145. };
  146. await ctx.render('wap/dashboard.ejs', renderData);
  147. }
  148. /**
  149. * 标段列表页
  150. *
  151. * @param {Object} ctx - egg全局变量
  152. * @return {void}
  153. */
  154. async list(ctx) {
  155. try {
  156. // 获取用户新建标段权利
  157. const accountInfo = await this.ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);
  158. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  159. const tenderList = await this.ctx.service.tender.getBuildList('', userPermission);
  160. for (const t of tenderList) {
  161. await this.ctx.service.tenderCache.loadTenderCache(t, this.ctx.session.sessionUser.accountId);
  162. }
  163. const categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  164. const valuations = await this.ctx.service.valuation.getProjectValidValuation(this.ctx.session.sessionProject.id);
  165. const renderData = {
  166. tenderList,
  167. categoryData,
  168. auditConst,
  169. userPermission,
  170. valuations,
  171. uid: this.ctx.session.sessionUser.accountId,
  172. pid: this.ctx.session.sessionProject.id,
  173. };
  174. await ctx.render('wap/list.ejs', renderData);
  175. } catch (err) {
  176. this.log(err);
  177. this.ctx.redirect('/wap/dashboard');
  178. }
  179. }
  180. /**
  181. * 标段详细页
  182. *
  183. * @param {Object} ctx - egg全局变量
  184. * @return {void}
  185. */
  186. async tender(ctx) {
  187. try {
  188. const tender = ctx.tender.data;
  189. if (tender.user_id === this.ctx.session.sessionUser.accountId && (
  190. tender.ledger_status === auditConst.ledger.status.checkNo || tender.ledger_status === auditConst.ledger.status.uncheck)) {
  191. const sum = await this.ctx.service.ledger.addUp({tender_id: tender.id/*, is_leaf: true*/});
  192. tender.total_price = sum.total_price;
  193. tender.deal_tp = sum.deal_tp;
  194. }
  195. const [change_tp, change_p_tp, change_n_tp] = await ctx.service.change.getChangeTp(ctx.tender.id);
  196. tender.change_tp = change_tp;
  197. tender.change_p_tp = change_p_tp;
  198. tender.change_n_tp = change_n_tp;
  199. const stages = await ctx.service.stage.getValidStages(ctx.tender.id);
  200. const lastStage = stages.length > 0 ? stages[0] : null; //await ctx.service.stage.getLastestStage(ctx.tender.id);
  201. if (lastStage) {
  202. await this.ctx.service.stage.checkStageGatherData(lastStage);
  203. tender.gather_tp = ctx.helper.sum([lastStage.contract_tp, lastStage.qc_tp, lastStage.pc_tp]);
  204. tender.end_contract_tp = ctx.helper.sum([lastStage.contract_tp, lastStage.pre_contract_tp, lastStage.contract_pc_tp]);
  205. tender.end_qc_tp = ctx.helper.sum([lastStage.qc_tp, lastStage.pre_qc_tp, lastStage.qc_pc_tp]);
  206. tender.pre_gather_tp = ctx.helper.add(lastStage.pre_contract_tp, lastStage.pre_qc_tp);
  207. tender.end_gather_tp = ctx.helper.add(tender.gather_tp, tender.pre_gather_tp);
  208. tender.yf_tp = lastStage.yf_tp;
  209. tender.qc_ratio = ctx.helper.mul(ctx.helper.div(tender.end_qc_tp, ctx.tender.info.deal_param.contractPrice, 2), 100);
  210. tender.sum = ctx.helper.add(tender.total_price, tender.change_tp);
  211. tender.pre_ratio = ctx.helper.mul(ctx.helper.div(tender.pre_gather_tp, tender.sum, 2), 100);
  212. tender.cur_ratio = ctx.helper.mul(ctx.helper.div(tender.gather_tp, tender.sum, 2), 100);
  213. tender.other_tp = ctx.helper.sub(ctx.helper.sub(tender.sum, tender.pre_gather_tp), tender.gather_tp);
  214. tender.other_ratio = Math.max(0, 100 - tender.pre_ratio - tender.cur_ratio);
  215. }
  216. const monthProgress = [];
  217. for (const s of stages) {
  218. if (s.s_time) {
  219. let progress = monthProgress.find(function(x) {
  220. return x.month === s.s_time;
  221. });
  222. if (!progress) {
  223. progress = { month: s.s_time };
  224. monthProgress.push(progress);
  225. }
  226. progress.tp = ctx.helper.add(ctx.helper.add(progress.tp, s.contract_tp), s.qc_tp);
  227. }
  228. }
  229. monthProgress.sort(function(x, y) {
  230. return Date.parse(x.month) - Date.parse(y.month);
  231. });
  232. let sum = 0;
  233. for (const p of monthProgress) {
  234. p.ratio = ctx.helper.mul(ctx.helper.div(p.tp, tender.sum, 4), 100);
  235. sum = ctx.helper.add(sum, p.tp);
  236. p.end_tp = sum;
  237. p.end_ratio = ctx.helper.mul(ctx.helper.div(p.end_tp, tender.sum, 4), 100);
  238. }
  239. // 变更令列表
  240. const changes = await ctx.service.change.getListByStatus(tender.id, 0, 0);
  241. for (const c of changes) {
  242. c.showApprovalBtn = false;
  243. c.curAuditors = await ctx.service.changeAudit.getAuditorsByStatus(c.cid, c.status, c.times);
  244. if (c.status === auditConst.change.status.checkNoPre) {
  245. c.curAuditors2 = await ctx.service.stageAudit.getAuditorsByStatus(c.cid, auditConst.change.status.checking, c.times);
  246. const curAudit = c.curAuditors2.find(function(x) {
  247. return x.uid === ctx.session.sessionUser.accountId;
  248. });
  249. if (curAudit && curAudit.status === auditConst.change.status.checking) {
  250. c.showApprovalBtn = true;
  251. }
  252. } else {
  253. const curAudit = c.curAuditors.find(function(x) {
  254. return x.uid === ctx.session.sessionUser.accountId;
  255. });
  256. if (curAudit && curAudit.status === auditConst.change.status.checking) {
  257. c.showApprovalBtn = true;
  258. }
  259. }
  260. }
  261. // 变更令列表
  262. let changePlans = [];
  263. if (ctx.session.sessionProject.page_show.openChangePlan) {
  264. changePlans = await ctx.service.changePlan.getListByStatus(tender.id, 0, 0);
  265. for (const c of changePlans) {
  266. c.curAuditor = await ctx.service.changePlanAudit.getAuditorByStatus(c.id, c.status, c.times);
  267. }
  268. }
  269. // 台账修订列表
  270. const revises = await ctx.service.ledgerRevise.getReviseList(ctx.tender.id);
  271. for (const lr of revises) {
  272. if (lr.valid) {
  273. lr.curAuditor = await ctx.service.reviseAudit.getAuditorByStatus(lr.id, lr.status, lr.times);
  274. }
  275. }
  276. // 预付款期数获取
  277. const advanceList = [];
  278. for (const t of advanceConst.typeCol) {
  279. const advance = await ctx.service.advance.getLastestAdvance(ctx.tender.id, t.type, true);
  280. advanceList.push(advance);
  281. }
  282. const renderData = {
  283. tender,
  284. stages,
  285. changes,
  286. changePlans,
  287. revises,
  288. advanceList,
  289. auditConst: auditConst.stage,
  290. auditChangeConst: auditConst.change,
  291. auditChangePlanConst: auditConst.changePlan,
  292. auditReviseConst: auditConst.revise,
  293. changeConst,
  294. advanceConst,
  295. tpUnit: ctx.tender.info.decimal.tp,
  296. monthProgress,
  297. stagesEcharts: JSON.parse(JSON.stringify(stages)).reverse(),
  298. auditType: auditConst.auditType,
  299. };
  300. if (stages.length > 0) {
  301. for (const s of stages) {
  302. // s.curAuditor = null;
  303. // 根据期状态返回展示用户
  304. s.curAuditors = await ctx.service.stageAudit.getAuditorsByStatus(s.id, s.status, s.times);
  305. if (s.status === auditConst.stage.status.checkNoPre) {
  306. s.curAuditors2 = await ctx.service.stageAudit.getAuditorsByStatus(s.id, auditConst.stage.status.checking);
  307. }
  308. }
  309. renderData.stage = stages[0];
  310. const times = renderData.stage.status === auditConst.stage.status.checkNo ? renderData.stage.times - 1 : renderData.stage.times;
  311. renderData.stage.user = await ctx.service.projectAccount.getAccountInfoById(renderData.stage.user_id);
  312. renderData.stage.auditors = await ctx.service.stageAudit.getAuditors(renderData.stage.id, times);
  313. // 获取审批流程中左边列表
  314. renderData.stage.auditors2 = await ctx.service.stageAudit.getAuditGroupByList(renderData.stage.id, times);
  315. }
  316. await ctx.render('wap/tender.ejs', renderData);
  317. } catch (err) {
  318. this.log(err);
  319. ctx.redirect('/wap/list');
  320. }
  321. }
  322. /**
  323. * 期审批详细页
  324. *
  325. * @param {Object} ctx - egg全局变量
  326. * @return {void}
  327. */
  328. async stage(ctx) {
  329. try {
  330. const tender = ctx.tender.data;
  331. const stages = await ctx.service.stage.getValidStages(ctx.tender.id);
  332. const lastStage = stages.length > 0 ? stages[0] : null;
  333. if (lastStage) {
  334. await this.ctx.service.stage.checkStageGatherData(lastStage);
  335. }
  336. const stage = lastStage;
  337. const renderData = {
  338. tender,
  339. stage,
  340. auditConst: auditConst.stage,
  341. };
  342. const times = renderData.stage.status === auditConst.stage.status.checkNo ? renderData.stage.times - 1 : renderData.stage.times;
  343. renderData.stage.user = await ctx.service.projectAccount.getAccountInfoById(renderData.stage.user_id);
  344. renderData.stage.auditors = await ctx.service.stageAudit.getAuditors(renderData.stage.id, times);
  345. // 获取审批流程中左边列表
  346. renderData.stage.auditors2 = await ctx.service.stageAudit.getAuditGroupByList(renderData.stage.id, times);
  347. renderData.stage.lastAuditors = await ctx.service.stageAudit.getAuditorsByStatus(stage.id, stage.status, stage.times);
  348. await ctx.render('wap/shenpi_stage.ejs', renderData);
  349. } catch (err) {
  350. this.log(err);
  351. ctx.redirect('/wap/list');
  352. }
  353. }
  354. /**
  355. * 变更审批详细页
  356. *
  357. * @param {Object} ctx - egg全局变量
  358. * @return {void}
  359. */
  360. async change(ctx) {
  361. try {
  362. const tender = ctx.tender.data;
  363. const change = ctx.change;
  364. const times = change.status !== auditConst.change.status.checkNo && change.status !== auditConst.change.status.revise ? change.times : change.times - 1;
  365. const auditList = await ctx.service.changeAudit.getListOrderByTimes(change.cid, times);
  366. const auditGroupList = await ctx.service.changeAudit.getListGroupByTimes(change.cid, times);
  367. await ctx.service.change.loadChangeAuditViewData(ctx.change);
  368. const renderData = {
  369. moment,
  370. tender,
  371. change,
  372. auditList,
  373. auditGroupList,
  374. auditConst: auditConst.change,
  375. changeConst,
  376. tpUnit: ctx.tender.info.decimal.tp,
  377. auditType,
  378. };
  379. await ctx.render('wap/shenpi_change.ejs', renderData);
  380. } catch (err) {
  381. this.log(err);
  382. ctx.redirect('/wap/list');
  383. }
  384. }
  385. /**
  386. * 变更方案审批详细页
  387. *
  388. * @param {Object} ctx - egg全局变量
  389. * @return {void}
  390. */
  391. async changePlan(ctx) {
  392. try {
  393. const tender = ctx.tender.data;
  394. if (!ctx.session.sessionProject.page_show.openChangePlan) {
  395. throw '该功能已关闭';
  396. }
  397. const change = await ctx.service.changePlan.getDataById(ctx.params.cpid);
  398. const renderData = {
  399. tender,
  400. change,
  401. auditConst: auditConst.changePlan,
  402. changeConst,
  403. tpUnit: ctx.tender.info.decimal.tp,
  404. };
  405. const times = change.status !== auditConst.changePlan.status.checkNo ? change.times : change.times - 1;
  406. renderData.change.user = await ctx.service.projectAccount.getAccountInfoById(change.uid);
  407. renderData.change.auditors = await ctx.service.changePlanAudit.getAuditors(change.id, times);
  408. // 获取审批流程中左边列表
  409. renderData.change.auditors2 = await ctx.service.changePlanAudit.getAuditGroupByList(change.id, times);
  410. await ctx.render('wap/shenpi_change_plan.ejs', renderData);
  411. } catch (err) {
  412. this.log(err);
  413. ctx.redirect('/wap/list');
  414. }
  415. }
  416. /**
  417. * 修订审批详细页
  418. *
  419. * @param {Object} ctx - egg全局变量
  420. * @return {void}
  421. */
  422. async revise(ctx) {
  423. try {
  424. const tender = ctx.tender.data;
  425. const revise = await ctx.service.ledgerRevise.getDataByCondition({ id: ctx.params.rid });
  426. revise.user = await ctx.service.projectAccount.getAccountInfoById(revise.uid);
  427. const times = revise.status === auditConst.revise.status.checkNo ? revise.times - 1 : revise.times;
  428. revise.curAuditor = await ctx.service.reviseAudit.getAuditorByStatus(revise.id, revise.status, times);
  429. revise.auditors = await ctx.service.reviseAudit.getAuditors(revise.id, times);
  430. console.log(times, revise.auditors);
  431. const renderData = {
  432. tender,
  433. revise,
  434. auditConst: auditConst.revise,
  435. };
  436. await ctx.render('wap/shenpi_revise.ejs', renderData);
  437. } catch (err) {
  438. this.log(err);
  439. ctx.redirect('/wap/list');
  440. }
  441. }
  442. /**
  443. * 预付款列表页
  444. *
  445. * @param {Object} ctx - egg全局变量
  446. * @return {void}
  447. */
  448. async advance(ctx) {
  449. try {
  450. const tender = ctx.tender.data;
  451. const { decimal } = ctx.tender.info;
  452. this.decimal = decimal.pay ? decimal.payTp : decimal.tp;
  453. const advancePayTotalList = [];
  454. const advanceList = [];
  455. for (const t of advanceConst.typeCol) {
  456. const advancePayTotal = ctx.tender.info.deal_param[t.key + 'Advance'];
  457. advancePayTotalList.push(advancePayTotal);
  458. const advances = await ctx.service.advance.getAdvanceList(ctx.tender.id, t.type, this.decimal, advancePayTotal);
  459. advanceList.push(advances);
  460. }
  461. const renderData = {
  462. tender,
  463. advancePayTotalList,
  464. advanceList,
  465. auditConst: auditConst.advance,
  466. advanceConst,
  467. };
  468. await ctx.render('wap/shenpi_advance.ejs', renderData);
  469. } catch (err) {
  470. this.log(err);
  471. ctx.redirect('/wap/list');
  472. }
  473. }
  474. /**
  475. * 预付款详情页 GET
  476. * @param {Object} ctx 全局上下文
  477. */
  478. async advanceDetail(ctx) {
  479. try {
  480. const tender = ctx.tender.data;
  481. ctx.advance.advancePayTotal = ctx.tender.info.deal_param[ advanceConst.typeCol[ctx.advance.type].key + 'Advance'];
  482. const times = ctx.advance.status === auditConst.advance.status.checkNo ? ctx.advance.times - 1 : ctx.advance.times;
  483. if (ctx.advance.status === auditConst.advance.status.checkNo) {
  484. ctx.advance.curAuditor = await ctx.service.advanceAudit.getAuditorByStatus(ctx.advance.id, ctx.advance.status, times);
  485. ctx.advance.auditors = await ctx.service.advanceAudit.getAuditors(ctx.advance.id, times);
  486. }
  487. // 获取审批流程中左边列表
  488. ctx.advance.auditors2 = await ctx.service.advanceAudit.getAuditGroupByList(ctx.advance.id, times);
  489. const renderData = {
  490. tender,
  491. advance: ctx.advance,
  492. auditConst: auditConst.advance,
  493. advanceConst,
  494. };
  495. await ctx.render('wap/shenpi_advance_detail.ejs', renderData);
  496. } catch (error) {
  497. this.log(error);
  498. ctx.redirect('/wap/tender/' + ctx.tender.id + '/advance');
  499. }
  500. }
  501. /**
  502. * 变更令审批
  503. * @param {Object} ctx - egg全局变量
  504. * @return {void}
  505. */
  506. async changeApproval(ctx) {
  507. try {
  508. const cid = ctx.request.body.change_id;
  509. const changeData = await ctx.service.change.getDataByCondition({ cid });
  510. if (!changeData) {
  511. throw '变更令数据错误';
  512. }
  513. // 判断是否到你审批,如果不是则无法审批
  514. const curAuditor = await ctx.service.changeAudit.getCurAuditors(changeData.cid, changeData.times);
  515. if (!curAuditor || (curAuditor && ctx.helper._.findIndex(curAuditor, { uid: ctx.session.sessionUser.accountId }) === -1)) {
  516. throw '该变更令当前您无权操作';
  517. }
  518. const readySettle = await ctx.service.settle.getReadySettle(changeData.tid);
  519. if (readySettle && readySettle.settle_order !== ctx.tender.data.settle_order) {
  520. throw '结算数据发生变化,请刷新页面再提交';
  521. }
  522. const status = parseInt(ctx.request.body.status);
  523. const pid = this.ctx.session.sessionProject.id;
  524. let result = false;
  525. switch (status) {
  526. case 3:// 审批通过
  527. result = await ctx.service.change.approvalSuccess(pid, ctx.request.body, changeData);
  528. break;
  529. // case 4:// 审批终止
  530. // result = await ctx.service.change.approvalStop(ctx.request.body);
  531. // break;
  532. case 5:// 审批退回到原报人
  533. result = await ctx.service.change.approvalCheckNo(pid, ctx.request.body, changeData);
  534. break;
  535. case 6:// 审批退回到上一个审批人
  536. result = await ctx.service.change.approvalCheckNoPre(pid, ctx.request.body, changeData);
  537. break;
  538. default:break;
  539. }
  540. if (!result) {
  541. throw '审批失败';
  542. }
  543. ctx.redirect(ctx.request.header.referer);
  544. } catch (err) {
  545. console.log(err);
  546. ctx.redirect(ctx.request.header.referer);
  547. }
  548. }
  549. /**
  550. * 收方单附件上传页
  551. * @param {Object} ctx - egg全局变量
  552. * @return {void}
  553. */
  554. async shoufangUpload(ctx) {
  555. try {
  556. const tid = parseInt(ctx.query.tid) || 0;
  557. const order = parseInt(ctx.query.order) || 0;
  558. const sfid = parseInt(ctx.query.sfid) || 0;
  559. if (!tid || !order || !sfid) {
  560. throw '参数有误';
  561. }
  562. const tender = await ctx.service.tender.getDataById(tid);
  563. if (!tender) {
  564. throw '该标段不存在';
  565. }
  566. const sfInfo = await ctx.service.stageShoufang.getDataByCondition({ id: sfid, tid, order });
  567. if (!sfInfo) {
  568. throw '该收方单不存在';
  569. }
  570. // 查找对应的台账或计量单元名称
  571. const ledger = await ctx.service.ledger.getData(tid);
  572. const ledgerInfo = ctx.helper._.find(ledger, { id: sfInfo.lid });
  573. let name = ledgerInfo.b_code;
  574. if (sfInfo.pid) {
  575. const pos = await ctx.service.pos.getPosData({ tid });
  576. const posInfo = ctx.helper._.find(pos, { lid: sfInfo.lid, id: sfInfo.pid });
  577. name += ' / ' + posInfo.name;
  578. }
  579. const renderData = {
  580. tender,
  581. order,
  582. name,
  583. sfid,
  584. whiteList: this.ctx.app.config.multipart.whitelist,
  585. };
  586. await ctx.render('wap/shoufangupload.ejs', renderData);
  587. } catch (error) {
  588. this.log(error);
  589. ctx.redirect('/wap/login');
  590. }
  591. }
  592. /**
  593. * 上传附件
  594. * @param {Object} ctx - egg全局变量
  595. * @return {void}
  596. */
  597. async shoufangUpFile(ctx) {
  598. const responseData = {
  599. err: 0,
  600. msg: '',
  601. data: [],
  602. };
  603. let stream;
  604. try {
  605. const parts = ctx.multipart({ autoFields: true });
  606. const files = [];
  607. let index = 0;
  608. while ((stream = await parts()) !== undefined) {
  609. // 判断是否存在
  610. const sfInfo = await ctx.service.stageShoufang.getDataById(parts.field.sfid);
  611. if (!sfInfo) {
  612. throw '该清单 / 计量单元下不存在收方单';
  613. }
  614. // 判断用户是否选择上传文件
  615. if (!stream.filename) {
  616. throw '请选择上传的文件!';
  617. }
  618. // const dirName = 'app/public/upload/stage/' + moment().format('YYYYMMDD');
  619. // 判断文件夹是否存在,不存在则直接创建文件夹
  620. // if (!fs.existsSync(path.join(this.app.baseDir, dirName))) {
  621. // await fs.mkdirSync(path.join(this.app.baseDir, dirName));
  622. // }
  623. const fileInfo = path.parse(stream.filename);
  624. const now_time = new Date();
  625. const create_time = Date.parse(now_time) / 1000;
  626. const filepath = `app/public/upload/${sfInfo.tid}/stage/shoufang/fujian_${create_time + index.toString() + fileInfo.ext}`;
  627. // await ctx.helper.saveStreamFile(stream, path.resolve(this.app.baseDir, filepath));
  628. await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);
  629. // console.log(await fs.existsSync(path.resolve(this.app.baseDir, 'app', filepath)));
  630. // const fileInfo = path.parse(stream.filename);
  631. // const fileName = 'stage' + create_time + '_' + index + fileInfo.ext;
  632. // 保存文件
  633. // await ctx.helper.saveStreamFile(stream, path.resolve(this.app.baseDir, dirName, fileName));
  634. if (stream) {
  635. await sendToWormhole(stream);
  636. }
  637. // 保存数据到att表
  638. const fileData = {
  639. tid: sfInfo.tid,
  640. sid: sfInfo.sid,
  641. sfid: sfInfo.id,
  642. in_time: now_time,
  643. filename: fileInfo.name,
  644. fileext: fileInfo.ext,
  645. filesize: Array.isArray(parts.field.size) ? parts.field.size[index] : parts.field.size,
  646. filepath,
  647. };
  648. // if (ctx.reUploadPermission) {
  649. // fileData.re_upload = 1;
  650. // }
  651. const result = await ctx.service.stageShoufangAtt.save(fileData);
  652. if (!result) {
  653. throw '导入数据库保存失败';
  654. }
  655. const attData = await ctx.service.stageShoufangAtt.getDataByFid(result.insertId);
  656. attData.in_time = moment(create_time * 1000).format('YYYY-MM-DD');
  657. files.length !== 0 ? files.unshift(attData) : files.push(attData);
  658. ++index;
  659. }
  660. responseData.data = files;
  661. } catch (err) {
  662. this.log(err);
  663. // 失败需要消耗掉stream 以防卡死
  664. if (stream) {
  665. await sendToWormhole(stream);
  666. }
  667. this.setMessage(err.toString(), this.messageType.ERROR);
  668. responseData.err = 1;
  669. responseData.msg = err.toString();
  670. }
  671. ctx.body = responseData;
  672. }
  673. /**
  674. * 删除附件
  675. * @param {Object} ctx - egg全局变量
  676. * @return {void}
  677. */
  678. async shoufangDeleteFile(ctx) {
  679. const responseData = {
  680. err: 0,
  681. msg: '',
  682. data: '',
  683. };
  684. try {
  685. const data = JSON.parse(ctx.request.body.data);
  686. const fileInfo = await ctx.service.stageShoufangAtt.getDataById(data.id);
  687. if (!fileInfo || !Object.keys(fileInfo).length) {
  688. throw '该文件不存在';
  689. }
  690. if (fileInfo !== undefined && fileInfo !== '') {
  691. // 先删除文件
  692. // await fs.unlinkSync(path.join(this.app.baseDir, fileInfo.filepath));
  693. await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + fileInfo.filepath);
  694. // 再删除数据库
  695. await ctx.service.stageShoufangAtt.deleteById(data.id);
  696. responseData.data = '';
  697. } else {
  698. throw '不存在该文件';
  699. }
  700. } catch (err) {
  701. responseData.err = 1;
  702. responseData.msg = err;
  703. }
  704. ctx.body = responseData;
  705. }
  706. /**
  707. * 编辑附件
  708. * @param {Object} ctx - egg全局变量
  709. * @return {void}
  710. */
  711. async shoufangEditFile(ctx) {
  712. const responseData = {
  713. err: 0,
  714. msg: '',
  715. data: '',
  716. };
  717. try {
  718. const data = JSON.parse(ctx.request.body.data);
  719. const fileInfo = await ctx.service.stageShoufangAtt.getDataById(data.id);
  720. if (!fileInfo || !Object.keys(fileInfo).length) {
  721. throw '该文件不存在';
  722. }
  723. await ctx.service.stageShoufangAtt.edit(data);
  724. } catch (err) {
  725. responseData.err = 1;
  726. responseData.msg = err;
  727. }
  728. ctx.body = responseData;
  729. }
  730. /**
  731. * 下载附件
  732. * @param {Object} ctx - egg全局变量
  733. * @return {void}
  734. */
  735. async shoufangDownloadFile(ctx) {
  736. const id = ctx.params.fid;
  737. if (id) {
  738. try {
  739. const fileInfo = await ctx.service.stageShoufangAtt.getDataById(id);
  740. if (fileInfo !== undefined && fileInfo !== '') {
  741. // const fileName = path.join(this.app.baseDir, fileInfo.filepath);
  742. // 解决中文无法下载问题
  743. const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();
  744. let disposition = '';
  745. if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {
  746. disposition = 'attachment; filename=' + encodeURIComponent(fileInfo.filename + fileInfo.fileext);
  747. } else if (userAgent.indexOf('firefox') >= 0) {
  748. disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(fileInfo.filename + fileInfo.fileext) + '"';
  749. } else {
  750. /* safari等其他非主流浏览器只能自求多福了 */
  751. disposition = 'attachment; filename=' + new Buffer(fileInfo.filename + fileInfo.fileext).toString('binary');
  752. }
  753. ctx.response.set({
  754. 'Content-Type': 'application/octet-stream',
  755. 'Content-Disposition': disposition,
  756. 'Content-Length': fileInfo.filesize,
  757. });
  758. // ctx.body = await fs.createReadStream(fileName);
  759. ctx.body = await ctx.helper.ossFileGet(fileInfo.filepath);
  760. } else {
  761. throw '不存在该文件';
  762. }
  763. } catch (err) {
  764. this.log(err);
  765. this.setMessage(err.toString(), this.messageType.ERROR);
  766. }
  767. }
  768. }
  769. async msg(ctx) {
  770. try {
  771. const msgId = parseInt(ctx.params.id) || 0;
  772. if (!msgId) {
  773. throw '参数有误';
  774. }
  775. const msgInfo = await ctx.service.message.getDataById(msgId);
  776. if (!msgInfo) {
  777. throw '项目通知不存在';
  778. }
  779. if (msgInfo && msgInfo.type === 1 && msgInfo.project_id !== ctx.session.sessionProject.id) {
  780. throw '非该项目通知无权查看';
  781. }
  782. const renderData = {
  783. msgInfo,
  784. moment,
  785. };
  786. await ctx.render('wap/msg.ejs', renderData);
  787. } catch (error) {
  788. console.log(error);
  789. this.log(error);
  790. ctx.redirect('/wap/dashboard');
  791. }
  792. }
  793. }
  794. return WapController;
  795. };