index.tsx 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316
  1. import {
  2. addUser,
  3. batchDeleteUsers,
  4. deleteUser,
  5. getUsers,
  6. oneClickClear,
  7. updateUser,
  8. updateUserPoints,
  9. } from '@/services/api';
  10. import { DeleteOutlined, PlusOutlined } from '@ant-design/icons';
  11. import type { ActionType, ProColumns } from '@ant-design/pro-components';
  12. import { ModalForm, ProFormDigit, ProFormSelect, ProFormText } from '@ant-design/pro-form';
  13. import { FooterToolbar, PageContainer, ProTable } from '@ant-design/pro-components';
  14. import { Button, FormInstance, message, Modal, Popconfirm } from 'antd';
  15. import React, { useRef, useState } from 'react';
  16. import UpdateUser from './components/UpdateUser';
  17. import UpdateUserPoints from './components/UpdateUserPoints';
  18. const UserList: React.FC = () => {
  19. const [createModalOpen, handleModalOpen] = useState<boolean>(false);
  20. const [updateModalOpen, handleUpdateModalOpen] = useState<boolean>(false);
  21. const actionRef = useRef<ActionType>();
  22. const [currentRow, setCurrentRow] = useState<API.UserItem>();
  23. const [selectedRows, setSelectedRows] = useState<API.UserItem[]>([]);
  24. const [pageSize, setPageSize] = useState<number>(10);
  25. const formRef = useRef<FormInstance>(null);
  26. const [updatePointsModalOpen, setUpdatePointsModalOpen] = useState(false);
  27. const handleDelete = async (record: API.UserItem) => {
  28. const res = await deleteUser(record._id);
  29. if (res.success) {
  30. message.success('用户已成功删除');
  31. actionRef.current?.reload();
  32. }
  33. };
  34. const handleBatchDelete = async (selectedRowKeys: string[]) => {
  35. if (selectedRowKeys.length === 0) return;
  36. const res = await batchDeleteUsers(selectedRowKeys);
  37. if (res.success) {
  38. message.success(`${res.message}`);
  39. actionRef.current?.reload();
  40. }
  41. };
  42. const handleClear = async () => {
  43. try {
  44. const result = await oneClickClear();
  45. if (result.success) {
  46. message.success('清空成功');
  47. actionRef.current?.reload();
  48. } else {
  49. message.error(result.error);
  50. }
  51. } catch (error) {
  52. message.error('操作失败');
  53. }
  54. };
  55. const columns: ProColumns<API.UserItem>[] = [
  56. {
  57. title: '用户名',
  58. dataIndex: 'username',
  59. valueType: 'text',
  60. },
  61. {
  62. title: '角色',
  63. dataIndex: 'role',
  64. valueType: 'select',
  65. valueEnum: {
  66. user: { text: '普通用户' },
  67. admin: { text: '管理员' },
  68. },
  69. },
  70. {
  71. title: '积分',
  72. dataIndex: 'points',
  73. valueType: 'digit',
  74. },
  75. {
  76. title: '密保问题',
  77. dataIndex: 'securityQuestion',
  78. valueType: 'text',
  79. },
  80. {
  81. title: '密保答案',
  82. dataIndex: 'securityAnswer',
  83. valueType: 'text',
  84. },
  85. {
  86. title: '创建时间',
  87. dataIndex: 'createdAt',
  88. valueType: 'dateTime',
  89. },
  90. {
  91. title: '最后登录时间',
  92. dataIndex: 'lastLoginAt',
  93. valueType: 'dateTime',
  94. },
  95. {
  96. title: '操作',
  97. dataIndex: 'option',
  98. valueType: 'option',
  99. render: (_, record) => [
  100. <a
  101. key="edit"
  102. onClick={() => {
  103. handleUpdateModalOpen(true);
  104. setCurrentRow(record);
  105. }}
  106. >
  107. 编辑用户
  108. </a>,
  109. <a
  110. key="editPoints"
  111. onClick={() => {
  112. setUpdatePointsModalOpen(true);
  113. setCurrentRow(record);
  114. }}
  115. >
  116. 修改积分
  117. </a>,
  118. <Popconfirm
  119. key="delete"
  120. title="确认删除"
  121. description={`您确定要删除用户 ${record.username} 吗?`}
  122. onConfirm={() => handleDelete(record)}
  123. okText="确定"
  124. cancelText="取消"
  125. >
  126. <a>删除</a>
  127. </Popconfirm>,
  128. ],
  129. },
  130. ];
  131. return (
  132. <PageContainer>
  133. <ProTable<API.UserItem, API.PageParams>
  134. actionRef={actionRef}
  135. rowKey="_id"
  136. search={{
  137. labelWidth: 120,
  138. }}
  139. toolBarRender={() => [
  140. <Button
  141. type="primary"
  142. key="primary"
  143. onClick={() => {
  144. formRef.current?.resetFields();
  145. handleModalOpen(true);
  146. }}
  147. >
  148. <PlusOutlined /> 新建用户
  149. </Button>,
  150. <Button
  151. danger
  152. onClick={() => {
  153. Modal.confirm({
  154. title: '清空确认',
  155. content: '确定要清空一年内未活跃用户的积分和记录?此操作不可恢复!',
  156. okText: '确认清空',
  157. cancelText: '取消',
  158. okButtonProps: {
  159. danger: true,
  160. },
  161. onOk: async () => {
  162. await handleClear();
  163. actionRef.current?.reload();
  164. },
  165. });
  166. }}
  167. >
  168. <DeleteOutlined /> 一键清空
  169. </Button>,
  170. ]}
  171. request={getUsers}
  172. columns={columns}
  173. pagination={{
  174. showSizeChanger: true,
  175. pageSize: pageSize,
  176. onChange: (page, pageSize) => {
  177. setPageSize(pageSize);
  178. if (actionRef.current) {
  179. actionRef.current.reload();
  180. }
  181. },
  182. }}
  183. rowSelection={{
  184. onChange: (_, selectedRows) => {
  185. setSelectedRows(selectedRows);
  186. },
  187. }}
  188. />
  189. {selectedRows?.length > 0 && (
  190. <FooterToolbar
  191. extra={
  192. <div>
  193. 已选择 <a style={{ fontWeight: 600 }}>{selectedRows.length}</a> 项 &nbsp;&nbsp;
  194. </div>
  195. }
  196. >
  197. <Button
  198. onClick={async () => {
  199. const selectedRowKeys = selectedRows.map((row) => row._id);
  200. await handleBatchDelete(selectedRowKeys);
  201. setSelectedRows([]);
  202. actionRef.current?.reloadAndRest?.();
  203. }}
  204. >
  205. 批量删除
  206. </Button>
  207. </FooterToolbar>
  208. )}
  209. <ModalForm
  210. title="新建用户"
  211. width="400px"
  212. open={createModalOpen}
  213. onOpenChange={handleModalOpen}
  214. formRef={formRef}
  215. onFinish={async (value) => {
  216. const res = await addUser(value as API.UserItem);
  217. if (res.success) {
  218. message.success('创建成功');
  219. handleModalOpen(false);
  220. if (actionRef.current) {
  221. actionRef.current.reload();
  222. }
  223. formRef.current?.resetFields();
  224. }
  225. }}
  226. >
  227. <ProFormText
  228. rules={[
  229. {
  230. required: true,
  231. message: '用户名是必填项',
  232. },
  233. ]}
  234. width="md"
  235. name="username"
  236. label="用户名"
  237. />
  238. <ProFormText.Password
  239. rules={[
  240. {
  241. required: true,
  242. message: '密码是必填项',
  243. },
  244. ]}
  245. width="md"
  246. name="password"
  247. label="密码"
  248. />
  249. <ProFormSelect
  250. rules={[
  251. {
  252. required: true,
  253. message: '角色是必填项',
  254. },
  255. ]}
  256. width="md"
  257. name="role"
  258. label="角色"
  259. valueEnum={{
  260. user: '普通用户',
  261. admin: '管理员',
  262. }}
  263. />
  264. <ProFormDigit width="md" name="points" label="积分" initialValue={0} />
  265. </ModalForm>
  266. <UpdateUser
  267. onSubmit={async (value) => {
  268. const res = await updateUser(value._id, value as API.UserItem);
  269. if (res.success) {
  270. message.success('修改成功');
  271. handleUpdateModalOpen(false);
  272. setCurrentRow(undefined);
  273. if (actionRef.current) {
  274. actionRef.current.reload();
  275. }
  276. }
  277. }}
  278. onCancel={() => {
  279. handleUpdateModalOpen(false);
  280. }}
  281. updateUserModalOpen={updateModalOpen}
  282. values={currentRow || {}}
  283. />
  284. {/* 修改积分 */}
  285. <UpdateUserPoints
  286. onSubmit={async (value: API.UpdateUserPointsParams) => {
  287. const res = await updateUserPoints(value.userId, value.points, value.reason);
  288. if (res.success) {
  289. message.success('积分更新成功');
  290. setUpdatePointsModalOpen(false);
  291. setCurrentRow(undefined);
  292. if (actionRef.current) {
  293. actionRef.current.reload();
  294. }
  295. }
  296. }}
  297. onCancel={() => {
  298. setUpdatePointsModalOpen(false);
  299. }}
  300. updatePointsModalOpen={updatePointsModalOpen}
  301. currentUser={currentRow || {}}
  302. />
  303. </PageContainer>
  304. );
  305. };
  306. export default UserList;