financial_pay.js 46 KB

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