pages_submission_records.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. import React, { useState } from 'react';
  2. import { useQueryClient } from '@tanstack/react-query';
  3. import {
  4. Button, Table, Space,
  5. Form, Input, Select, message, Modal,
  6. Card, Row, Col,
  7. Popconfirm, Tag, DatePicker
  8. } from 'antd';
  9. import {
  10. useQuery,
  11. } from '@tanstack/react-query';
  12. import dayjs from 'dayjs';
  13. import weekday from 'dayjs/plugin/weekday';
  14. import localeData from 'dayjs/plugin/localeData';
  15. import 'dayjs/locale/zh-cn';
  16. import type {
  17. SubmissionRecord,
  18. SubmissionRecordListResponse
  19. } from '../share/types_stock.ts';
  20. import {
  21. SubmissionRecordStatus, SubmissionRecordStatusNameMap,
  22. } from '../share/types_stock.ts';
  23. import { getEnumOptions } from './utils.ts';
  24. import {
  25. SubmissionRecordsAPI
  26. } from './api/index.ts';
  27. // 配置 dayjs 插件
  28. dayjs.extend(weekday);
  29. dayjs.extend(localeData);
  30. dayjs.locale('zh-cn');
  31. // 提交记录管理页面组件
  32. export const SubmissionRecordsPage = () => {
  33. const queryClient = useQueryClient();
  34. const [modalVisible, setModalVisible] = useState(false);
  35. const [formMode, setFormMode] = useState<'create' | 'edit'>('create');
  36. const [editingId, setEditingId] = useState<number | null>(null);
  37. const [form] = Form.useForm();
  38. const [searchForm] = Form.useForm();
  39. const [searchParams, setSearchParams] = useState({
  40. user_id: '',
  41. nickname: '',
  42. code: '',
  43. training_date: '',
  44. training_date_end: '',
  45. page: 1,
  46. limit: 10,
  47. });
  48. // 使用React Query获取提交记录列表
  49. const { data: recordsData, isLoading: isListLoading, refetch } = useQuery({
  50. queryKey: ['submissionRecords', searchParams],
  51. queryFn: () => SubmissionRecordsAPI.getSubmissionRecords({
  52. page: searchParams.page,
  53. pageSize: searchParams.limit,
  54. user_id: searchParams.user_id,
  55. nickname: searchParams.nickname,
  56. code: searchParams.code,
  57. training_date: searchParams.training_date
  58. }),
  59. placeholderData: {
  60. data: [],
  61. pagination: {
  62. current: 1,
  63. pageSize: 10,
  64. total: 0,
  65. totalPages: 1
  66. }
  67. }
  68. });
  69. const records = React.useMemo(() => (recordsData as SubmissionRecordListResponse)?.data || [], [recordsData]);
  70. const pagination = React.useMemo(() => ({
  71. current: (recordsData as SubmissionRecordListResponse)?.pagination?.current || 1,
  72. pageSize: (recordsData as SubmissionRecordListResponse)?.pagination?.pageSize || 10,
  73. total: (recordsData as SubmissionRecordListResponse)?.pagination?.total || 0,
  74. totalPages: (recordsData as SubmissionRecordListResponse)?.pagination?.totalPages || 1
  75. }), [recordsData]);
  76. // 获取单个提交记录
  77. const fetchRecord = async (id: number) => {
  78. try {
  79. const response = await SubmissionRecordsAPI.getSubmissionRecord(id);
  80. return response.data;
  81. } catch (error) {
  82. message.error('获取提交记录详情失败');
  83. return null;
  84. }
  85. };
  86. // 处理表单提交
  87. const handleSubmit = async (values: Omit<SubmissionRecord, 'id'>) => {
  88. try {
  89. const response = formMode === 'create'
  90. ? await SubmissionRecordsAPI.createSubmissionRecord(values)
  91. : await SubmissionRecordsAPI.updateSubmissionRecord(editingId!, values);
  92. message.success(formMode === 'create' ? '创建提交记录成功' : '更新提交记录成功');
  93. setModalVisible(false);
  94. form.resetFields();
  95. refetch();
  96. } catch (error) {
  97. message.error((error as Error).message);
  98. }
  99. };
  100. // 处理编辑
  101. const handleEdit = async (id: number) => {
  102. const record = await fetchRecord(id);
  103. if (record) {
  104. setFormMode('edit');
  105. setEditingId(id);
  106. form.setFieldsValue({
  107. ...record,
  108. training_date: record.training_date ? dayjs(record.training_date) : null
  109. });
  110. setModalVisible(true);
  111. }
  112. };
  113. // 处理删除
  114. const handleDelete = async (id: number) => {
  115. try {
  116. await SubmissionRecordsAPI.deleteSubmissionRecord(id);
  117. message.success('删除提交记录成功');
  118. refetch();
  119. } catch (error) {
  120. message.error((error as Error).message);
  121. }
  122. };
  123. // 处理搜索
  124. const handleSearch = async (values: any) => {
  125. try {
  126. queryClient.removeQueries({ queryKey: ['submissionRecords'] });
  127. setSearchParams({
  128. user_id: values.user_id || '',
  129. nickname: values.nickname || '',
  130. code: values.code || '',
  131. training_date: values.training_date?.[0]?.format('YYYY-MM-DD') || '',
  132. training_date_end: values.training_date?.[1]?.format('YYYY-MM-DD') || '',
  133. page: 1,
  134. limit: searchParams.limit,
  135. });
  136. } catch (error) {
  137. message.error('搜索失败');
  138. }
  139. };
  140. // 处理分页
  141. const handlePageChange = (page: number, pageSize?: number) => {
  142. setSearchParams(prev => ({
  143. ...prev,
  144. page,
  145. limit: pageSize || prev.limit,
  146. }));
  147. };
  148. // 处理添加
  149. const handleAdd = () => {
  150. setFormMode('create');
  151. setEditingId(null);
  152. form.resetFields();
  153. setModalVisible(true);
  154. };
  155. // 状态映射
  156. const statusOptions = getEnumOptions(SubmissionRecordStatus, SubmissionRecordStatusNameMap);
  157. // 表格列定义
  158. const columns = [
  159. {
  160. title: 'ID',
  161. dataIndex: 'id',
  162. key: 'id',
  163. width: 80,
  164. },
  165. {
  166. title: '用户ID',
  167. dataIndex: 'user_id',
  168. key: 'user_id',
  169. },
  170. {
  171. title: '昵称',
  172. dataIndex: 'nickname',
  173. key: 'nickname',
  174. },
  175. {
  176. title: '成绩',
  177. dataIndex: 'score',
  178. key: 'score',
  179. },
  180. {
  181. title: '代码',
  182. dataIndex: 'code',
  183. key: 'code',
  184. },
  185. {
  186. title: '训练日期',
  187. dataIndex: 'training_date',
  188. key: 'training_date',
  189. render: (date: string) => date ? dayjs(date).format('YYYY-MM-DD') : '-',
  190. },
  191. {
  192. title: '标记',
  193. dataIndex: 'mark',
  194. key: 'mark',
  195. },
  196. {
  197. title: '状态',
  198. dataIndex: 'status',
  199. key: 'status',
  200. render: (status: SubmissionRecordStatus) => {
  201. const statusText = SubmissionRecordStatusNameMap[status];
  202. const colorMap = {
  203. [SubmissionRecordStatus.PENDING]: 'orange',
  204. [SubmissionRecordStatus.APPROVED]: 'green',
  205. [SubmissionRecordStatus.REJECTED]: 'red'
  206. };
  207. return statusText ? (
  208. <Tag color={colorMap[status]}>{statusText}</Tag>
  209. ) : null;
  210. },
  211. },
  212. {
  213. title: '操作',
  214. key: 'action',
  215. render: (_: any, record: SubmissionRecord) => (
  216. <Space size="middle">
  217. <Button type="link" onClick={() => handleEdit(record.id)}>编辑</Button>
  218. <Popconfirm
  219. title="确定要删除这条记录吗?"
  220. onConfirm={() => handleDelete(record.id)}
  221. okText="确定"
  222. cancelText="取消"
  223. >
  224. <Button type="link" danger>删除</Button>
  225. </Popconfirm>
  226. </Space>
  227. ),
  228. },
  229. ];
  230. return (
  231. <div>
  232. <Card title="提交记录管理" className="mb-4">
  233. <Form
  234. form={searchForm}
  235. layout="inline"
  236. onFinish={handleSearch}
  237. style={{ marginBottom: '16px' }}
  238. >
  239. <Form.Item name="user_id" label="用户ID">
  240. <Input placeholder="要搜索的用户ID" />
  241. </Form.Item>
  242. <Form.Item name="nickname" label="昵称">
  243. <Input placeholder="要搜索的用户昵称" />
  244. </Form.Item>
  245. <Form.Item name="code" label="代码">
  246. <Input placeholder="要搜索的代码" />
  247. </Form.Item>
  248. <Form.Item name="training_date" label="训练日期">
  249. <DatePicker.RangePicker />
  250. </Form.Item>
  251. <Form.Item>
  252. <Space>
  253. <Button type="primary" htmlType="submit">
  254. 搜索
  255. </Button>
  256. <Button htmlType="reset" onClick={() => {
  257. searchForm.resetFields();
  258. setSearchParams({
  259. user_id: '',
  260. nickname: '',
  261. code: '',
  262. training_date: '',
  263. training_date_end: '',
  264. page: 1,
  265. limit: 10,
  266. });
  267. }}>
  268. 重置
  269. </Button>
  270. <Button type="primary" onClick={handleAdd}>
  271. 添加记录
  272. </Button>
  273. </Space>
  274. </Form.Item>
  275. </Form>
  276. <Table
  277. columns={columns}
  278. dataSource={records}
  279. rowKey="id"
  280. loading={{
  281. spinning: isListLoading,
  282. tip: '正在加载数据...',
  283. }}
  284. pagination={{
  285. current: pagination.current,
  286. pageSize: pagination.pageSize,
  287. total: pagination.total,
  288. onChange: handlePageChange,
  289. showSizeChanger: true,
  290. showTotal: (total) => `共 ${total} 条`,
  291. }}
  292. />
  293. </Card>
  294. <Modal
  295. title={formMode === 'create' ? '添加提交记录' : '编辑提交记录'}
  296. open={modalVisible}
  297. onOk={() => {
  298. form.validateFields()
  299. .then(values => {
  300. handleSubmit({
  301. ...values,
  302. training_date: values.training_date?.format('YYYY-MM-DD'),
  303. code: values.code || '',
  304. user_id: values.user_id || 0,
  305. score: values.score || 0,
  306. status: values.status || SubmissionRecordStatus.PENDING
  307. });
  308. })
  309. .catch(info => {
  310. console.log('表单验证失败:', info);
  311. });
  312. }}
  313. onCancel={() => setModalVisible(false)}
  314. width={800}
  315. okText="确定"
  316. cancelText="取消"
  317. destroyOnClose
  318. >
  319. <Form
  320. form={form}
  321. layout="vertical"
  322. >
  323. <Row gutter={16}>
  324. <Col span={12}>
  325. <Form.Item
  326. name="user_id"
  327. label="用户ID"
  328. rules={[{ required: true, message: '请输入用户ID' }]}
  329. >
  330. <Input placeholder="请输入用户ID" />
  331. </Form.Item>
  332. </Col>
  333. <Col span={12}>
  334. <Form.Item
  335. name="nickname"
  336. label="昵称"
  337. >
  338. <Input placeholder="请输入用户昵称" />
  339. </Form.Item>
  340. </Col>
  341. </Row>
  342. <Row gutter={16}>
  343. <Col span={12}>
  344. <Form.Item
  345. name="score"
  346. label="成绩"
  347. rules={[{ required: true, message: '请输入成绩' }]}
  348. >
  349. <Input type="number" placeholder="请输入成绩" />
  350. </Form.Item>
  351. </Col>
  352. <Col span={12}>
  353. <Form.Item
  354. name="training_date"
  355. label="训练日期"
  356. rules={[{ required: true, message: '请选择训练日期' }]}
  357. >
  358. <DatePicker style={{ width: '100%' }} />
  359. </Form.Item>
  360. </Col>
  361. </Row>
  362. <Form.Item
  363. name="code"
  364. label="代码"
  365. rules={[{ required: true, message: '请输入代码' }]}
  366. >
  367. <Input.TextArea rows={4} placeholder="请输入代码" />
  368. </Form.Item>
  369. <Form.Item
  370. name="mark"
  371. label="标记"
  372. >
  373. <Input placeholder="请输入标记" />
  374. </Form.Item>
  375. <Form.Item
  376. name="status"
  377. label="状态"
  378. >
  379. <Select options={statusOptions} />
  380. </Form.Item>
  381. </Form>
  382. </Modal>
  383. </div>
  384. );
  385. };