TaskList.tsx 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. import PageContent from '@/components/PageContent';
  2. import PageTitle from '@/components/PageTitle';
  3. import {
  4. IPropsType,
  5. ITopFilter,
  6. IUserType,
  7. } from '@/pages/TaskManage/Detail/TaskList/taskList.types';
  8. import {
  9. MandateClass,
  10. MandateStatus,
  11. MandateType,
  12. OrderStatus,
  13. OrderType,
  14. } from '@/pages/TaskManage/constent';
  15. import { connect, useLocation, useRequest } from '@umijs/max';
  16. import TopFilter from '@/pages/TaskManage/components/TopFilter';
  17. import { IMandateType } from '@/pages/TaskManage/index.types';
  18. import { useNavigate } from '@@/exports';
  19. import { DownOutlined } from '@ant-design/icons';
  20. import { Col, Collapse, CollapseProps, Divider, List, Row } from 'antd';
  21. import ScrollLoading from '@/components/ScrollLoading';
  22. import { getMandateList } from '@/services/TaskManage';
  23. import dayjs from 'dayjs';
  24. import { useEffect, useState } from 'react';
  25. import styles from './taskList.less';
  26. const TaskList: React.FC<IPropsType> = (props) => {
  27. const { userList, loading, dispatch } = props;
  28. const location = useLocation();
  29. const queryParams = new URLSearchParams(location.search);
  30. const project_id = Number(queryParams.get('project_id'));
  31. const mandateType = Number(queryParams.get('mandateType'));
  32. const navigate = useNavigate();
  33. // 顶部选择器配置
  34. const [topFiltersConfig, setTopFiltersConfig] = useState<ITopFilter[]>([]);
  35. const [mandateList, setMandateList] = useState<IMandateType[]>([]);
  36. const [currentParams, setCurrentParams] = useState({
  37. project_id,
  38. mandate_type: mandateType,
  39. pageSize: 20,
  40. currentPage: 1,
  41. });
  42. const [pagination, setPagination] = useState({
  43. current: 1,
  44. total: 0,
  45. pageSize: 20,
  46. });
  47. const { run: getList, loading: loadData } = useRequest(getMandateList, {
  48. defaultParams: [currentParams],
  49. formatResult: (result) => {
  50. const pageInfo = result.data.pagination;
  51. if (result.data.pagination.current === 1) {
  52. setMandateList(result.data.list);
  53. } else {
  54. if (mandateList.length < pageInfo.total) {
  55. setMandateList([...mandateList, ...result.data.list]);
  56. }
  57. }
  58. setPagination(pageInfo);
  59. },
  60. });
  61. // 获取用户
  62. useEffect(() => {
  63. if (userList.length === 0) {
  64. dispatch({
  65. type: 'taskUser/fetchUserList',
  66. payload: { project_id },
  67. });
  68. }
  69. }, []);
  70. // 配置顶部下拉过滤器
  71. useEffect(() => {
  72. const filters: ITopFilter[] = [];
  73. filters.push({
  74. key: 'mandate_class',
  75. placeholder: '任务类别',
  76. // @ts-ignore
  77. options: MandateClass.map((item) => {
  78. if (item.MandateType === mandateType) {
  79. return {
  80. value: item.value,
  81. label: item.label,
  82. key: item.value + '任务类别',
  83. };
  84. }
  85. return undefined;
  86. }).filter((item) => item),
  87. });
  88. filters.push({
  89. key: 'status',
  90. placeholder: '任务状态',
  91. options: MandateStatus.map((item) => {
  92. return {
  93. ...item,
  94. };
  95. }),
  96. });
  97. setTopFiltersConfig(filters);
  98. }, [mandateType]);
  99. /**
  100. * 处理选择器变化
  101. * @param value 选择的值,当清空时等于null
  102. */
  103. const onTopFilterChange = (value: any) => {
  104. if (topFiltersConfig.length === 0) {
  105. return;
  106. }
  107. const params: any = {
  108. project_id,
  109. mandate_type: mandateType,
  110. pageSize: 20,
  111. currentPage: 1,
  112. };
  113. for (let i = 0; i < value.length; i++) {
  114. if (value[i] !== null && topFiltersConfig[i] !== undefined) {
  115. params[topFiltersConfig[i].key] = value[i];
  116. }
  117. }
  118. setCurrentParams(params);
  119. getList(params);
  120. };
  121. const goTaskDetail = (mandate: IMandateType) => {
  122. navigate(
  123. `/task-manage/list/detail?project_id=${project_id}&mandate_id=${mandate.Id}`,
  124. );
  125. };
  126. const goTaskOrder = (
  127. orderID: number,
  128. orderType: number | undefined,
  129. mandateClass: number,
  130. ) => {
  131. if (orderType === undefined) {
  132. return;
  133. }
  134. navigate(
  135. `/task-manage/list/order-detail?project_id=${project_id}&order_id=${orderID}&order_type=${orderType}&mandate_class=${mandateClass}`,
  136. );
  137. };
  138. const buildTaskList = (item: IMandateType) => {
  139. const formatItem = {
  140. ...item,
  141. Status: MandateStatus.find((status) => status.value === item.Status),
  142. MandateType: MandateType.find((type) => type.value === item.MandateType),
  143. MandateClass: MandateClass.find(
  144. (itemClass) => itemClass.value === item.MandateClass,
  145. ),
  146. ResponsiblePeople: userList.find(
  147. (user) => user.ID === item.ResponsiblePeople,
  148. ),
  149. CreateTime: dayjs(item.CreateTime).format('YYYY-MM-DD HH:mm'),
  150. };
  151. const workOrder = item.Records.map((record) => {
  152. return {
  153. ...record,
  154. key: record.Id,
  155. Status: OrderStatus.find((status) => status.value === record.Status),
  156. RecordType: OrderType.find((type) => type.value === record.RecordType),
  157. Responsible: userList.find((user) => user.ID === record.Responsible),
  158. CreateTime: dayjs(record.CreateTime).format('YYYY-MM-DD HH:mm'),
  159. };
  160. });
  161. const collapseData: CollapseProps['items'] = [
  162. {
  163. key: '1',
  164. label: (
  165. <span style={{ color: '#5697e4' }}>关联工单({workOrder.length})</span>
  166. ),
  167. children: workOrder.map((order) => {
  168. return (
  169. <div key={order.Id} className={styles.workOrderCard}>
  170. <div className={styles.leftInfo}>
  171. <Row style={{ marginBottom: '15px' }}>
  172. <Col className={styles.fontS24} span={12}>
  173. 工单类型:{order.RecordType?.label || '-'}
  174. </Col>
  175. <Col className={styles.fontS24} span={12}>
  176. 时间:{order.CreateTime}
  177. </Col>
  178. </Row>
  179. <Row>
  180. <Col className={styles.fontS24} span={12}>
  181. 工单状态:
  182. <span style={{ color: '#5697e4' }}>
  183. {order.Status?.label}
  184. </span>
  185. </Col>
  186. <Col className={styles.fontS24} span={12}>
  187. 工单负责人:{order.Responsible?.CName}
  188. </Col>
  189. </Row>
  190. </div>
  191. <Divider type="vertical" style={{ height: '40px' }} />
  192. <div
  193. className={styles.rightButton}
  194. style={{ color: '#5697e4' }}
  195. onClick={() => {
  196. goTaskOrder(
  197. order.Id,
  198. order.RecordType?.value,
  199. item.MandateClass,
  200. );
  201. }}
  202. >
  203. 查看工单
  204. </div>
  205. </div>
  206. );
  207. }),
  208. },
  209. ];
  210. return (
  211. <List.Item style={{ borderBottom: '0' }}>
  212. <div className={`${styles.cardContainer} card-box`}>
  213. <Row justify="space-between" style={{ marginBottom: '20px' }}>
  214. <Col className={styles.fontS24}>时间:{formatItem.CreateTime}</Col>
  215. <Col className={styles.fontS24}>
  216. 任务类别:{formatItem.MandateClass?.label}
  217. </Col>
  218. <Col className={styles.fontS24}>
  219. 任务负责人:{formatItem.ResponsiblePeople?.CName || '-'}
  220. </Col>
  221. </Row>
  222. <Row
  223. justify="space-between"
  224. style={{
  225. paddingBottom: '10px',
  226. borderBottom: '1px solid #D5D5D5',
  227. }}
  228. >
  229. <Col className={styles.fontS24}>
  230. 任务状态:{formatItem.Status?.label || '-'}
  231. </Col>
  232. <Col>
  233. <div
  234. className={styles.fontS24}
  235. style={{
  236. backgroundColor: '#f5a623',
  237. color: 'white',
  238. width: '150px',
  239. height: '50px',
  240. display: 'flex',
  241. justifyContent: 'center',
  242. alignItems: 'center',
  243. }}
  244. onClick={() => {
  245. goTaskDetail(item);
  246. }}
  247. >
  248. 任务详情
  249. </div>
  250. </Col>
  251. </Row>
  252. <Row>
  253. <Collapse
  254. className={styles.collapseLabel}
  255. ghost
  256. expandIcon={({ isActive }) => (
  257. <DownOutlined
  258. style={{ color: '#5697e4' }}
  259. rotate={isActive ? 180 : 0}
  260. />
  261. )}
  262. items={collapseData}
  263. />
  264. </Row>
  265. </div>
  266. </List.Item>
  267. );
  268. };
  269. return (
  270. <PageContent closeable={false}>
  271. <PageTitle returnable>
  272. {MandateType.find((item) => item.value === mandateType)?.label}
  273. </PageTitle>
  274. <TopFilter filters={topFiltersConfig} onChange={onTopFilterChange} />
  275. <ScrollLoading
  276. height={180}
  277. loading={loading || loadData}
  278. pagination={pagination}
  279. handleLoadData={(current: number) =>
  280. getList({ ...currentParams, currentPage: current })
  281. }
  282. >
  283. <List
  284. itemLayout="horizontal"
  285. dataSource={mandateList}
  286. renderItem={buildTaskList}
  287. />
  288. </ScrollLoading>
  289. </PageContent>
  290. );
  291. };
  292. export default connect(
  293. ({
  294. taskUser,
  295. loading,
  296. }: any): {
  297. userList: IUserType[];
  298. loading: boolean;
  299. } => {
  300. return {
  301. userList: taskUser.userList,
  302. loading: loading.models['taskUser'],
  303. };
  304. },
  305. )(TaskList);