CommitAuditModal.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685
  1. import React, { useEffect, useState, useRef, useMemo, useCallback } from 'react';
  2. import '@ant-design/compatible/assets/index.css';
  3. import {
  4. Modal,
  5. Input,
  6. Select,
  7. message,
  8. Cascader,
  9. Form,
  10. Tabs,
  11. Row,
  12. Col,
  13. Empty,
  14. Button,
  15. Steps,
  16. Popover,
  17. Upload,
  18. Table,
  19. Divider,
  20. Collapse,
  21. } from 'antd';
  22. import { PlusOutlined, UploadOutlined } from '@ant-design/icons';
  23. import { connect } from 'dva';
  24. import { isArray, result, set } from 'lodash';
  25. import { useForm } from 'rc-field-form';
  26. import { async } from '@antv/x6/lib/registry/marker/async';
  27. import AuditDetailed from './AuditDetailed';
  28. import AuditFlow from './AuditFlow';
  29. import {
  30. queryDingSchema,
  31. queryGetBomForm,
  32. queryProcessFlows,
  33. querySaveBomForm,
  34. } from '@/services/boom';
  35. import { Form as Form3x } from '@ant-design/compatible';
  36. import { getCurrentUser } from '@/utils/authority';
  37. import DDCode from '@/components/DDComponents/DDCode';
  38. import { uploadFile, queryUserListByRoleID } from '@/services/boom';
  39. import ApprovalProcess from './ApprovalProcess';
  40. import { uuidv4 } from '@antv/xflow';
  41. import AliyunOSSUpload from '@/components/OssUpload/AliyunOssUploader';
  42. import AttachmentTable from '@/components/AttachmentTable';
  43. const { TextArea } = Input;
  44. const { Option } = Select;
  45. const { TabPane } = Tabs;
  46. const { Step } = Steps;
  47. // 提交
  48. function CommitAuditModal(props) {
  49. const {
  50. dispatch,
  51. visible,
  52. onClose,
  53. loading,
  54. version,
  55. versionList,
  56. flowDetail,
  57. currentUser,
  58. luckysheet,
  59. userList,
  60. templateId,
  61. OSSData,
  62. } = props;
  63. // console.log(loading);
  64. const [auditId, setAuditId] = useState();
  65. const [data, setData] = useState([]);
  66. const [length, setLength] = useState(1);
  67. const [formData, setFromData] = useState({});
  68. const [auditList, setAuditList] = useState([]); //用于创建Tabs表单
  69. const [formComponentValues, setFormComponentValues] = useState({}); //用于创建Tabs表单
  70. const [form] = Form.useForm();
  71. const [approvalProcess, setApprovalProcess] = useState({});
  72. const [selectUserList, setSelectUserList] = useState([]);
  73. const [curNodeIdx, setCurNodeIdx] = useState(-1);
  74. const [dataSource, setDataSource] = useState([]);
  75. const uploadList = useRef([]);
  76. useEffect(() => {
  77. if (!visible) return;
  78. const { edges, nodes } = flowDetail;
  79. // initFormList().then(approvalProcess => {
  80. let Id = version.template_node_id;
  81. const currentId = flowDetail.nodes.find?.(item => item.Id == Id)?.node_id;
  82. const data = treeData(currentId);
  83. console.log('===============审批节点======', data);
  84. if (data.length <= 0) {
  85. setAuditId(currentId);
  86. } else {
  87. let defaultValues = {};
  88. if (data.length == 1) {
  89. let value = getDataValue(data[0]);
  90. defaultValues[`circle`] = value;
  91. } else {
  92. data.forEach((item, index) => {
  93. let value = getDataValue(item);
  94. defaultValues[`circle${index}`] = value;
  95. });
  96. }
  97. // 设置延迟,等待组件渲染
  98. setTimeout(async () => {
  99. form.setFieldsValue(defaultValues);
  100. const approvalProcess = await initFormList();
  101. Object.values(defaultValues).forEach(value => onChange(value, approvalProcess || {}));
  102. }, 200);
  103. }
  104. setData(data);
  105. // });
  106. }, [version.template_node_id, visible]);
  107. useEffect(() => {
  108. if (!visible) return;
  109. dispatch({
  110. type: 'detail/getChartOSSData',
  111. payload: {
  112. projectId: version.project_id,
  113. },
  114. });
  115. }, [visible]);
  116. useEffect(() => {
  117. form.resetFields();
  118. setAuditList([]);
  119. }, [visible]);
  120. const OnModelFileDone = file => {
  121. var path = OSSData.host + '/' + file.url;
  122. uploadList.current = [...uploadList.current, path];
  123. // const files = form.getFieldsValue('files');
  124. // form.setFieldValue('files', files.concat(path));
  125. console.log(uploadList.current);
  126. // setThumbnail(path);
  127. };
  128. const OnUploading = file => {};
  129. const uploadProps = {
  130. OSSData: OSSData,
  131. onDone: OnModelFileDone,
  132. onUploading: OnUploading,
  133. noStyle: false,
  134. // showUploadList: false,
  135. };
  136. const initFormList = async () => {
  137. const res = await queryGetBomForm({
  138. project_id: version.project_id,
  139. node_id: version.template_node_id,
  140. });
  141. if (res.data) {
  142. const formList = JSON.parse(res.data.json);
  143. setApprovalProcess(formList.approvalProcess || {});
  144. return formList.approvalProcess;
  145. // setFormComponentValues(defaultFormData);
  146. }
  147. };
  148. const treeData = currentId => {
  149. const list = getNextNodes(currentId, 'custom-circle');
  150. const fun = nodes => {
  151. const re = nodes?.forEach((item, idx) => {
  152. const data = getNextNodes(item.Id, 'custom-circle');
  153. if (data || data.length > 0) list.push(...data);
  154. fun(data);
  155. });
  156. };
  157. fun(list);
  158. const fun2 = list => {
  159. const parents = list.filter(item => list.findIndex(node => node.Id == item.parentId) == -1);
  160. let translator = (parents, children) => {
  161. setLength(length + 1);
  162. parents.forEach(parent => {
  163. children.forEach((current, index) => {
  164. if (current.parentId === parent.Id) {
  165. let temp = JSON.parse(JSON.stringify(children));
  166. temp.splice(index, 1);
  167. translator([current], temp);
  168. if (!parent.children.find(item => item.Id == current.Id))
  169. parent.children.push(current);
  170. }
  171. });
  172. });
  173. };
  174. translator(parents, list);
  175. return parents;
  176. };
  177. return fun2(list);
  178. };
  179. const currentNodeId = useMemo(() => {
  180. let Id = version.template_node_id;
  181. setAuditId(currentNodeId);
  182. return flowDetail.nodes.find?.(item => item.Id == Id)?.node_id;
  183. }, [flowDetail, version]);
  184. /**
  185. *
  186. * @param {*} currentId 当前节点
  187. * @param {*} type 下一个节点的类型 custom-circle: 审批节点 custom-rect: 业务节点
  188. * @returns
  189. */
  190. const getNextNodes = (currentId, type) => {
  191. const { edges, nodes } = flowDetail;
  192. if (!currentId) return [];
  193. //删除虚线通向的节点
  194. // let targetIds = edges
  195. // .filter(edge => {
  196. // let line = edge.attrs?.line?.strokeDasharray?.split(' ');
  197. // return edge.source.cell == currentId && line && line[0] == '0';
  198. // })
  199. // .map(item => item.target.cell);
  200. let targetIds = edges
  201. .filter(edge => edge.source.cell == currentId)
  202. .map(item => item.target.cell);
  203. edges.filter(edge => edge.source.cell == currentId);
  204. let auditNodes = nodes.filter(node => {
  205. if (type && node.name != type) {
  206. return false;
  207. }
  208. return targetIds.indexOf(node.id) != -1;
  209. });
  210. const result = auditNodes.map(item => {
  211. return {
  212. label: item.label,
  213. value: item.Id,
  214. Id: item.node_id,
  215. parentId: currentId,
  216. children: [],
  217. };
  218. });
  219. return result || [];
  220. };
  221. const nextNodesList = useMemo(() => {
  222. if (!auditId && !currentNodeId) return [];
  223. return getNextNodes(auditId || currentNodeId, 'custom-rect');
  224. }, [auditId, currentNodeId, flowDetail]);
  225. const changeAudit = id => {
  226. let node = flowDetail.nodes.find?.(item => item.Id == id);
  227. setAuditId(node?.node_id);
  228. };
  229. const onChange = (value, approvalProcess) => {
  230. if (value) {
  231. changeAudit(value[value.length - 1]);
  232. setAuditListFun(approvalProcess);
  233. } else {
  234. changeAudit('');
  235. setAuditList([]);
  236. setApprovalProcess({});
  237. }
  238. form.setFieldValue('next_template_node_id', '');
  239. };
  240. const getReComputeAudit = (items, changedValues) => {
  241. const id = Object.keys(changedValues)[0];
  242. const formItem = items?.find(item => item.props.id == id);
  243. if (formItem && formItem.props?.required) return true;
  244. return false;
  245. };
  246. //填写表单实时计算审批流程
  247. const advanceSubmit = async () => {
  248. console.log('重重新计算审批流程');
  249. var fieldsValue = await form.validateFields();
  250. let hasFlowId = true; //是否都绑定审批节点
  251. let result = Object.values(fieldsValue)
  252. .map(item => {
  253. if (item && Array.isArray(item)) return item;
  254. })
  255. .filter(item => item);
  256. const formList = await getFromData(result);
  257. let params = {
  258. desc: fieldsValue.desc,
  259. // 审核流程id
  260. flow_id: 0,
  261. node_level_id: 0,
  262. id: version.id,
  263. project_id: version.project_id,
  264. cur_template_node_id: version.template_node_id * 1, // 当前节点
  265. next_template_node_id: 0, // 审核完成后的业务节点
  266. template_node_id: null, // 将要流转的节点审批节点
  267. flow_path: null, //审批节点数组
  268. // 模板id.一致就行
  269. template_id: version.template_id,
  270. cur_template_id: version.template_id,
  271. next_template_id: version.template_id,
  272. form_list: formList,
  273. };
  274. dispatch({
  275. type: 'detail/advanceSubmitNextNode',
  276. payload: params, //values,
  277. callback: data => {
  278. if (data) {
  279. setApprovalProcess(data);
  280. }
  281. },
  282. });
  283. };
  284. //处理tabs页
  285. const setAuditListFun = async (approvalProcess = {}) => {
  286. var fieldsValue = await form.validateFields();
  287. let addAuditList = [];
  288. let result = Object.values(fieldsValue)
  289. .map(item => {
  290. if (item && Array.isArray(item)) return item;
  291. })
  292. .filter(item => item)
  293. .flat(Infinity);
  294. let nodeList = [...new Set(result)]
  295. .map(Id => {
  296. return flowDetail.nodes.find?.(item => item.Id == Id);
  297. })
  298. .filter(item => item);
  299. let flowIds = [...new Set(nodeList.map(item => item.flow_id))].join(',');
  300. let data = await queryProcessFlows({ ids: flowIds });
  301. if (data && data?.length > 0) {
  302. let newlist = nodeList.map(node => {
  303. let curData = data.find(item => item.id == node.flow_id);
  304. let newItem = {
  305. name: curData?.name,
  306. nodeId: node.Id,
  307. items: JSON.parse(curData.form_json || '[]'),
  308. };
  309. return newItem;
  310. });
  311. addAuditList = [...addAuditList, ...newlist];
  312. }
  313. addAuditList.forEach((item, index) => {
  314. let Components = Form3x.create({
  315. onValuesChange: (props, changedValues, allValues) => {
  316. const { items } = props;
  317. formComponentValues[item.nodeId] = items
  318. .map(item => {
  319. const itemProps = item.props;
  320. let val = allValues[itemProps.id];
  321. if (!itemProps.label || val === '') return;
  322. if (val instanceof Object) {
  323. return {
  324. name: itemProps.label,
  325. id: itemProps.id,
  326. value: [...val],
  327. };
  328. } else if (allValues[itemProps.id]) {
  329. return {
  330. name: itemProps.label,
  331. id: itemProps.id,
  332. value: [allValues[itemProps.id]] || undefined,
  333. };
  334. }
  335. })
  336. .filter(item => item);
  337. if (getReComputeAudit(items, changedValues)) advanceSubmit();
  338. setFormComponentValues({ ...formComponentValues });
  339. },
  340. })(AuditDetailed);
  341. item.FormComponents = <Components items={item.items} />;
  342. });
  343. setAuditList(addAuditList);
  344. if (Object.keys(approvalProcess).length == 0) advanceSubmit();
  345. };
  346. const getFromData = async idList => {
  347. const data = formComponentValues;
  348. const result = [];
  349. //获取流转节点的层级关系
  350. let len = 0;
  351. let list = [];
  352. idList.forEach(item => {
  353. if (len < item.length) len = item.length;
  354. });
  355. for (let i = 0; i < len; i++) {
  356. idList.forEach(item => {
  357. if (item && item[i]) list.push(item[i]);
  358. });
  359. }
  360. let firstList = [...new Set(list)];
  361. // let attachment = await upload();
  362. firstList.forEach(id => {
  363. let approvalNode = flowDetail.nodes.find?.(item => item.Id == id);
  364. if (!approvalNode) return;
  365. let values = data[approvalNode.Id] || [];
  366. const audit_list = approvalProcess[approvalNode.Id]?.map(item => {
  367. if (item[0].type == 'role') return item[0].nowValue;
  368. return item[0].value;
  369. });
  370. const formItem = {
  371. flow_id: approvalNode.flow_id,
  372. template_node_id: approvalNode.Id,
  373. formComponentValues: [...values], //{ name: '附件', value: JSON.stringify(attachment) }
  374. audit_list: audit_list || [],
  375. };
  376. result.push(JSON.stringify(formItem));
  377. });
  378. return result;
  379. };
  380. const getFlowPath = node => {
  381. //[134, 135]
  382. let itemData = {};
  383. const Function = (curId, index) => {
  384. if (!curId) return;
  385. let data = {};
  386. let approvalNode = flowDetail.nodes.find?.(item => item.Id == curId);
  387. data.template_id = version.template_id;
  388. data.flow_id = approvalNode?.flow_id || 0;
  389. data.node_level_id = approvalNode?.flow_id ? 1 : 0;
  390. data.template_node_id = approvalNode?.Id;
  391. index++;
  392. if (approvalNode?.Id) {
  393. if (!approvalNode?.flow_id) {
  394. hasFlowId = false;
  395. }
  396. }
  397. const res = Function(node[index], index);
  398. if (res) {
  399. data.flow_path = [res];
  400. }
  401. return data;
  402. };
  403. itemData = Function(node[0], 0);
  404. return itemData;
  405. };
  406. const onFinish = async () => {
  407. const isOk = Object.values(approvalProcess).every(item => {
  408. return item.every(cur => {
  409. if (cur[0].type == 'role') return cur[0].nowValue;
  410. return true;
  411. });
  412. });
  413. if (!isOk) {
  414. message.error('请选择审批人。');
  415. return;
  416. }
  417. var fieldsValue = await form.validateFields();
  418. let hasFlowId = true; //是否都绑定审批节点
  419. const getFlowPath = node => {
  420. //[134, 135]
  421. let itemData = {};
  422. const Function = (curId, index) => {
  423. if (!curId) return;
  424. let data = {};
  425. let approvalNode = flowDetail.nodes.find?.(item => item.Id == curId);
  426. data.template_id = version.template_id;
  427. data.flow_id = approvalNode?.flow_id || 0;
  428. data.node_level_id = approvalNode?.flow_id ? 1 : 0;
  429. data.template_node_id = approvalNode?.Id;
  430. index++;
  431. if (approvalNode?.Id) {
  432. if (!approvalNode?.flow_id) {
  433. hasFlowId = false;
  434. }
  435. }
  436. const res = Function(node[index], index);
  437. if (res) {
  438. data.flow_path = [res];
  439. }
  440. return data;
  441. };
  442. itemData = Function(node[0], 0);
  443. return itemData;
  444. };
  445. let result = Object.values(fieldsValue)
  446. .map(item => {
  447. if (item && Array.isArray(item)) return item;
  448. })
  449. .filter(item => item);
  450. let serviceNode = flowDetail.nodes.find?.(item => item.Id == fieldsValue.next_template_node_id);
  451. if (!serviceNode) {
  452. message.error('请选择需要流转的业务节点。');
  453. return;
  454. }
  455. const flowPath = result.map(item => getFlowPath(item));
  456. const formList = await getFromData(result);
  457. let params = {
  458. desc: fieldsValue.desc,
  459. // 审核流程id
  460. // flow_id: approvalNode?.flow_id || 0,
  461. // node_level_id: approvalNode?.flow_id ? 1 : 0,
  462. id: version.id,
  463. project_id: version.project_id,
  464. cur_template_node_id: version.template_node_id * 1, // 当前节点
  465. next_template_node_id: serviceNode.Id * 1, // 审核完成后的业务节点
  466. // template_node_id: result[0][0], // 将要流转的节点审批节点
  467. // flow_path:flow_path, //审批节点数组
  468. // 模板id.一致就行
  469. template_id: version.template_id,
  470. cur_template_id: version.template_id,
  471. next_template_id: version.template_id,
  472. };
  473. if (result.length <= 0) {
  474. //直接走业务节点
  475. } else if (result.length <= 1 && result[0]?.length <= 1) {
  476. //单个审批节点
  477. let approvalNode = flowDetail.nodes.find?.(item => item.Id == result[0][0]);
  478. params.flow_id = approvalNode?.flow_id || 0;
  479. params.node_level_id = approvalNode?.flow_id ? 1 : 0;
  480. params.template_node_id = result[0][0]; // 将要流转的节点审批节点
  481. params.form_list = formList; //创建钉钉表单所需数据
  482. if (approvalNode?.Id) {
  483. if (!approvalNode?.flow_id) {
  484. hasFlowId = false;
  485. }
  486. }
  487. } else {
  488. //多节点审批
  489. params.template_node_id = result[0][0]; // 将要流转的节点审批节点
  490. params.flow_path = flowPath;
  491. params.form_list = formList; //创建钉钉表单所需数据
  492. }
  493. if (!hasFlowId) {
  494. message.error('当前存在审批节点未绑定审批流程!请联系管理员。');
  495. return;
  496. }
  497. await querySaveBomForm({
  498. project_id: version.project_id,
  499. node_id: version.template_node_id,
  500. json: JSON.stringify({ approvalProcess }),
  501. });
  502. params.audit_series = uuidv4();
  503. params.files = uploadList.current.join(',');
  504. console.log(params);
  505. onSubmitNextNode(params);
  506. };
  507. const onSubmitNextNode = values => {
  508. dispatch({
  509. type: 'detail/submitNextNode',
  510. payload: values,
  511. callback: newVersion => {
  512. onClose();
  513. // 更新flow流程图
  514. dispatch({
  515. type: 'xflow/queryBoomFlowDetail',
  516. payload: {
  517. id: templateId,
  518. },
  519. });
  520. },
  521. });
  522. };
  523. const CascaderNode = index => {
  524. return (
  525. <Form.Item
  526. labelCol={{ span: 7 }}
  527. wrapperCol={{ span: 15 }}
  528. label={`审批节点${index + 1}`}
  529. name={`circle${index}`}
  530. key={`circle${index}`}
  531. >
  532. <Cascader style={{ width: '100%' }} options={data} onChange={onChange} />
  533. </Form.Item>
  534. );
  535. };
  536. const upload = async () => {
  537. let blob = await luckysheet.current.getExcelBolb();
  538. let formData = new FormData();
  539. formData.append('userid', currentUser.DingUserId);
  540. formData.append('file', new File([blob], `${version.version_name}_${version.version_no}.xlsx`));
  541. try {
  542. let res = await uploadFile(formData);
  543. let data = JSON.parse(res.dentry);
  544. return [
  545. {
  546. spaceId: String(data.spaceId),
  547. fileName: data.name,
  548. fileSize: String(data.spaceId),
  549. fileType: data.extension,
  550. fileId: data.id,
  551. },
  552. ];
  553. } catch (error) {
  554. message.error('附件上传失败');
  555. }
  556. };
  557. const columns = [
  558. {
  559. title: '文件名称',
  560. dataIndex: 'name',
  561. key: 'name',
  562. },
  563. {
  564. title: '操作',
  565. render: record => <a>删除</a>,
  566. },
  567. ];
  568. useEffect(() => {
  569. if (!visible) {
  570. // 清空数据
  571. uploadList.current = [];
  572. }
  573. }, [visible]);
  574. return (
  575. <Modal
  576. confirmLoading={loading.global}
  577. destroyOnClose
  578. title="提交流转目标"
  579. width={1000}
  580. visible={visible}
  581. onCancel={() => {
  582. setAuditId();
  583. onClose();
  584. }}
  585. onOk={onFinish}
  586. >
  587. <Form form={form}>
  588. {data.map((item, idx) => (data.length == 1 ? CascaderNode('') : CascaderNode(idx)))}
  589. <Form.Item
  590. labelCol={{ span: 7 }}
  591. wrapperCol={{ span: 15 }}
  592. label="业务节点"
  593. name="next_template_node_id"
  594. // rules={[{ required: true, message: '请选择业务节点' }]}
  595. >
  596. <Select style={{ width: '100%' }}>
  597. {nextNodesList.map(item => (
  598. <Option key={item.value}>{item.label}</Option>
  599. ))}
  600. </Select>
  601. </Form.Item>
  602. <Form.Item labelCol={{ span: 7 }} wrapperCol={{ span: 15 }} label="备注信息" name="desc">
  603. <Input.TextArea />
  604. </Form.Item>
  605. <Form.Item labelCol={{ span: 7 }} wrapperCol={{ span: 15 }} label="附件">
  606. {OSSData.host && <AliyunOSSUpload {...uploadProps} directory={false} label="上传文件" />}
  607. </Form.Item>
  608. </Form>
  609. <Collapse style={{ marginTop: 20 }}>
  610. <Collapse.Panel header="已上传附件" key="1">
  611. <AttachmentTable version={version} canDelete={version.last_version == 0} />
  612. </Collapse.Panel>
  613. </Collapse>
  614. <Tabs defaultActiveKey="1">
  615. {auditList.map((item, idx) => (
  616. <TabPane tab={item.name} key={`${idx}_${item.title}`}>
  617. <Row>
  618. <Col span={17}>{item.FormComponents}</Col>
  619. <Col offset={1} span={6}>
  620. {!approvalProcess[item.nodeId] ? ( //!formComponentValues[item.nodeId] ||
  621. <Empty description="请先填写表单" />
  622. ) : (
  623. <ApprovalProcess
  624. id={item.nodeId}
  625. approvalProcess={approvalProcess}
  626. onChange={setApprovalProcess}
  627. />
  628. )}
  629. </Col>
  630. </Row>
  631. </TabPane>
  632. ))}
  633. </Tabs>
  634. </Modal>
  635. );
  636. }
  637. function getDataValue(item) {
  638. let arr = [];
  639. arr.push(item.value);
  640. if (item.children?.length > 0) {
  641. let res = getDataValue(item.children[0]);
  642. arr = arr.concat(res);
  643. }
  644. return arr;
  645. }
  646. export default connect(({ xflow, detail, user, loading }) => ({
  647. flowDetail: xflow.flowDetail,
  648. versionList: detail.versionList,
  649. currentUser: user.currentUser,
  650. userList: user.list,
  651. OSSData: detail.OSSData,
  652. loading,
  653. }))(CommitAuditModal);