payment_controller.js 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806
  1. 'use strict';
  2. const accountGroup = require('../const/account_group').group;
  3. const JV = require('../reports/rpt_component/jpc_value_define');
  4. const shenpiConst = require('../const/shenpi');
  5. const auditConst = require('../const/audit').stage;
  6. const paymentConst = require('../const/payment');
  7. module.exports = app => {
  8. class PaymentController extends app.BaseController {
  9. /**
  10. * 构造函数
  11. *
  12. * @param {Object} ctx - egg全局变量
  13. * @return {void}
  14. */
  15. constructor(ctx) {
  16. super(ctx);
  17. ctx.showProject = true;
  18. // ctx.showTitle = true;
  19. }
  20. /**
  21. * 支付审批列表页
  22. *
  23. * @param {Object} ctx - egg全局页面
  24. * @return {void}
  25. */
  26. async index(ctx) {
  27. try {
  28. const auditPermission = await this.ctx.service.paymentPermissionAudit.getOnePermission(ctx.session.sessionUser.is_admin, ctx.session.sessionUser.accountId);
  29. if (!auditPermission) {
  30. throw '权限不足';
  31. }
  32. // 列表读取及目录读取
  33. const renderData = {
  34. auditPermission,
  35. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.payment.index),
  36. };
  37. if (ctx.session.sessionUser.is_admin) {
  38. const projectId = ctx.session.sessionProject.id;
  39. const permissionAudits = await ctx.service.paymentPermissionAudit.getList(projectId);
  40. // 获取所有项目参与者
  41. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  42. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  43. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  44. });
  45. const accountGroupList = accountGroup.map((item, idx) => {
  46. const groupList = accountList.filter(item => item.account_group === idx);
  47. return { groupName: item, groupList };
  48. });
  49. renderData.permissionAudits = permissionAudits;
  50. renderData.accountList = accountList;
  51. renderData.accountGroup = accountGroupList;
  52. }
  53. await this.layout('payment/index.ejs', renderData, 'payment/modal.ejs');
  54. } catch (err) {
  55. console.log(err);
  56. this.log(err);
  57. ctx.session.postError = err.toString();
  58. ctx.redirect(this.menu.menu.dashboard.url);
  59. }
  60. }
  61. async setting(ctx) {
  62. try {
  63. if (!ctx.session.sessionUser.is_admin) {
  64. throw '您无权打开此页';
  65. }
  66. const projectInfo = await ctx.service.project.getDataById(ctx.session.sessionProject.id);
  67. const modes = projectInfo.payment_setting ? JSON.parse(projectInfo.payment_setting) : ctx.helper._.cloneDeep(paymentConst.setting_modes);
  68. for (const m in modes) {
  69. const detailCount = await ctx.service.paymentDetail.getCountByPidType(ctx.session.sessionProject.id, modes[m].value);
  70. modes[m].can_check = !detailCount;
  71. }
  72. const renderData = {
  73. setting_modes: paymentConst.setting_modes,
  74. modes,
  75. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.payment.setting),
  76. };
  77. await this.layout('payment/setting.ejs', renderData);
  78. } catch (err) {
  79. console.log(err);
  80. this.log(err);
  81. ctx.session.postError = err.toString();
  82. ctx.redirect('/payment');
  83. }
  84. }
  85. async listLoad(ctx) {
  86. const responseData = {
  87. err: 0, msg: '', data: {},
  88. };
  89. // 先获取你创建的标段及参与的标段
  90. const tenderList = await ctx.service.paymentTender.getList(ctx.session.sessionUser.accountId);
  91. // 获取你创建的目录及对应目录下的所有目录
  92. const folderList = await ctx.service.paymentFolder.getList(ctx.session.sessionUser.accountId, tenderList);
  93. if (tenderList.length > 0) {
  94. for (const t of tenderList) {
  95. t.have_notice = await ctx.service.paymentDetail.haveNotice2Tender(t.id, ctx.session.sessionUser.accountId);
  96. t.have_detail = await ctx.service.paymentDetail.haveDetail2Tender(t.id);
  97. }
  98. }
  99. responseData.data.folderList = folderList;
  100. responseData.data.tenderList = tenderList;
  101. ctx.body = responseData;
  102. }
  103. async permissionSave(ctx) {
  104. try {
  105. if (ctx.session.sessionUser.is_admin === 0) throw '没有设置权限';
  106. const projectId = ctx.session.sessionProject.id;
  107. const responseData = {
  108. err: 0, msg: '', data: null,
  109. };
  110. const data = JSON.parse(ctx.request.body.data);
  111. if (!data.type) {
  112. throw '提交数据错误';
  113. }
  114. let uids;
  115. let result = false;
  116. let auditList = [];
  117. switch (data.type) {
  118. case 'add-audit':
  119. // 判断用户是单个还是数组
  120. uids = data.id instanceof Array ? data.id : [data.id];
  121. // 判断该用户的组是否已加入到表中,已加入则提示无需添加
  122. auditList = await ctx.service.paymentPermissionAudit.getAllDataByCondition({ where: { pid: projectId, uid: uids } });
  123. const addAidList = ctx.helper._.difference(uids, ctx.helper._.map(auditList, 'uid'));
  124. if (addAidList.length === 0) {
  125. throw '用户已存在权限中,无需重复添加';
  126. }
  127. const accountList = await ctx.service.projectAccount.getAllDataByCondition({ where: { id: addAidList } });
  128. await ctx.service.paymentPermissionAudit.saveAudits(projectId, accountList);
  129. responseData.data = await ctx.service.paymentPermissionAudit.getList(projectId);
  130. break;
  131. case 'del-audit':
  132. uids = data.id instanceof Array ? data.id : [data.id];
  133. auditList = await ctx.service.paymentPermissionAudit.getAllDataByCondition({ where: { id: uids } });
  134. if (auditList.length !== uids.length) {
  135. throw '该用户已不存在权限中,移除失败';
  136. }
  137. await ctx.service.paymentPermissionAudit.delAudit(uids);
  138. responseData.data = await ctx.service.paymentPermissionAudit.getList(projectId);
  139. break;
  140. case 'save-permission-one':
  141. result = await ctx.service.paymentPermissionAudit.updateOnePermission(data.updateData);
  142. if (!result) {
  143. throw '修改权限失败';
  144. }
  145. break;
  146. case 'save-permission-all':
  147. result = await ctx.service.paymentPermissionAudit.updateAllPermission(projectId, data.permission_type, data.value);
  148. if (!result) {
  149. throw '修改权限失败';
  150. }
  151. responseData.data = await ctx.service.paymentPermissionAudit.getList(projectId);
  152. break;
  153. default: throw '参数有误';
  154. }
  155. ctx.body = responseData;
  156. } catch (err) {
  157. this.log(err);
  158. ctx.body = { err: 1, msg: err.toString(), data: null };
  159. }
  160. }
  161. async save(ctx) {
  162. try {
  163. const projectId = ctx.session.sessionProject.id;
  164. const auditPermission = await this.ctx.service.paymentPermissionAudit.getOnePermission(ctx.session.sessionUser.is_admin, ctx.session.sessionUser.accountId);
  165. if (!auditPermission) {
  166. throw '权限不足';
  167. }
  168. const responseData = {
  169. err: 0, msg: '', data: {},
  170. };
  171. const data = JSON.parse(ctx.request.body.data);
  172. if (!data.type) {
  173. throw '提交数据错误';
  174. }
  175. let type = '';
  176. switch (data.type) {
  177. case 'add-folder':
  178. if (!auditPermission.folder) {
  179. throw '没有权限新建目录';
  180. }
  181. await ctx.service.paymentFolder.addFolder(projectId, ctx.session.sessionUser.accountId, data.parentId, data.name);
  182. break;
  183. case 'add-tender':
  184. if (!auditPermission.tender) {
  185. throw '没有权限新建标段';
  186. }
  187. await ctx.service.paymentTender.addTender(projectId, ctx.session.sessionUser.accountId, data.folderId, data.name);
  188. break;
  189. case 'edit-name':
  190. type = data.postData.type;
  191. const updateData = {
  192. name: data.postData.name,
  193. };
  194. const conditionData = {
  195. id: data.postData.id,
  196. };
  197. if (type === 'tender') {
  198. await ctx.service.paymentTender.update(updateData, conditionData);
  199. } else {
  200. await ctx.service.paymentFolder.update(updateData, conditionData);
  201. }
  202. break;
  203. case 'del':
  204. type = data.postData.type;
  205. if (type === 'tender') {
  206. await ctx.service.paymentTender.deleteTender(data.postData.id);
  207. } else {
  208. await ctx.service.paymentFolder.deleteFolder(data.postData.id);
  209. }
  210. break;
  211. case 'mode-setting':
  212. if (!ctx.session.sessionUser.is_admin) {
  213. throw '您无权操作';
  214. }
  215. const projectInfo = await ctx.service.project.getDataById(ctx.session.sessionProject.id);
  216. const modes = projectInfo.payment_setting ? JSON.parse(projectInfo.payment_setting) : ctx.helper._.cloneDeep(paymentConst.setting_modes);
  217. const checked = data.checked;
  218. if (modes[data.mode_type]) {
  219. const detailCount = await ctx.service.paymentDetail.getCountByPidType(ctx.session.sessionProject.id, modes[data.mode_type].value);
  220. if (detailCount > 0 && !checked) {
  221. throw '已存在对应模块的详情,无法关闭该模块';
  222. }
  223. const mode = modes[data.mode_type];
  224. if (mode.checked === checked) {
  225. throw '已选中该模块,无须重复选中';
  226. }
  227. mode.checked = checked;
  228. } else if (paymentConst.setting_modes[data.mode_type]) {
  229. modes[data.mode_type] = ctx.helper._.cloneDeep(paymentConst.setting_modes[data.mode_type]);
  230. modes[data.mode_type].checked = checked;
  231. } else {
  232. throw '该模块不存在';
  233. }
  234. await ctx.service.project.defaultUpdate({ id: ctx.session.sessionProject.id, payment_setting: JSON.stringify(modes) });
  235. break;
  236. default: throw '参数有误';
  237. }
  238. // 先获取你创建的标段及参与的标段
  239. const tenderList = await ctx.service.paymentTender.getList(ctx.session.sessionUser.accountId);
  240. // 获取你创建的目录及对应目录下的所有目录
  241. const folderList = await ctx.service.paymentFolder.getList(ctx.session.sessionUser.accountId, tenderList);
  242. responseData.data.folderList = folderList;
  243. responseData.data.tenderList = tenderList;
  244. ctx.body = responseData;
  245. } catch (err) {
  246. this.log(err);
  247. ctx.body = { err: 1, msg: err.toString(), data: null };
  248. }
  249. }
  250. /**
  251. * 获取审批界面所需的 原报、审批人数据等
  252. * @param ctx
  253. * @return {Promise<void>}
  254. * @private
  255. */
  256. async _getDetailAuditViewData(ctx) {
  257. const times = ctx.detail.status === auditConst.status.checkNo ? ctx.detail.times - 1 : ctx.detail.times;
  258. ctx.detail.user = await ctx.service.projectAccount.getAccountInfoById(ctx.detail.uid);
  259. ctx.detail.auditHistory = [];
  260. if (times >= 1) {
  261. for (let i = 1; i <= times; i++) {
  262. ctx.detail.auditHistory.push(await ctx.service.paymentDetailAudit.getAuditors(ctx.detail.id, i));
  263. }
  264. }
  265. // 获取审批流程中左边列表
  266. ctx.detail.auditors2 = ctx.detail.status === auditConst.status.checkNo && ctx.detail.uid !== ctx.session.sessionUser.accountId ?
  267. await ctx.service.paymentDetailAudit.getAuditorsWithOwner(ctx.detail.id, times) :
  268. await ctx.service.paymentDetailAudit.getAuditorsWithOwner(ctx.detail.id, ctx.detail.times);
  269. if (ctx.detail.status === auditConst.status.uncheck || ctx.detail.status === auditConst.status.checkNo) {
  270. ctx.detail.auditorList = await ctx.service.paymentDetailAudit.getAuditors(ctx.detail.id, ctx.detail.times);
  271. }
  272. // 是否已验证手机短信
  273. const pa = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);
  274. ctx.detail.authMobile = pa.auth_mobile;
  275. }
  276. /**
  277. * 支付表单页面
  278. *
  279. * @param {Object} ctx - egg全局页面
  280. * @return {void}
  281. */
  282. async detail(ctx) {
  283. try {
  284. await this._getDetailAuditViewData(ctx);
  285. const renderData = {
  286. trInfo: ctx.trInfo,
  287. paymentConst,
  288. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.payment.detail),
  289. auditConst,
  290. shenpiConst,
  291. preUrl: '/payment/' + ctx.tender.id + '/detail/' + ctx.detail.id,
  292. OSS_PATH: ctx.app.config.fujianOssPath,
  293. };
  294. renderData.nextDetail = await ctx.service.paymentDetail.getDataByCondition({ tr_id: ctx.trInfo.id, order: ctx.detail.order + 1 });
  295. let report_json = JSON.parse(ctx.detail.report_json);
  296. const content = [];
  297. // 获取当前报表人
  298. const rptAudit = await ctx.service.paymentRptAudit.getDataByCondition({ td_id: ctx.detail.id, uid: ctx.session.sessionUser.accountId });
  299. if (report_json.items[0].interact_cells.length > 0) {
  300. for (const [i, cell] of report_json.items[0].interact_cells.entries()) {
  301. const push_item = {
  302. type: paymentConst.rpt_dataType[cell.DataType],
  303. value: cell.Prefix ? ctx.helper._.replace(cell.Value, cell.Prefix, '') : cell.Value,
  304. label: cell.Label,
  305. index: i,
  306. };
  307. const thisControl = paymentConst.rpt_control[cell.control] ? cell.control : 'content';
  308. const oneShowContent = ctx.helper._.find(content, { control: thisControl });
  309. if (oneShowContent) {
  310. oneShowContent.items.push(push_item);
  311. } else {
  312. content.push({
  313. control: thisControl,
  314. title: paymentConst.rpt_control[thisControl],
  315. items: [push_item],
  316. });
  317. }
  318. }
  319. if (rptAudit && ((ctx.detail.status !== auditConst.status.checkNo && ctx.detail.status !== auditConst.status.checked) ||
  320. (ctx.detail.status === auditConst.status.checked && !renderData.nextDetail) ||
  321. (ctx.detail.status === auditConst.status.checkNo && ctx.detail.uid === ctx.session.sessionUser.accountId))) {
  322. // 获取个人签字,单位章,个人章地址
  323. const userInfo = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);
  324. const stampPathList = userInfo.stamp_path ? userInfo.stamp_path.split('!;!') : [];
  325. const companyInfo = userInfo.company ? await ctx.service.constructionUnit.getDataByCondition({ pid: ctx.session.sessionProject.id, name: userInfo.company }) : {};
  326. if (rptAudit.signature_msg) {
  327. const sign_msg = JSON.parse(rptAudit.signature_msg);
  328. report_json = await ctx.service.paymentDetail.signOneSignatureData(report_json, rptAudit.signature_name, sign_msg);
  329. rptAudit.signature_msg = sign_msg;
  330. } else {
  331. rptAudit.signature_msg = paymentConst.signature_msg;
  332. rptAudit.signature_msg.sign_path = userInfo.sign_path ? userInfo.sign_path : '';
  333. }
  334. renderData.signPath = userInfo.sign_path ? userInfo.sign_path : '';
  335. renderData.stampPathList = stampPathList;
  336. renderData.currentStamp = rptAudit && rptAudit.signature_msg.stamp_path ? rptAudit.signature_msg.stamp_path : (stampPathList.length > 0 ? stampPathList[0] : '');
  337. renderData.companyStamp = companyInfo && companyInfo.sign_path ? companyInfo.sign_path : '';
  338. }
  339. }
  340. renderData.rptAudit = rptAudit;
  341. renderData.content = content;
  342. renderData.report_json = report_json;
  343. if ((ctx.detail.status === auditConst.status.uncheck || ctx.detail.status === auditConst.status.checkNo) && ctx.session.sessionUser.accountId === ctx.detail.uid) {
  344. // data.accountGroup = accountGroup;
  345. // 获取所有项目参与者
  346. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  347. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  348. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  349. });
  350. renderData.accountList = accountList;
  351. renderData.accountGroup = accountGroup.map((item, idx) => {
  352. const groupList = accountList.filter(item => item.account_group === idx);
  353. return { groupName: item, groupList };
  354. });
  355. renderData.rptAuditList = await ctx.service.paymentRptAudit.getAllDataByCondition({ where: { td_id: ctx.detail.id } });
  356. }
  357. await this.layout('payment/detail.ejs', renderData, 'payment/detail_modal.ejs');
  358. } catch (err) {
  359. console.log(err);
  360. this.log(err);
  361. ctx.session.postError = err.toString();
  362. if (ctx.detail.tender_id && ctx.detail.tr_id) {
  363. ctx.redirect('/payment' + ctx.detail.tender_id + '/list/' + ctx.detail.tr_id);
  364. }
  365. ctx.redirect(this.menu.menu.dashboard.url);
  366. }
  367. }
  368. async detailSave(ctx) {
  369. try {
  370. const data = JSON.parse(ctx.request.body.data);
  371. if (!data.type) {
  372. throw '提交数据错误';
  373. }
  374. // 检查权限等
  375. if (ctx.detail.uid !== ctx.session.sessionUser.accountId && data.type !== 'update_sign') {
  376. throw '您无权操作';
  377. }
  378. if (data.type !== 'update_sign' && (ctx.detail.status === auditConst.status.checking || ctx.detail.status === auditConst.status.checked)) {
  379. throw '您无权操作';
  380. }
  381. const responseData = {
  382. err: 0, msg: '', data: {},
  383. };
  384. switch (data.type) {
  385. case 'update_rpt':
  386. responseData.data = await ctx.service.paymentDetail.updateReportJson(ctx.detail.id, data.report_json);
  387. break;
  388. case 'update_sign':
  389. if (ctx.detail.status === auditConst.status.checked) {
  390. throw '您无法操作签字/签章';
  391. }
  392. responseData.data = await ctx.service.paymentRptAudit.updateSignatureMsg(ctx.detail.id, ctx.session.sessionUser.accountId, data.signature_msg);
  393. break;
  394. case 'add_audit':
  395. const auditorId = this.app._.toInteger(data.auditorId);
  396. if (isNaN(auditorId) || auditorId <= 0) {
  397. throw '参数错误';
  398. }
  399. ctx.detail.auditorList = await ctx.service.paymentDetailAudit.getAuditors(ctx.detail.id, ctx.detail.times);
  400. // 检查审核人是否已存在
  401. const exist = this.app._.find(ctx.detail.auditorList, { aid: auditorId });
  402. if (exist) {
  403. throw '该审核人已存在,请勿重复添加';
  404. }
  405. const shenpiInfo = await ctx.service.paymentShenpiAudit.getDataByCondition({ tr_id: ctx.trInfo.id, sp_status: shenpiConst.sp_status.gdzs });
  406. const is_gdzs = shenpiInfo && ctx.trinfo.sp_status === shenpiConst.sp_status.gdzs ? 1 : 0;
  407. const result = await ctx.service.paymentDetailAudit.addAuditor(ctx.detail.id, auditorId, ctx.detail.times, is_gdzs);
  408. if (!result) {
  409. throw '添加审核人失败';
  410. }
  411. responseData.data = await ctx.service.paymentDetailAudit.getAuditorsWithOwner(ctx.detail.id, ctx.detail.times);
  412. break;
  413. case 'del_audit':
  414. const auditorId2 = data.auditorId instanceof Number ? data.auditorId : this.app._.toNumber(data.auditorId);
  415. if (isNaN(auditorId2) || auditorId2 <= 0) {
  416. throw '参数错误';
  417. }
  418. const result2 = await ctx.service.paymentDetailAudit.deleteAuditor(ctx.detail.id, auditorId2, ctx.detail.times);
  419. if (!result2) {
  420. throw '移除审核人失败';
  421. }
  422. responseData.data = await ctx.service.paymentDetailAudit.getAuditors(ctx.detail.id, ctx.detail.times);
  423. break;
  424. default: throw '参数有误';
  425. }
  426. ctx.body = responseData;
  427. } catch (err) {
  428. this.log(err);
  429. ctx.body = { err: 1, msg: err.toString(), data: null };
  430. }
  431. }
  432. async deleteDetail(ctx) {
  433. try {
  434. const detail_id = ctx.request.body.detail_id;
  435. const detailInfo = await ctx.service.paymentDetail.getDataById(detail_id);
  436. if (!detailInfo) {
  437. throw '所选报表表单详情已删除';
  438. }
  439. // 获取最新的期数
  440. const detail_highOrder = await ctx.service.paymentDetail.count({
  441. tr_id: detailInfo.tr_id,
  442. });
  443. if (!(ctx.session.sessionUser.is_admin || ((detailInfo.status === auditConst.status.uncheck || detailInfo.status === auditConst.status.checkNo) && detailInfo.uid === ctx.session.sessionUser.accountId)) || detail_highOrder !== detailInfo.order) {
  444. throw '您无权删除所选报表表单详情';
  445. }
  446. const result = await ctx.service.paymentDetail.deleteDetail(detail_id);
  447. if (!result) {
  448. throw '删除报表表单详情失败,请重试';
  449. }
  450. ctx.redirect('/payment/' + ctx.tender.id + '/list/' + detailInfo.tr_id);
  451. } catch (err) {
  452. this.log(err);
  453. ctx.session.postError = err.toString();
  454. ctx.redirect(ctx.request.header.referer);
  455. }
  456. }
  457. /**
  458. * 期审批流程(Get)
  459. * @param ctx
  460. * @return {Promise<void>}
  461. */
  462. async detailAuditors(ctx) {
  463. try {
  464. const responseData = {
  465. err: 0, msg: '', data: {},
  466. };
  467. const order = JSON.parse(ctx.request.body.data).order;
  468. const tr_id = ctx.params.trid;
  469. const detailInfo = await ctx.service.paymentDetail.getDataByCondition({ tr_id, order });
  470. // 获取审批流程中右边列表
  471. const auditHistory = [];
  472. const times = detailInfo.status === auditConst.status.checkNo ? detailInfo.times - 1 : detailInfo.times;
  473. if (times >= 1) {
  474. for (let i = 1; i <= times; i++) {
  475. auditHistory.push(await ctx.service.paymentDetailAudit.getAuditors(detailInfo.id, i));
  476. }
  477. }
  478. responseData.data.auditHistory = auditHistory;
  479. // 获取审批流程中左边列表
  480. responseData.data.auditors = await ctx.service.paymentDetailAudit.getAuditorsWithOwner(detailInfo.id, times);
  481. responseData.data.user = await ctx.service.projectAccount.getAccountInfoById(detailInfo.uid);
  482. ctx.body = responseData;
  483. } catch (error) {
  484. this.log(error);
  485. ctx.body = { err: 1, msg: error.toString(), data: null };
  486. }
  487. }
  488. async _returnRptProjectList(ctx, formProcess = false) {
  489. // 获取报表表单列表
  490. if (ctx.tender.uid === ctx.session.sessionUser.accountId || formProcess) {
  491. const rptProject = await ctx.service.rptTreeNode.getDataByCondition({ pid: ctx.session.sessionProject.id, name: '01.支付审批报表' });
  492. const rptProjectList = rptProject && rptProject.items ? JSON.parse(rptProject.items) : [];
  493. const tenderRptList = await ctx.service.paymentTenderRpt.getProcessList(ctx.tender.id);
  494. return await ctx.service.paymentTenderRpt.checkAndUpdateList(tenderRptList, rptProjectList, formProcess);
  495. }
  496. return await ctx.service.paymentTenderRpt.getList(ctx.tender.id, ctx.session.sessionUser.accountId);
  497. }
  498. async process(ctx) {
  499. try {
  500. const auditPermission = await this.ctx.service.paymentPermissionAudit.getOnePermission(ctx.session.sessionUser.is_admin, ctx.session.sessionUser.accountId);
  501. if (!auditPermission || !auditPermission.process) {
  502. throw '权限不足';
  503. }
  504. let [tenderRptList, rptProjectList] = await this._returnRptProjectList(ctx, true);
  505. tenderRptList = ctx.helper._.filter(tenderRptList, { type: 0, is_del: 0 });
  506. // for (const tr of tenderRptList) {
  507. // if (tr.status === shenpiConst.sp_status.gdspl) {
  508. // tr.auditList = await ctx.service.paymentShenpiAudit.getAuditList(ctx.tender.id, tr.id, tr.sp_status);
  509. // } else if (tr.status === shenpiConst.sp_status.gdzs) {
  510. // tr.audit = await ctx.service.paymentShenpiAudit.getAudit(ctx.tender.id, tr.id, tr.sp_status);
  511. // }
  512. // }
  513. // 获取所有项目参与者
  514. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  515. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  516. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  517. });
  518. const accountGroupList = accountGroup.map((item, idx) => {
  519. const groupList = accountList.filter(item => item.account_group === idx);
  520. return { groupName: item, groupList };
  521. });
  522. const renderData = {
  523. tender: ctx.tender,
  524. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.payment.process),
  525. rptProjectList,
  526. tenderRptList,
  527. shenpi: shenpiConst,
  528. accountList,
  529. accountGroup: accountGroupList,
  530. };
  531. await this.layout('payment/process.ejs', renderData, 'payment/process_modal.ejs');
  532. } catch (err) {
  533. console.log(err);
  534. this.log(err);
  535. ctx.session.postError = err.toString();
  536. ctx.redirect(this.request.headers.referer ? this.request.headers.referer : '/payment');
  537. }
  538. }
  539. async processSave(ctx) {
  540. try {
  541. const auditPermission = await this.ctx.service.paymentPermissionAudit.getOnePermission(ctx.session.sessionUser.is_admin, ctx.session.sessionUser.accountId);
  542. if (!auditPermission || !auditPermission.process) {
  543. throw '权限不足';
  544. }
  545. const responseData = {
  546. err: 0, msg: '', data: {},
  547. };
  548. const data = JSON.parse(ctx.request.body.data);
  549. if (!data.type) {
  550. throw '提交数据错误';
  551. }
  552. switch (data.type) {
  553. case 'add-rpt':
  554. responseData.data = await ctx.service.paymentTenderRpt.setRpt(ctx.tender.id, data.rpt_list);
  555. break;
  556. case 'change-status':
  557. responseData.data = await ctx.service.paymentTenderRpt.setStatus(data.tr_id, data.status);
  558. break;
  559. case 'get-audits':
  560. responseData.data = await ctx.service.paymentShenpiAudit.getShenpiAudit(data.tr_id, data.status);
  561. break;
  562. case 'add-audit':
  563. responseData.data = await ctx.service.paymentShenpiAudit.addAudit(data);
  564. break;
  565. case 'del-audit':
  566. responseData.data = await ctx.service.paymentShenpiAudit.removeAudit(data);
  567. break;
  568. default: throw '参数有误';
  569. }
  570. ctx.body = responseData;
  571. } catch (err) {
  572. this.log(err);
  573. ctx.body = { err: 1, msg: err.toString(), data: null };
  574. }
  575. }
  576. async rptList(ctx) {
  577. try {
  578. const tenderRptList = await this._returnRptProjectList(ctx);
  579. if (tenderRptList.length === 0) {
  580. throw '当前报表表单您无权打开或还没进行模块或表单审批设置';
  581. }
  582. for (const tr of tenderRptList) {
  583. tr.have_notice = await ctx.service.paymentDetail.haveNotice2TenderRpt(tr.id, ctx.session.sessionUser.accountId);
  584. }
  585. const trid = ctx.params.trid ? parseInt(ctx.params.trid) : 0;
  586. const trInfo = trid ? ctx.helper._.find(tenderRptList, { id: trid }) : tenderRptList[0];
  587. if (!trInfo) {
  588. throw '该模块已关闭或未进行表单审批设置';
  589. }
  590. // 获取列表
  591. const trDetailList = await ctx.service.paymentDetail.getValidDetails(trInfo.id);
  592. for (const s of trDetailList) {
  593. // s.curAuditor = null;
  594. // 根据期状态返回展示用户
  595. s.curAuditor = await ctx.service.paymentDetailAudit.getAuditorByStatus(s.id, s.status, s.times);
  596. }
  597. const renderData = {
  598. tender: ctx.tender,
  599. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.payment.list),
  600. tenderRptList,
  601. trInfo,
  602. trDetailList,
  603. rptMsg: null,
  604. auditConst,
  605. accountGroup: [],
  606. accountList: [],
  607. paymentConst,
  608. preUrl: '/payment/' + ctx.tender.id + '/list/' + trInfo.id,
  609. };
  610. // 获取报表信息,新增时及设置报表角色时使用
  611. if (trInfo.uid === ctx.session.sessionUser.accountId && trInfo.is_del === 0 && trInfo.rpt_id) {
  612. const rptTpl = await ctx.service.rptTpl.getDataById(trInfo.rpt_id);
  613. if (rptTpl) {
  614. const pageRst = await ctx.service.jpcReport.getAllPreviewPagesCommon(rptTpl, 'A4');
  615. renderData.rptMsg = pageRst.items[0];
  616. // 判断与原有的报表审批人列表是否有区别
  617. let difference = false;
  618. if (trInfo.report_items_json) {
  619. const report_items_json = JSON.parse(trInfo.report_items_json);
  620. const items = ['cells', 'signature_audit_cells', 'signature_cells', 'signature_date_cells', 'interact_cells'];
  621. for (const item of items) {
  622. if (!difference && report_items_json[item] &&
  623. !ctx.helper._.isEmpty(ctx.helper._.differenceWith(JSON.parse(JSON.stringify(renderData.rptMsg[item])), report_items_json[item], ctx.helper._.isEqual))) {
  624. // 因为interact_cells里存在undefind值,必须先用JSON.parse和JSON.stringify转义才能对比
  625. difference = true;
  626. break;
  627. }
  628. }
  629. }
  630. if (difference) {
  631. // 删除rpt_audit重新配置
  632. await ctx.service.paymentTenderRpt.defaultUpdate({
  633. id: trInfo.id,
  634. report_items_json: JSON.stringify(ctx.helper._.cloneDeep(renderData.rptMsg)),
  635. rpt_audit: null,
  636. is_change: 1,
  637. });
  638. trInfo.rpt_audit = null;
  639. trInfo.is_change = 1;
  640. trInfo.report_items_json = renderData.rptMsg;
  641. }
  642. if (trInfo.rpt_audit) {
  643. trInfo.rpt_audit = JSON.parse(trInfo.rpt_audit);
  644. const prjAccList = await ctx.service.projectAccount.getAllAccountByProjectId(ctx.session.sessionProject.id);
  645. for (const audit of trInfo.rpt_audit) {
  646. if (audit.uid) {
  647. const info = ctx.helper._.find(prjAccList, { id: audit.uid });
  648. audit.name = info.name;
  649. }
  650. }
  651. } else {
  652. trInfo.rpt_audit = [];
  653. for (const sc of renderData.rptMsg.signature_cells) {
  654. trInfo.rpt_audit.push({
  655. uid: null,
  656. rpt_name: sc.signature_name,
  657. });
  658. }
  659. }
  660. renderData.trInfo = trInfo;
  661. }
  662. if (renderData.rptMsg) {
  663. // 获取所有项目参与者
  664. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  665. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  666. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  667. });
  668. const accountGroupList = accountGroup.map((item, idx) => {
  669. const groupList = accountList.filter(item => item.account_group === idx);
  670. return { groupName: item, groupList };
  671. });
  672. renderData.accountList = accountList;
  673. renderData.accountGroup = accountGroupList;
  674. }
  675. }
  676. await this.layout('payment/list.ejs', renderData, 'payment/list_modal.ejs');
  677. } catch (err) {
  678. console.log(err);
  679. this.log(err);
  680. ctx.session.postError = err.toString();
  681. ctx.redirect(this.request && this.request.headers && this.request.headers.referer ? this.request.headers.referer : '/payment');
  682. }
  683. }
  684. async rptSave(ctx) {
  685. try {
  686. const tr_id = ctx.params.trid;
  687. if (!tr_id) {
  688. throw '参数有误';
  689. }
  690. const trInfo = await ctx.service.paymentTenderRpt.getDataById(tr_id);
  691. if (!trInfo) {
  692. throw '标段报表不存在';
  693. }
  694. if (ctx.session.sessionUser.accountId !== trInfo.uid) {
  695. throw '权限不足';
  696. }
  697. const responseData = {
  698. err: 0, msg: '', data: {},
  699. };
  700. const data = JSON.parse(ctx.request.body.data);
  701. if (!data.type) {
  702. throw '提交数据错误';
  703. }
  704. switch (data.type) {
  705. case 'rpt_audit':
  706. responseData.data = await ctx.service.paymentTenderRpt.updateRptAudit(trInfo, data.rpt_audit);
  707. break;
  708. case 'add-detail':
  709. responseData.data = await ctx.service.paymentDetail.addDetail(trInfo, data.code, data.s_time);
  710. break;
  711. default: throw '参数有误';
  712. }
  713. ctx.body = responseData;
  714. } catch (err) {
  715. this.log(err);
  716. ctx.body = { err: 1, msg: err.toString(), data: null };
  717. }
  718. }
  719. /**
  720. * 上报和重新上报
  721. * @param ctx
  722. * @return {Promise<void>}
  723. */
  724. async startAudit(ctx) {
  725. try {
  726. if (ctx.detail.uid !== ctx.session.sessionUser.accountId) {
  727. throw '您无权上报报表表单详情数据';
  728. }
  729. if (ctx.detail.status === auditConst.status.checking || ctx.detail.status === auditConst.status.checked) {
  730. throw '该报表表单详情数据当前无法上报';
  731. }
  732. await ctx.service.paymentDetailAudit.start(ctx.detail.id, ctx.detail.times);
  733. ctx.redirect(ctx.request.header.referer);
  734. } catch (err) {
  735. this.log(err);
  736. ctx.session.postError = err.toString();
  737. ctx.redirect(ctx.request.header.referer);
  738. }
  739. }
  740. /**
  741. * 审批
  742. * @param ctx
  743. * @return {Promise<void>}
  744. */
  745. async checkAudit(ctx) {
  746. try {
  747. if (!ctx.detail || ctx.detail.status !== auditConst.status.checking) {
  748. throw '当前报表表单详情数据有误';
  749. }
  750. if (!ctx.detail.curAuditor || ctx.detail.curAuditor.aid !== ctx.session.sessionUser.accountId) {
  751. throw '您无权进行该操作';
  752. }
  753. const data = {
  754. checkType: parseInt(ctx.request.body.checkType),
  755. opinion: ctx.request.body.opinion,
  756. };
  757. if (!data.checkType || isNaN(data.checkType)) {
  758. throw '提交数据错误';
  759. }
  760. if (data.checkType === auditConst.status.checkNo) {
  761. if (!data.checkType || isNaN(data.checkType)) {
  762. throw '提交数据错误';
  763. }
  764. }
  765. await ctx.service.paymentDetailAudit.check(ctx.detail.id, data, ctx.detail.times);
  766. ctx.redirect(ctx.request.header.referer);
  767. } catch (err) {
  768. this.log(err);
  769. ctx.session.postError = err.toString();
  770. ctx.redirect(ctx.request.header.referer);
  771. }
  772. }
  773. }
  774. return PaymentController;
  775. };