financial_pay.js 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940
  1. let auditUtils;
  2. $(function () {
  3. autoFlashHeight();
  4. $('#tid_select').select2({
  5. language: 'zh-CN',
  6. theme: 'bootstrap4',
  7. selectOnClose: true,
  8. // width: '150',
  9. });
  10. $('#tid_select').change(function () {
  11. const tid = parseInt($(this).val()) || 0;
  12. setSelectValue('tid', tid);
  13. });
  14. $('#status_select .to-log-link').click(function () {
  15. const status = parseInt($(this).data('val')) || null;
  16. setSelectValue('status', status);
  17. });
  18. $('#used_select .to-log-link').click(function () {
  19. const used = $(this).data('val') || null;
  20. setSelectValue('used', used);
  21. });
  22. function setSelectValue(select, value) {
  23. const routes = [];
  24. const tid = select === 'tid' ? value : $('#tid_select').val();
  25. if (tid) {
  26. routes.push('tid=' + tid);
  27. }
  28. const status = select === 'status' ? value : $('#status_selected').data('value');
  29. if (status) {
  30. routes.push('status=' + status);
  31. }
  32. const used = select === 'used' ? value : $('#used_selected').data('value');
  33. if (used) {
  34. routes.push('used=' + used);
  35. }
  36. window.location.href = `/sp/${spid}/financial/pay` + (routes.length ? '?' + routes.join('&') : '');
  37. }
  38. let timer = null
  39. let oldSearchVal = null
  40. $('#liucheng').on('input propertychange', '.gr-search', function(e) {
  41. oldSearchVal = e.target.value;
  42. timer && clearTimeout(timer);
  43. timer = setTimeout(() => {
  44. const newVal = $(this).val();
  45. const code = $(this).attr('data-code');
  46. let html = '';
  47. if (newVal && newVal === oldSearchVal) {
  48. accountList.filter(item => item && (item.name.indexOf(newVal) !== -1 || (item.mobile && item.mobile.indexOf(newVal) !== -1))).forEach(item => {
  49. html += `<dd class="border-bottom p-2 mb-0 " data-id="${item.id}" >
  50. <p class="mb-0 d-flex"><span class="text-primary">${item.name}</span><span
  51. class="ml-auto">${item.mobile || ''}</span></p>
  52. <span class="text-muted">${item.role || ''}</span>
  53. </dd>`
  54. });
  55. $('#' + code + '_dropdownMenu .book-list').empty();
  56. $('#' + code + '_dropdownMenu .book-list').append(html);
  57. } else {
  58. if (!$('#' + code + '_dropdownMenu .acc-btn').length) {
  59. accountGroup.forEach((group, idx) => {
  60. if (!group) return;
  61. html += `<dt><a href="javascript: void(0);" class="acc-btn" data-groupid="${idx}" data-type="hide"><i class="fa fa-plus-square"></i>
  62. </a> ${group.groupName}</dt>
  63. <div class="dd-content" data-toggleid="${idx}">`;
  64. group.groupList.forEach(item => {
  65. html += `<dd class="border-bottom p-2 mb-0 " data-id="${item.id}" >
  66. <p class="mb-0 d-flex"><span class="text-primary">${item.name}</span><span
  67. class="ml-auto">${item.mobile || ''}</span></p>
  68. <span class="text-muted">${item.role || ''}</span>
  69. </dd>`;
  70. });
  71. html += '</div>';
  72. });
  73. $('#' + code + '_dropdownMenu .book-list').empty();
  74. $('#' + code + '_dropdownMenu .book-list').append(html);
  75. }
  76. }
  77. }, 400);
  78. });
  79. $('#liucheng').on('show.bs.modal', function (e) {
  80. $('#shenpi-tender-list tr').removeClass('bg-warning');
  81. if (tenders.length > 0) {
  82. $('#shenpi-tender-list tr').eq(0).addClass('bg-warning');
  83. auditUtils. makeReportListHtml(tenders[0]);
  84. auditUtils.makeShenpiListHtml(tenders[0]);
  85. }
  86. });
  87. $('#shenpi-tender-list').on('click', '.change-tender', function () {
  88. if ($(this).hasClass('bg-warning')) {
  89. return;
  90. }
  91. $('#shenpi-tender-list tr').removeClass('bg-warning');
  92. $(this).parents('tr').addClass('bg-warning');
  93. const tid = parseInt($(this).parents('tr').data('tid')) || 0;
  94. const tender = tenders.find(t => t.id === tid);
  95. if (!tender) {
  96. toastr.error('请选择标段');
  97. return;
  98. }
  99. auditUtils.makeReportListHtml(tender);
  100. auditUtils.makeShenpiListHtml(tender);
  101. });
  102. auditUtils = {
  103. makeReportListHtml: function (flow) {
  104. let addHtml = '';
  105. $('#select-all-ptAudits').prop('checked', false);
  106. for (const pl of flow.permissionList) {
  107. addHtml += `<tr>
  108. <td class="text-center"><input type="checkbox" class="select-ptAudit" data-id="${pl.id}"></td><td>${pl.name}</td><td>${pl.company}</td>
  109. <td class="text-center"><input type="checkbox" class="save-report" data-id="${pl.id}" ${pl.is_report ? 'checked' : ''}></td><td class="text-center"><a href="javascript:void(0);" class="text-danger remove-audit" data-id="${pl.id}">移除</a></td>
  110. </tr>`;
  111. }
  112. $('#report-list').html(addHtml);
  113. },
  114. makeShenpiListHtml: function (flow) {
  115. let addhtml = '<ul class="list-unstyled">\n';
  116. addhtml += this.getgdsplHtml(flow, 'financial');
  117. addhtml += '</ul>\n';
  118. $('#shenpi-list').html(addhtml);
  119. },
  120. getAuditHtml: function(audit, is_report = 0) {
  121. return '<span class="d-inline-block"><span class="badge badge-light">'+ audit.name +' <span class="dropdown">\n' +
  122. ' <a href="javascript:void(0);" class="btn-sm text-danger px-1" title="移除" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"><i class="fa fa-remove"></i></a>\n' +
  123. ' <div class="dropdown-menu">\n' +
  124. ' <a class="dropdown-item" href="javascript:void(0);">确认移除' + ( is_report ? '填报人' : '审批人') + '?</a>\n' +
  125. ' <div class="dropdown-divider"></div>\n' +
  126. ' <div class="px-2 py-1 text-center">\n' +
  127. ' <button class="remove-audit btn btn-sm btn-danger" data-id="' + (is_report ? audit.id : audit.audit_id) + '">移除</button>\n' +
  128. ' <button class="btn btn-sm btn-secondary">取消</button>\n' +
  129. ' </div>\n' +
  130. ' </div>\n' +
  131. ' </span> ' +
  132. ' </span></span>\n'
  133. },
  134. getAuditTypeHtml: function(code, type) {
  135. const html = [];
  136. html.push(`<span class="d-inline-block"><select class="form-control form-control-sm audit-type-key" data-type="${type}">`);
  137. for (const t of auditType.types) {
  138. if (t.valid && t.valid.indexOf(code) < 0) continue;
  139. html.push(`<option value="${t.value}" ${t.value === type ? 'selected' : ''}>${t.name}</option>`);
  140. }
  141. html.push('</select></span> ');
  142. return html.join('');
  143. },
  144. getSelectAuditHtml: function (code, is_report = 0) {
  145. let divhtml = '';
  146. accountGroup.forEach((group, idx) => {
  147. let didivhtml = '';
  148. if(group) {
  149. group.groupList.forEach(item => {
  150. didivhtml += '<dd class="border-bottom p-2 mb-0 " data-id="' + item.id + '" >\n' +
  151. '<p class="mb-0 d-flex"><span class="text-primary">' + item.name + '</span><span\n' +
  152. ' class="ml-auto">' + item.mobile + '</span></p>\n' +
  153. ' <span class="text-muted">' + item.role + '</span>\n' +
  154. ' </dd>\n';
  155. });
  156. divhtml += '<dt><a href="javascript: void(0);" class="acc-btn" data-groupid="' + idx + '" data-type="hide"><i class="fa fa-plus-square"></i></a> ' + group.groupName + '</dt>\n' +
  157. ' <div class="dd-content" data-toggleid="' + idx + '">\n' + didivhtml +
  158. ' </div>\n';
  159. }
  160. });
  161. const html =
  162. ' <span class="d-inline-block">\n' +
  163. ' <div class="dropdown text-right">\n' +
  164. ' <button class="btn btn-outline-primary btn-sm dropdown-toggle" type="button" id="' + code + '_dropdownMenuButton" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">\n' +
  165. ' ' + (is_report ? '添加填报人' : '选择审批人') + '\n' +
  166. ' </button>\n' +
  167. ' <div class="dropdown-menu dropdown-menu-right" id="' + code + '_dropdownMenu" aria-labelledby="' + code + '_dropdownMenuButton" style="width:220px">\n' +
  168. ' <div class="mb-2 p-2"><input class="form-control form-control-sm gr-search"\n' +
  169. ' placeholder="姓名/手机 检索" autocomplete="off" data-code="' + code + '"></div>\n' +
  170. ' <dl class="list-unstyled book-list">\n' + divhtml +
  171. ' </dl>\n' +
  172. ' </div>\n' +
  173. ' </div>\n' +
  174. ' </span>\n';
  175. return html;
  176. },
  177. // 以下i从1开始
  178. getAuditGroupInnerHtml: function(code, auditGroup, i) {
  179. const html = [];
  180. const type = auditGroup.length > 0 ? auditGroup[0].audit_type : auditType.key.common;
  181. html.push(`<span class="col-auto">${transFormToChinese(i)}审</span><span class="col-10 spr-span">`);
  182. html.push(this.getAuditTypeHtml(code, type));
  183. for (const audit of auditGroup) {
  184. html.push(this.getAuditHtml(audit));
  185. }
  186. if (type !== auditType.key.common || auditGroup.length === 0) {
  187. html.push(this.getSelectAuditHtml(code));
  188. }
  189. if (type === auditType.key.union && auditGroup.length > 0) {
  190. html.push(`<button class="btn btn-sm btn-outline-primary" sp_type="${code}" audit_order="${i}" name="union-set">协同设置</button>`);
  191. }
  192. html.push('</span>');
  193. return html.join('');
  194. },
  195. getAuditGroupHtml: function (code, auditGroup, i) {
  196. return `<li class="d-flex justify-content-start align-items-center mb-3">${this.getAuditGroupInnerHtml(code, auditGroup, i)}</li>`;
  197. },
  198. getgdsplHtml(flow, this_code) {
  199. let addhtml = '';
  200. if (flow.auditGroupList.length !== 0) {
  201. for(const [i, auditGroup] of flow.auditGroupList.entries()) {
  202. addhtml += this.getAuditGroupHtml(this_code, auditGroup, i + 1);
  203. }
  204. const addGroupHtml = this_code === 'change' && (!flow.groupList || (flow.groupList && flow.groupList.length === 0)) ?
  205. `<span class="pl-3"><a href="javascript:void(0);" class="show-spzsave" data-code="${this_code}"><i class="fa fa-save"></i> 存为审批组</a></span>\n` : '';
  206. addhtml += '<li>\n' +
  207. ' <span class="pl-3"><a href="javascript:void(0);" class="add-audit" ><i class="fa fa-plus"></i> 添加流程</a></span>\n' + addGroupHtml +
  208. ' </li>';
  209. } else {
  210. addhtml += this.getAuditGroupHtml(this_code, [], 1);
  211. }
  212. return addhtml;
  213. },
  214. // 以下i从0开始
  215. addAudit: function (tender, user, i) {
  216. const flow = tender;
  217. if (!flow.auditGroupList) flow.auditGroupList = [];
  218. if (!flow.auditGroupList[i]) flow.auditGroupList[i] = [];
  219. flow.auditGroupList[i].push(user);
  220. return flow.auditGroupList[i];
  221. },
  222. removeAudit: function (tender, audit_id, i) {
  223. const flow = tender;
  224. if (flow.auditGroupList[i].length === 1) {
  225. flow.auditGroupList.splice(i, 1);
  226. return null;
  227. }
  228. flow.auditGroupList[i].splice(flow.auditGroupList[i].findIndex(x => { return x.audit_id === audit_id; }), 1);
  229. return flow.auditGroupList[i];
  230. },
  231. setAuditType: function (tender, audit_type, i) {
  232. const flow = tender;
  233. if (!flow || !flow.auditGroupList || !flow.auditGroupList[i]) return;
  234. flow.auditGroupList[i].forEach(x => { x.audit_type = audit_type});
  235. return flow.auditGroupList[i];
  236. }
  237. };
  238. // 选中填报人
  239. $('body').on('click', 'div[id="report_audit_dropdownMenu"] dl dd', function () {
  240. const tid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  241. const tender = tenders.find(t => t.id === tid);
  242. if (!tender) {
  243. toastr.error('请选择标段');
  244. return;
  245. }
  246. const id = parseInt($(this).data('id'));
  247. if (!id) return;
  248. if (!isNaN(id) && id !== 0) {
  249. postData(`/sp/${spid}/financial/pay/save`, {type: 'add-tender-audit', id: id, tid: tender.id }, function (result) {
  250. tender.permissionList = result;
  251. auditUtils.makeReportListHtml(tender);
  252. })
  253. }
  254. });
  255. // 移除填报人
  256. $('body').on('click', '#report-list .remove-audit', function () {
  257. const tid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  258. const tender = tenders.find(t => t.id === tid);
  259. if (!tender) {
  260. toastr.error('请选择标段');
  261. return;
  262. }
  263. const id = parseInt($(this).data('id'));
  264. deleteAfterHint(function () {
  265. postData(`/sp/${spid}/financial/pay/save`, {type: 'del-tender-audit', id, tid: tender.id }, function (result) {
  266. tender.permissionList = result.permissionList;
  267. tender.auditGroupList = result.auditGroupList;
  268. auditUtils.makeReportListHtml(tender);
  269. auditUtils.makeShenpiListHtml(tender);
  270. });
  271. }, '确认删除该标段用户?');
  272. });
  273. // 勾选是否为填报人
  274. $('body').on('click', '#report-list .save-report', function () {
  275. const tid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  276. const tender = tenders.find(t => t.id === tid);
  277. if (!tender) {
  278. toastr.error('请选择标段');
  279. return;
  280. }
  281. const id = parseInt($(this).data('id'));
  282. const isReport = $(this).prop('checked');
  283. const permission = tender.permissionList.find(p => p.id === id);
  284. if (!permission) {
  285. toastr.error('该用户不存在');
  286. return;
  287. }
  288. postData(`/sp/${spid}/financial/pay/save`, {type: 'save-permission', updateData: { id, is_report: isReport } }, function (result) {
  289. permission.is_report = isReport;
  290. });
  291. });
  292. $('#select-all-ptAudits').click(function () {
  293. $('#report-list .select-ptAudit').prop('checked', $(this).prop('checked'));
  294. });
  295. $('#batch-del-ptAudit').click(function () {
  296. const tid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  297. const tender = tenders.find(t => t.id === tid);
  298. if (!tender) {
  299. toastr.error('请选择标段');
  300. return;
  301. }
  302. const ids = [];
  303. $('#report-list .select-ptAudit:checked').each(function () {
  304. ids.push(parseInt($(this).data('id')));
  305. });
  306. if (ids.length === 0) {
  307. toastr.warning('请勾选要删除的用户');
  308. return;
  309. }
  310. deleteAfterHint(function () {
  311. postData(`/sp/${spid}/financial/pay/save`, {type: 'del-tender-audit', id: ids, tid: tender.id }, function (result) {
  312. tender.permissionList = result.permissionList;
  313. tender.auditGroupList = result.auditGroupList;
  314. auditUtils.makeReportListHtml(tender);
  315. auditUtils.makeShenpiListHtml(tender);
  316. });
  317. }, '确认删除已勾选的标段用户?');
  318. });
  319. $('#batch-other-ptAudit').click(function () {
  320. const cur_tenderid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  321. const tender = tenders.find(t => t.id === cur_tenderid);
  322. if (!tender) {
  323. toastr.error('请选择标段');
  324. return;
  325. }
  326. const ids = [];
  327. $('#report-list .select-ptAudit:checked').each(function () {
  328. ids.push(parseInt($(this).data('id')));
  329. });
  330. if (ids.length === 0) {
  331. toastr.warning('请勾选要同步的用户');
  332. return;
  333. }
  334. const num = $('#shenpi-tender-list input:checked').length;
  335. if (num === 0 || (num === 1 && parseInt($('#shenpi-tender-list input:checked').eq(0).parents('tr').data('tid')) === cur_tenderid)) {
  336. toastr.warning('请选择需要设置审批同步的标段');
  337. return;
  338. }
  339. const tenderList = [];
  340. for (let i = 0; i < num; i++) {
  341. const tid = parseInt($('#shenpi-tender-list input:checked').eq(i).parents('tr').data('tid'));
  342. if (tid !== cur_tenderid) {
  343. tenderList.push(tid);
  344. }
  345. }
  346. const data = {
  347. type: 'copy-tender-audit',
  348. id: ids,
  349. this_tid: tender.id,
  350. }
  351. data.tidList = tenderList.join(',');
  352. postData(`/sp/${spid}/financial/pay/save`, data, function (result) {
  353. toastr.success('已同步到其它勾选标段');
  354. for (let i = 0; i < num; i++) {
  355. const tid = parseInt($('#shenpi-tender-list input:checked').eq(i).parents('tr').data('tid'));
  356. if (tid !== cur_tenderid) {
  357. const other_tender = tenders.find(t => t.id === tid);
  358. const permissionList = _.filter(result.otherPermissionList, { tid: tid });
  359. other_tender.permissionList = permissionList;
  360. }
  361. }
  362. });
  363. });
  364. // 选中审批人
  365. $('body').on('click', '#shenpi-list div[id$="_dropdownMenu"] dl dd', function () {
  366. const tid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  367. const tender = tenders.find(t => t.id === tid);
  368. if (!tender) {
  369. toastr.error('请选择标段');
  370. return;
  371. }
  372. const id = parseInt($(this).data('id'));
  373. if (!id) return;
  374. let this_code = 'financial';
  375. const user = _.find(accountList, function (item) {
  376. return item.id === id;
  377. });
  378. // 判断是否已存在审批人
  379. const aid_num = $(this).parents('ul').find('.remove-audit').length;
  380. for (let i = 0; i < aid_num; i++) {
  381. const aid = parseInt($(this).parents('ul').find('.remove-audit').eq(i).data('id'));
  382. if (aid === id) {
  383. toastr.warning('该审核人已存在,请勿重复添加');
  384. return;
  385. }
  386. }
  387. const prop = {
  388. status: sp_status.gdspl,
  389. code: sp_type[this_code],
  390. audit_id: id,
  391. type: 'add',
  392. audit_type: parseInt($(this).parents('li').find('select[class*="audit-type-key"]')[0].value),
  393. audit_order: $(this).parents('li').index() + 1,
  394. };
  395. const _self = $(this);
  396. postData(`/sp/${spid}/financial/pay/save`, { type: 'add-shenpi-audit', shenpi: prop, tid }, function (result) {
  397. const data = result.shenpi;
  398. const auditGroup = auditUtils.addAudit(tender, { audit_id: data.audit_id, name: user.name, audit_type: data.audit_type, audit_order: data.audit_order }, prop.audit_order - 1);
  399. if (_self.parents('ul').find('.add-audit').length === 0) {
  400. _self.parents('ul').append('<li>\n' +
  401. ' <span class="pl-3"><a href="javascript:void(0);" class="add-audit" ><i class="fa fa-plus"></i> 添加流程</a></span>\n' +
  402. ' </li>');
  403. }
  404. _self.parents('li').html(auditUtils.getAuditGroupInnerHtml(this_code, auditGroup, prop.audit_order));
  405. tender.permissionList = result.permissionList;
  406. auditUtils.makeReportListHtml(tender);
  407. });
  408. });
  409. // 移除审批人
  410. $('body').on('click', '#shenpi-list .remove-audit', function () {
  411. const tid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  412. const tender = tenders.find(t => t.id === tid);
  413. if (!tender) {
  414. toastr.error('请选择标段');
  415. return;
  416. }
  417. const id = parseInt($(this).data('id'));
  418. const this_code = 'financial';
  419. const prop = {
  420. status: sp_status.gdspl,
  421. code: sp_type[this_code],
  422. audit_id: id,
  423. type: 'del',
  424. };
  425. const _self = $(this);
  426. postData('/tender/' + tid + '/shenpi/audit/save', prop, function (data) {
  427. const index = _self.parents('li').index();
  428. const auditGroup = auditUtils.removeAudit(tender, id, index);
  429. if (auditGroup) {
  430. _self.parents('li').html(auditUtils.getAuditGroupInnerHtml(this_code, auditGroup, index + 1));
  431. } else {
  432. const _selflc = _self.parents('#shenpi-list');
  433. _self.parents('li').remove();
  434. const aid_num = parseInt(_selflc.children('ul').find('li.d-flex').length);
  435. if (aid_num === 0) {
  436. _selflc.children('ul').html(auditUtils.getAuditGroupHtml(this_code, [], 1));
  437. } else {
  438. for (let i = 0; i < aid_num; i++) {
  439. _selflc.find('li.d-flex').eq(i).find('.col-auto').text(transFormToChinese(i+1) + '审');
  440. }
  441. }
  442. }
  443. })
  444. });
  445. $('body').on('click', '#shenpi-list .add-audit', function () {
  446. const num = $(this).parents('ul').children('li').length;
  447. const this_code = 'financial';
  448. const addhtml = auditUtils.getAuditGroupHtml(this_code, [], num);
  449. $(this).parents('ul').append(addhtml);
  450. $(this).parents('li').remove();
  451. });
  452. // 设置会签、或签
  453. $('body').on('change', 'select[class*="audit-type-key"]', function() {
  454. const tid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  455. const tender = tenders.find(t => t.id === tid);
  456. if (!tender) {
  457. toastr.error('请选择标段');
  458. return;
  459. }
  460. const removes = $(this).parents('.d-flex').find('.remove-audit');
  461. if (removes.length === 0) return;
  462. const this_status = sp_status.gdspl;
  463. const this_code = 'financial';
  464. const ids = [];
  465. const liParent = $(this).parents('li');
  466. removes.each((i, r) => { ids.push(parseInt(r.getAttribute('data-id'))); });
  467. const prop = {
  468. status: this_status,
  469. code: sp_type[this_code],
  470. audit_id: ids,
  471. audit_type: parseInt(this.value),
  472. type: 'audit-type',
  473. };
  474. if (prop.audit_type === auditType.key.common && ids.length > 1) {
  475. toastr.warning('设置个人审批前请先删除多余的审批人');
  476. this.value = this.getAttribute('data-type');
  477. return;
  478. }
  479. const _self = this;
  480. postData('/tender/'+ tid +'/shenpi/audit/save', prop, function () {
  481. _self.setAttribute('data-type', _self.value);
  482. const auditGroup = auditUtils.setAuditType(tender, prop.audit_type, liParent.index());
  483. liParent.html(auditUtils.getAuditGroupInnerHtml(this_code, auditGroup, liParent.index() + 1));
  484. });
  485. });
  486. $('#set-other-tenders').on('click', function () {
  487. const cur_tenderid = parseInt($('#shenpi-tender-list tr.bg-warning').data('tid'));
  488. const tender = tenders.find(t => t.id === cur_tenderid);
  489. if (!tender) {
  490. toastr.error('请选择标段');
  491. return;
  492. }
  493. if (tender.auditGroupList.length === 0) {
  494. toastr.warning('请先设置审批流程再同步到其它标段');
  495. return;
  496. }
  497. const this_code = 'financial';
  498. const num = $('#shenpi-tender-list input:checked').length;
  499. if (num === 0 || (num === 1 && parseInt($('#shenpi-tender-list input:checked').eq(0).parents('tr').data('tid')) === cur_tenderid)) {
  500. toastr.warning('请选择需要设置审批同步的标段');
  501. return;
  502. }
  503. const data = {
  504. type: 'copy-shenpi-audit',
  505. status: sp_status.gdspl,
  506. code: this_code,
  507. this_tid: cur_tenderid,
  508. };
  509. // 二维数组变一维
  510. data.auditList = [];
  511. for (const auditGroup of tender.auditGroupList) {
  512. data.auditList.push(...auditGroup);
  513. }
  514. const tenderList = [];
  515. for (let i = 0; i < num; i++) {
  516. const tid = parseInt($('#shenpi-tender-list input:checked').eq(i).parents('tr').data('tid'));
  517. if (tid !== cur_tenderid) {
  518. tenderList.push(tid);
  519. }
  520. }
  521. data.tidList = tenderList.join(',');
  522. console.log(data);
  523. postData(`/sp/${spid}/financial/pay/save`, data, function (result) {
  524. toastr.success('已同步到其它勾选标段');
  525. for (let i = 0; i < num; i++) {
  526. const tid = parseInt($('#shenpi-tender-list input:checked').eq(i).parents('tr').data('tid'));
  527. if (tid !== cur_tenderid) {
  528. const other_tender = tenders.find(t => t.id === tid);
  529. other_tender.auditGroupList = tender.auditGroupList;
  530. const permissionList = _.filter(result.otherPermissionList, { tid: tid });
  531. other_tender.permissionList = permissionList;
  532. }
  533. }
  534. });
  535. });
  536. // 添加到成员中
  537. $('body').on('click', '.book-list dt', function () {
  538. const idx = $(this).find('.acc-btn').attr('data-groupid')
  539. const type = $(this).find('.acc-btn').attr('data-type')
  540. if (type === 'hide') {
  541. $(this).parent().find(`div[data-toggleid="${idx}"]`).show(() => {
  542. $(this).children().find('i').removeClass('fa-plus-square').addClass('fa-minus-square-o')
  543. $(this).find('.acc-btn').attr('data-type', 'show')
  544. })
  545. } else {
  546. $(this).parent().find(`div[data-toggleid="${idx}"]`).hide(() => {
  547. $(this).children().find('i').removeClass('fa-minus-square-o').addClass('fa-plus-square')
  548. $(this).find('.acc-btn').attr('data-type', 'hide')
  549. })
  550. }
  551. return false
  552. });
  553. $('#add-pay').on('show.bs.modal', function () {
  554. let t = null;
  555. if (tenders.length > 0) {
  556. if (is_admin) {
  557. t = tenders[0];
  558. } else {
  559. const filterTender = tenders.filter(t => _.includes(fptReportTids, t.id));
  560. if (filterTender.length > 0) {
  561. t = filterTender[0];
  562. }
  563. }
  564. }
  565. $('#add-pay-tender').val(t ? t.id : '');
  566. changeTender(t);
  567. });
  568. $('#add-pay-tender').on('change', function () {
  569. const tender = tenders.find(t => t.id === parseInt($(this).val()));
  570. changeTender(tender);
  571. });
  572. $('#add-pay-btn').on('click', function () {
  573. const tid = $('#add-pay-tender').val();
  574. const tender = tenders.find(t => t.id === parseInt(tid));
  575. if (!tender) {
  576. toastr.error('请选择支付标段');
  577. return;
  578. }
  579. const code = $('#add-pay-code').val();
  580. if (!code) {
  581. toastr.error('请先去标段属性,填写合同编号');
  582. return;
  583. }
  584. const prop = {
  585. tid: tender.id,
  586. code: code,
  587. used: $('#add-pay-used').val(),
  588. };
  589. postData(`/sp/${spid}/financial/pay/save`, { type: 'add-pay', updateData: prop }, function (result) {
  590. window.location.href = `/sp/${spid}/financial/pay/${result.id}/detail`;
  591. });
  592. });
  593. $('body').on('click', '#pay-list .del-pay-btn', function () {
  594. const fpid = $(this).data('id');
  595. deleteAfterHint(function () {
  596. postData(`/sp/${spid}/financial/pay/save`, {type: 'del-pay', postData: { node: fpid }}, function (result) {
  597. window.location.reload();
  598. })
  599. }, '确认删除该资金支付?');
  600. });
  601. $('#pay-list tr').on('click', function () {
  602. const tid = parseInt($(this).data('tid'));
  603. const tender = tenders.find(t => t.id === tid);
  604. if (tid && tender) {
  605. $(this).siblings().removeClass('alert-warning');
  606. $(this).addClass('alert-warning');
  607. $('#show-pay-account').show();
  608. $('#payaccount input[name="tid"]').val(tender.id);
  609. } else {
  610. $(this).siblings().removeClass('alert-warning');
  611. $('#show-pay-account').hide();
  612. $('#payaccount input[name="tid"]').val('');
  613. }
  614. });
  615. $('#payaccount').on('show.bs.modal', function () {
  616. const tid = parseInt($('#payaccount input[name="tid"]').val());
  617. const tender = tenders.find(t => t.id === tid);
  618. $('#payaccount .modal-title').text('付款账号()');
  619. $('#payaccount input[name="name"]').val('');
  620. $('#payaccount input[name="bank"]').val('');
  621. $('#payaccount input[name="bank_account"]').val('');
  622. $('#payaccount input[name="contact"]').val('');
  623. $('#payaccount input[name="phone"]').val('');
  624. if (tid && tender) {
  625. $('#payaccount .modal-title').text('付款账号(' + tender.name + ')');
  626. $('#payaccount input[name="name"]').val(tender.pt.name);
  627. $('#payaccount input[name="bank"]').val(tender.pt.bank);
  628. $('#payaccount input[name="bank_account"]').val(tender.pt.bank_account);
  629. $('#payaccount input[name="contact"]').val(tender.pt.contact);
  630. $('#payaccount input[name="phone"]').val(tender.pt.phone);
  631. if (is_admin || (fptReportTids && _.includes(fptReportTids, tid))) {
  632. $('#payaccount table input').attr('readonly', false);
  633. $('#get-form-tender').show();
  634. $('#set-pay-btn').show();
  635. $('#payaccount input[name="id"]').val(tender.pt.id);
  636. } else {
  637. $('#payaccount table input').attr('readonly', true);
  638. $('#get-form-tender').hide();
  639. $('#set-pay-btn').hide();
  640. $('#payaccount input[name="id"]').val('');
  641. $('#payaccount input[name="tid"]').val('');
  642. }
  643. }
  644. });
  645. $('#get-form-tender').on('click', function () {
  646. const tid = $('#payaccount input[name="tid"]').val();
  647. const tender = tenders.find(t => t.id === parseInt(tid));
  648. if (!tid || !tender) {
  649. toastr.error('标段不存在');
  650. return;
  651. }
  652. $('#payaccount input[name="name"]').val(tender.pay_account.name);
  653. $('#payaccount input[name="bank"]').val(tender.pay_account.bank);
  654. $('#payaccount input[name="bank_account"]').val(tender.pay_account.account);
  655. $('#payaccount input[name="contact"]').val(tender.pay_account.contact);
  656. $('#payaccount input[name="phone"]').val(tender.pay_account.phone);
  657. toastr.success('已同步');
  658. });
  659. $('#set-pay-btn').on('click', function () {
  660. const tid = $('#payaccount input[name="tid"]').val();
  661. const tender = tenders.find(t => t.id === parseInt(tid));
  662. if (!tid || !tender) {
  663. toastr.error('标段不存在');
  664. return;
  665. }
  666. if (is_admin || (fptReportTids && _.includes(fptReportTids, tender.id))) {
  667. const data = {
  668. id: parseInt($('#payaccount input[name="id"]').val()),
  669. tid: tender.id,
  670. name: $('#payaccount input[name="name"]').val(),
  671. bank: $('#payaccount input[name="bank"]').val(),
  672. bank_account: $('#payaccount input[name="bank_account"]').val(),
  673. contact: $('#payaccount input[name="contact"]').val(),
  674. phone: $('#payaccount input[name="phone"]').val(),
  675. }
  676. if (!data.name) {
  677. toastr.error('请填写开户名称');
  678. return;
  679. }
  680. if (!data.bank) {
  681. toastr.error('请填写开户银行');
  682. return;
  683. }
  684. if (!data.bank_account) {
  685. toastr.error('请填写开户账号');
  686. return;
  687. }
  688. postData(`/sp/${spid}/financial/pay/save`, { type: 'set-pay-tender', updateData: data }, function (result) {
  689. toastr.success('保存成功');
  690. tender.pt = result;
  691. });
  692. } else {
  693. toastr.error('无权限操作');
  694. return;
  695. }
  696. });
  697. $('#pay-list tr').eq(0).click();
  698. function changeTender(tender) {
  699. $('#add-pay-tender').val(tender ? tender.id : '');
  700. if (!tender) {
  701. toastr.warning('请先为项目添加标段再申请支付');
  702. return;
  703. }
  704. if (tender.dealCode) {
  705. $('#add-pay-code').val(tender.dealCode + '-' + moment().format('YYYYMMDD') + '-' + makeNum(tender.startNum));
  706. $('#add-pay-code').siblings('span').text('');
  707. } else {
  708. $('#add-pay-code').val('');
  709. $('#add-pay-code').siblings('span').html('请先去<a href="/tender/' + tender.id + '" target="_blank">标段属性</a>,填写合同编号');
  710. }
  711. }
  712. function makeNum(num) {
  713. let str = num.toString();
  714. while (str.length < 3) {
  715. str = '0' + str;
  716. }
  717. return str;
  718. }
  719. $('#audit-list').on('click', 'a', function() {
  720. const type = $(this).data('target')
  721. const auditCard = $(this).parent().parent()
  722. if (type === 'show') {
  723. $(this).data('target', 'hide')
  724. auditCard.find('.fold-card').slideDown('swing', () => {
  725. auditCard.find('#end-target').text($(this).data('idx') + '#')
  726. auditCard.find('#fold-btn').text('收起历史审核记录')
  727. })
  728. } else {
  729. $(this).data('target', 'show')
  730. auditCard.find('.fold-card').slideUp('swing', () => {
  731. auditCard.find('#end-target').text('1#')
  732. auditCard.find('#fold-btn').text('展开历史审核记录')
  733. })
  734. }
  735. });
  736. // 获取审批流程
  737. $('a[data-target="#sp-list" ]').on('click', function () {
  738. const data = {
  739. type: 'get-auditors',
  740. id: $(this).attr('c-id'),
  741. };
  742. postData(`/sp/${spid}/financial/pay/save`, data, function (result) {
  743. const { auditHistory, auditors2, user } = result;
  744. let auditorsHTML = [];
  745. auditors2.forEach((group, idx) => {
  746. if (idx === 0) {
  747. auditorsHTML.push(`<li class="list-group-item d-flex justify-content-between align-items-center">
  748. <span class="mr-1"><i class="fa fa fa-play-circle fa-rotate-90"></i></span>
  749. <span class="text-muted">${getGroupAuditHtml(group)}</span>
  750. <span class="badge badge-light badge-pill ml-auto"><small>原报</small></span>
  751. </li>`);
  752. } else if(idx === auditors2.length -1 && idx !== 0) {
  753. auditorsHTML.push(`<li class="list-group-item d-flex justify-content-between align-items-center">
  754. <span class="mr-1"><i class="fa fa fa-stop-circle fa-rotate-90"></i></span>
  755. <span class="text-muted">${getGroupAuditHtml(group)}</span>
  756. <div class="d-flex ml-auto">
  757. ${getAuditTypeHtml(group[0].audit_type)}
  758. <span class="badge badge-light badge-pill ml-auto"><small>终审</small></span>
  759. </div>
  760. </li>`);
  761. } else {
  762. auditorsHTML.push(`<li class="list-group-item d-flex justify-content-between align-items-center">
  763. <span class="mr-1"><i class="fa fa fa-chevron-circle-down"></i></span>
  764. <span class="text-muted">${getGroupAuditHtml(group)}</span>
  765. <div class="d-flex ml-auto">
  766. ${getAuditTypeHtml(group[0].audit_type)}
  767. <span class="badge badge-light badge-pill"><small>${transFormToChinese(idx)}审</small></span>
  768. </div>
  769. </li>`);
  770. }
  771. });
  772. $('#auditor-list').empty();
  773. $('#auditor-list').append(auditorsHTML.join(''));
  774. let historyHTML = [];
  775. auditHistory.forEach((his, idx) => {
  776. if (idx === auditHistory.length - 1 && auditHistory.length !== 1) {
  777. historyHTML.push(`<div class="text-right"><a href="javascript: void(0);" id="fold-btn" data-target="show">展开历史审批流程</a></div>`);
  778. }
  779. historyHTML.push(`<div class="${idx < auditHistory.length - 1 ? 'fold-card' : ''}">`);
  780. historyHTML.push(`<div class="text-center text-muted">${idx+1}#</div>`);
  781. historyHTML.push(`<ul class="timeline-list list-unstyled mt-2 ${ idx === auditHistory.length - 1 && auditHistory.length !== 1 ? 'last-auditor-list' : '' }">`);
  782. his.forEach((group, index) => {
  783. if (index === 0) {
  784. historyHTML.push(`<li class="timeline-list-item pb-2">
  785. <div class="timeline-item-date">
  786. ${group.beginYear}
  787. <span>${group.beginDate}</span>
  788. <span>${group.beginTime}</span>
  789. </div>
  790. <div class="timeline-item-tail"></div>
  791. <div class="timeline-item-icon bg-success text-light"><i class="fa fa-caret-down"></i></div>
  792. <div class="timeline-item-content">
  793. <div class="py-1">
  794. <span class="text-black-50">原报</span>
  795. <span class="pull-right text-success">${idx !== 0 ? '重新' : '' }上报审批</span>
  796. </div>
  797. <div class="card">
  798. <div class="card-body px-3 py-0">
  799. <div class="card-text p-2 py-3 row">
  800. <div class="col">
  801. <span class="h6">${user.name}</span>
  802. <span class="text-muted ml-1">${user.role}</span>
  803. </div>
  804. <div class="col">
  805. <span class="pull-right text-success"><i class="fa fa-check-circle"></i></span>
  806. </div>
  807. </div>
  808. </div>
  809. </div>
  810. </div>
  811. </li>`);
  812. }
  813. historyHTML.push(`<li class="timeline-list-item pb-2 ${ group.status === auditConst.status.uncheck && idx === auditHistory.length - 1 && auditHistory.length !== 1 ? 'is_uncheck' : ''}">`);
  814. if (group.endYear) {
  815. historyHTML.push(`<div class="timeline-item-date">${group.endYear}<span>${group.endDate}</span><span>${group.endTime}</span></div>`);
  816. }
  817. if (index < his.length - 1) {
  818. historyHTML.push('<div class="timeline-item-tail"></div>');
  819. }
  820. if (group.status === auditConst.status.checked) {
  821. historyHTML.push('<div class="timeline-item-icon bg-success text-light"><i class="fa fa-check"></i></div>');
  822. } else if (group.status === auditConst.status.checkNo) {
  823. historyHTML.push('<div class="timeline-item-icon bg-warning text-light"><i class="fa fa-level-up"></i></div>');
  824. } else if (group.status === auditConst.status.checking) {
  825. historyHTML.push('<div class="timeline-item-icon bg-warning text-light"><i class="fa fa-ellipsis-h"></i></div>');
  826. } else if (group.status === auditConst.status.checkAgain) {
  827. historyHTML.push('<div class="timeline-item-icon bg-warning text-light"><i class="fa fa-check"></i></div>');
  828. } else {
  829. historyHTML.push('<div class="timeline-item-icon bg-secondary text-light"></div>');
  830. }
  831. historyHTML.push('<div class="timeline-item-content">');
  832. historyHTML.push('<div class="py-1">');
  833. const statuStr = group.status !== auditConst.status.uncheck ?
  834. `<span class="pull-right ${auditConst.statusClass[group.status]}">${auditConst.statusString[group.status]}</span>` : '';
  835. historyHTML.push(`
  836. <span class="text-black-50">
  837. ${ group.audit_order === 0 ? '原报' : !group.is_final ? group.audit_order + '审' : '终审' } ${getAuditTypeText(group.audit_type)}
  838. </span>
  839. ${statuStr}`);
  840. historyHTML.push('</div>');
  841. historyHTML.push('<div class="card"><div class="card-body px-3 py-0">');
  842. for (const [i, auditor] of group.auditors.entries()) {
  843. historyHTML.push(`<div class="card-text p-2 py-3 row ${ ( i > 0 ? 'border-top' : '') }">`);
  844. historyHTML.push(`<div class="col"><span class="h6">${auditor.name}</span><span class="text-muted ml-1">${auditor.role}</span></div>`);
  845. historyHTML.push('<div class="col">');
  846. if (auditor.status === auditConst.status.checked) {
  847. historyHTML.push('<span class="pull-right text-success"><i class="fa fa-check-circle"></i></span>');
  848. } else if (auditor.status === auditConst.status.checkNo) {
  849. historyHTML.push('<span class="pull-right text-warning"><i class="fa fa-share-square fa-rotate-270"></i></span>');
  850. } else if (auditor.status === auditConst.status.checking) {
  851. historyHTML.push('<span class="pull-right text-warning"><i class="fa fa-commenting"></i></span>');
  852. } else if (auditor.status === auditConst.status.checkAgain) {
  853. historyHTML.push('<span class="pull-right text-warning"><i class="fa fa-check"></i></span>');
  854. }
  855. historyHTML.push('</div>');
  856. if (auditor.opinion) {
  857. historyHTML.push(`<div class="col-12 py-1 bg-light"><i class="fa fa-commenting-o mr-1"></i>${auditor.opinion}</div>`);
  858. }
  859. historyHTML.push('</div>');
  860. }
  861. historyHTML.push('</div></div>');
  862. historyHTML.push('</div>');
  863. historyHTML.push('</li>');
  864. });
  865. historyHTML.push('</div>');
  866. historyHTML.push('</ul>');
  867. });
  868. $('#audit-list').empty();
  869. $('#audit-list').append(historyHTML.join(''));
  870. });
  871. });
  872. $.subMenu({
  873. menu: '#sub-menu', miniMenu: '#sub-mini-menu', miniMenuList: '#mini-menu-list',
  874. toMenu: '#to-menu', toMiniMenu: '#to-mini-menu',
  875. key: 'menu.1.0.0',
  876. miniHint: '#sub-mini-hint', hintKey: 'menu.hint.1.0.1',
  877. callback: function (info) {
  878. if (info.mini) {
  879. $('.panel-title').addClass('fluid');
  880. $('#sub-menu').removeClass('panel-sidebar');
  881. } else {
  882. $('.panel-title').removeClass('fluid');
  883. $('#sub-menu').addClass('panel-sidebar');
  884. }
  885. autoFlashHeight();
  886. }
  887. });
  888. });
  889. const getGroupAuditHtml = function (group) {
  890. return group.map(u => { return `<small class="d-inline-block text-dark mx-1" title="${u.role}" data-auditorId="${u.aid}">${u.name}</small>`; }).join('');
  891. };
  892. const getAuditTypeHtml = function (type) {
  893. if (type === auditType.key.common) return '';
  894. return `<div class="li-subscript"><span class="badge badge-pill badge-${auditType.info[type].class} p-1 badge-bg-small"><small>${auditType.info[type].short}</small></span></div>`;
  895. };
  896. const getAuditTypeText = function (type) {
  897. if (type === auditType.key.common) return '';
  898. return `<span class="text-${auditType.info[type].class}">${auditType.info[type].long}</span>`;
  899. };