wap_controller.js 36 KB

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