index.tsx 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  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. sorter: true,
  75. },
  76. {
  77. title: '密保问题',
  78. dataIndex: 'securityQuestion',
  79. valueType: 'text',
  80. },
  81. {
  82. title: '密保答案',
  83. dataIndex: 'securityAnswer',
  84. valueType: 'text',
  85. },
  86. {
  87. title: '创建时间',
  88. dataIndex: 'createdAt',
  89. valueType: 'dateTime',
  90. sorter: true,
  91. },
  92. {
  93. title: '最后登录时间',
  94. dataIndex: 'lastLoginAt',
  95. valueType: 'dateTime',
  96. sorter: true,
  97. },
  98. {
  99. title: '登录IP',
  100. dataIndex: 'loginIp',
  101. valueType: 'text',
  102. },
  103. {
  104. title: '最后登录地址',
  105. dataIndex: 'lastLoginAddress',
  106. valueType: 'text',
  107. },
  108. {
  109. title: '操作',
  110. dataIndex: 'option',
  111. valueType: 'option',
  112. render: (_, record) => [
  113. <a
  114. key="edit"
  115. onClick={() => {
  116. handleUpdateModalOpen(true);
  117. setCurrentRow(record);
  118. }}
  119. >
  120. 编辑用户
  121. </a>,
  122. <a
  123. key="editPoints"
  124. onClick={() => {
  125. setUpdatePointsModalOpen(true);
  126. setCurrentRow(record);
  127. }}
  128. >
  129. 修改积分
  130. </a>,
  131. <Popconfirm
  132. key="delete"
  133. title="确认删除"
  134. description={`您确定要删除用户 ${record.username} 吗?`}
  135. onConfirm={() => handleDelete(record)}
  136. okText="确定"
  137. cancelText="取消"
  138. >
  139. <a>删除</a>
  140. </Popconfirm>,
  141. ],
  142. },
  143. ];
  144. return (
  145. <PageContainer>
  146. <ProTable<API.UserItem, API.PageParams>
  147. actionRef={actionRef}
  148. rowKey="_id"
  149. search={{
  150. labelWidth: 120,
  151. }}
  152. toolBarRender={() => [
  153. <Button
  154. type="primary"
  155. key="primary"
  156. onClick={() => {
  157. formRef.current?.resetFields();
  158. handleModalOpen(true);
  159. }}
  160. >
  161. <PlusOutlined /> 新建用户
  162. </Button>,
  163. <Button
  164. danger
  165. onClick={() => {
  166. Modal.confirm({
  167. title: '清空确认',
  168. content: '确定要清空一年内未活跃用户的积分和记录?此操作不可恢复!',
  169. okText: '确认清空',
  170. cancelText: '取消',
  171. okButtonProps: {
  172. danger: true,
  173. },
  174. onOk: async () => {
  175. await handleClear();
  176. actionRef.current?.reload();
  177. },
  178. });
  179. }}
  180. >
  181. <DeleteOutlined /> 一键清空
  182. </Button>,
  183. ]}
  184. request={async (params, sort, filter) => {
  185. return getUsers(params, sort)
  186. }}
  187. columns={columns}
  188. pagination={{
  189. showSizeChanger: true,
  190. pageSize: pageSize,
  191. onChange: (page, pageSize) => {
  192. setPageSize(pageSize);
  193. if (actionRef.current) {
  194. actionRef.current.reload();
  195. }
  196. },
  197. }}
  198. rowSelection={{
  199. onChange: (_, selectedRows) => {
  200. setSelectedRows(selectedRows);
  201. },
  202. }}
  203. />
  204. {selectedRows?.length > 0 && (
  205. <FooterToolbar
  206. extra={
  207. <div>
  208. 已选择 <a style={{ fontWeight: 600 }}>{selectedRows.length}</a> 项 &nbsp;&nbsp;
  209. </div>
  210. }
  211. >
  212. <Button
  213. onClick={async () => {
  214. const selectedRowKeys = selectedRows.map((row) => row._id);
  215. await handleBatchDelete(selectedRowKeys);
  216. setSelectedRows([]);
  217. actionRef.current?.reloadAndRest?.();
  218. }}
  219. >
  220. 批量删除
  221. </Button>
  222. </FooterToolbar>
  223. )}
  224. <ModalForm
  225. title="新建用户"
  226. width="400px"
  227. open={createModalOpen}
  228. onOpenChange={handleModalOpen}
  229. formRef={formRef}
  230. onFinish={async (value) => {
  231. const res = await addUser(value as API.UserItem);
  232. if (res.success) {
  233. message.success('创建成功');
  234. handleModalOpen(false);
  235. if (actionRef.current) {
  236. actionRef.current.reload();
  237. }
  238. formRef.current?.resetFields();
  239. }
  240. }}
  241. >
  242. <ProFormText
  243. rules={[
  244. {
  245. required: true,
  246. message: '用户名是必填项',
  247. },
  248. ]}
  249. width="md"
  250. name="username"
  251. label="用户名"
  252. />
  253. <ProFormText.Password
  254. rules={[
  255. {
  256. required: true,
  257. message: '密码是必填项',
  258. },
  259. ]}
  260. width="md"
  261. name="password"
  262. label="密码"
  263. />
  264. <ProFormSelect
  265. rules={[
  266. {
  267. required: true,
  268. message: '角色是必填项',
  269. },
  270. ]}
  271. width="md"
  272. name="role"
  273. label="角色"
  274. valueEnum={{
  275. user: '普通用户',
  276. admin: '管理员',
  277. }}
  278. />
  279. <ProFormDigit width="md" name="points" label="积分" initialValue={0} />
  280. </ModalForm>
  281. <UpdateUser
  282. onSubmit={async (value) => {
  283. const res = await updateUser(value._id, value as API.UserItem);
  284. if (res.success) {
  285. message.success('修改成功');
  286. handleUpdateModalOpen(false);
  287. setCurrentRow(undefined);
  288. if (actionRef.current) {
  289. actionRef.current.reload();
  290. }
  291. }
  292. }}
  293. onCancel={() => {
  294. handleUpdateModalOpen(false);
  295. }}
  296. updateUserModalOpen={updateModalOpen}
  297. values={currentRow || {}}
  298. />
  299. {/* 修改积分 */}
  300. <UpdateUserPoints
  301. onSubmit={async (value: API.UpdateUserPointsParams) => {
  302. const res = await updateUserPoints(value.userId, value.points, value.reason);
  303. if (res.success) {
  304. message.success('积分更新成功');
  305. setUpdatePointsModalOpen(false);
  306. setCurrentRow(undefined);
  307. if (actionRef.current) {
  308. actionRef.current.reload();
  309. }
  310. }
  311. }}
  312. onCancel={() => {
  313. setUpdatePointsModalOpen(false);
  314. }}
  315. updatePointsModalOpen={updatePointsModalOpen}
  316. currentUser={currentRow || {}}
  317. />
  318. </PageContainer>
  319. );
  320. };
  321. export default UserList;