UserView.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436
  1. <script setup lang="ts">
  2. import { getUserController } from '@/api/user-controller'
  3. import type { QueryByPageParams, UserDto, UserVo } from '@/api/models'
  4. import { ElMessage, ElMessageBox, type FormInstance, type FormRules, ElIcon } from 'element-plus'
  5. import { Plus, RefreshRight, Search, Delete, Edit } from '@element-plus/icons-vue'
  6. import { ref, reactive, computed } from 'vue'
  7. const api = getUserController()
  8. // 搜索表单
  9. const searchFormRef = ref<FormInstance>()
  10. const searchForm = reactive({
  11. username: '',
  12. account: '',
  13. role: '',
  14. enable: 1
  15. })
  16. // 用户表单(新增/编辑)
  17. const dialogVisible = ref(false)
  18. const dialogTitle = ref('新增用户')
  19. const formRef = ref<FormInstance>()
  20. const userForm = reactive<Partial<UserDto>>({
  21. id: undefined,
  22. username: '',
  23. account: '',
  24. password: '',
  25. role: ''
  26. })
  27. const formRules: FormRules = {
  28. username: [
  29. { required: true, message: '请输入用户名', trigger: 'blur' },
  30. { min: 2, max: 32, message: '用户名长度需在 2 到 32 个字符之间', trigger: 'blur' }
  31. ],
  32. account: [
  33. { required: true, message: '请输入账号', trigger: 'blur' },
  34. { min: 4, max: 20, message: '账号长度需在 4 到 20 个字符之间', trigger: 'blur' }
  35. ],
  36. password: [
  37. { required: true, message: '请输入密码', trigger: 'blur' },
  38. { min: 4, max: 20, message: '密码长度需在 4 到 20 个字符之间', trigger: 'blur' }
  39. ],
  40. role: [
  41. { required: true, message: '请选择角色', trigger: 'change' },
  42. { min: 4, max: 32, message: '角色长度需在 4 到 32 个字符之间', trigger: 'change' }
  43. ]
  44. }
  45. // 表格数据
  46. const tableData = ref<UserVo[]>([])
  47. const loading = ref(false)
  48. const selectedRows = ref<UserVo[]>([])
  49. // 分页
  50. const pagination = reactive({
  51. pageNum: 1,
  52. pageSize: 10,
  53. total: 0
  54. })
  55. // 角色选项
  56. const roleOptions = [
  57. { label: '管理员', value: 'admin' },
  58. { label: '普通用户', value: 'user' },
  59. // { label: '访客', value: 'guest' }
  60. ]
  61. // 是否编辑模式
  62. const isEdit = computed(() => !!userForm.id)
  63. // 搜索
  64. const handleSearch = () => {
  65. pagination.pageNum = 1
  66. fetchData()
  67. }
  68. // 重置搜索
  69. const handleReset = () => {
  70. searchFormRef.value?.resetFields()
  71. handleSearch()
  72. }
  73. // 获取数据
  74. const fetchData = async () => {
  75. loading.value = true
  76. try {
  77. const condition: QueryByPageParams = {
  78. username: searchForm.username || '',
  79. account: searchForm.account || '',
  80. role: searchForm.role || '',
  81. enable: searchForm.enable,
  82. pageNum: pagination.pageNum,
  83. pageSize: pagination.pageSize
  84. }
  85. const res = await api.queryByPage(condition)
  86. const pageData = res?.data
  87. tableData.value = pageData?.data || []
  88. pagination.total = pageData?.total || 0
  89. } catch (err) {
  90. ElMessage.error('获取数据失败')
  91. tableData.value = []
  92. } finally {
  93. loading.value = false
  94. }
  95. }
  96. // 翻页
  97. const handlePageChange = (page: number) => {
  98. pagination.pageNum = page
  99. fetchData()
  100. }
  101. // 每页条数变化
  102. const handleSizeChange = (size: number) => {
  103. pagination.pageSize = size
  104. pagination.pageNum = 1
  105. fetchData()
  106. }
  107. // 打开新增
  108. const handleAdd = () => {
  109. dialogTitle.value = '新增用户'
  110. Object.assign(userForm, { id: undefined, username: '', account: '', password: '', role: '' })
  111. dialogVisible.value = true
  112. }
  113. // 打开编辑
  114. const handleEdit = async (row: UserVo) => {
  115. dialogTitle.value = '编辑用户'
  116. try {
  117. const id = row.id || ''
  118. const res = await api.queryById(id)
  119. if (res?.code === 0 || res?.code === 200) {
  120. Object.assign(userForm, {
  121. id: res.data?.id,
  122. username: res.data?.username,
  123. account: res.data?.account,
  124. password: '',
  125. role: res.data?.role
  126. })
  127. dialogVisible.value = true
  128. }
  129. } catch {
  130. ElMessage.error('获取用户信息失败')
  131. }
  132. }
  133. // 保存
  134. const handleSave = async () => {
  135. await formRef.value?.validate(async (valid) => {
  136. if (!valid) return
  137. try {
  138. const baseData = {
  139. username: userForm.username!,
  140. account: userForm.account!,
  141. role: userForm.role!
  142. }
  143. if (isEdit.value) {
  144. const data: Partial<UserDto> = {
  145. id: userForm.id,
  146. ...baseData,
  147. ...(userForm.password ? { password: userForm.password } : {})
  148. }
  149. const res = await api.edit(data as UserDto)
  150. if (res?.code === 0 || res?.code === 200) {
  151. ElMessage.success('修改成功')
  152. dialogVisible.value = false
  153. fetchData()
  154. } else {
  155. ElMessage.error(res?.message || '修改失败')
  156. }
  157. } else {
  158. if (!userForm.password) {
  159. ElMessage.warning('请输入密码')
  160. return
  161. }
  162. const data: UserDto = {
  163. ...baseData,
  164. password: userForm.password!
  165. }
  166. const res = await api.add(data)
  167. if (res?.code === 0 || res?.code === 200) {
  168. ElMessage.success('新增成功')
  169. dialogVisible.value = false
  170. fetchData()
  171. } else {
  172. ElMessage.error(res?.message || '新增失败')
  173. }
  174. }
  175. } catch {
  176. ElMessage.error(isEdit.value ? '修改失败' : '新增失败')
  177. }
  178. })
  179. }
  180. // 对话框关闭回调,重置表单验证
  181. const handleDialogClosed = () => {
  182. formRef.value?.resetFields()
  183. }
  184. // 删除
  185. const handleDelete = (row: UserVo) => {
  186. ElMessageBox.confirm('确认删除该用户吗?', '提示', {
  187. type: 'warning'
  188. }).then(async () => {
  189. try {
  190. const ids = row.id ? [row.id] : []
  191. const res = await api.deleteById({ ids })
  192. if (res?.code === 0 || res?.code === 200) {
  193. ElMessage.success('删除成功')
  194. fetchData()
  195. } else {
  196. ElMessage.error(res?.message || '删除失败')
  197. }
  198. } catch {
  199. ElMessage.error('删除失败')
  200. }
  201. }).catch(() => {})
  202. }
  203. // 批量删除
  204. const handleBatchDelete = () => {
  205. if (!selectedRows.value.length) {
  206. ElMessage.warning('请选择要删除的数据')
  207. return
  208. }
  209. ElMessageBox.confirm(`确认删除选中的 ${selectedRows.value.length} 条数据吗?`, '提示', {
  210. type: 'warning'
  211. }).then(async () => {
  212. try {
  213. const ids = selectedRows.value.map(row => row.id).filter((id): id is string => id !== undefined)
  214. const res = await api.deleteById({ ids })
  215. if (res?.code === 0 || res?.code === 200) {
  216. ElMessage.success('删除成功')
  217. selectedRows.value = []
  218. fetchData()
  219. } else {
  220. ElMessage.error(res?.message || '删除失败')
  221. }
  222. } catch {
  223. ElMessage.error('删除失败')
  224. }
  225. }).catch(() => {})
  226. }
  227. // 切换启用状态
  228. const handleToggleEnable = async (row: UserVo, newEnable: number) => {
  229. const oldEnable = row.enable
  230. row.enable = newEnable
  231. try {
  232. const res = await api.updateUserStatus({ id: row.id!, enable: newEnable })
  233. if (res?.code === 0 || res?.code === 200) {
  234. ElMessage.success('状态更新成功')
  235. } else {
  236. row.enable = oldEnable
  237. ElMessage.error(res?.message || '状态更新失败')
  238. }
  239. } catch {
  240. row.enable = oldEnable
  241. ElMessage.error('状态更新失败')
  242. }
  243. }
  244. // 表格选择
  245. const handleSelectionChange = (rows: UserVo[]) => {
  246. selectedRows.value = rows
  247. }
  248. // 初始加载
  249. fetchData()
  250. </script>
  251. <template>
  252. <div class="user-view">
  253. <!-- 搜索区域 -->
  254. <el-card class="search-card" shadow="never">
  255. <el-form ref="searchFormRef" :model="searchForm" inline>
  256. <el-form-item label="用户名" prop="username">
  257. <el-input v-model="searchForm.username" placeholder="请输入用户名" clearable style="width: 180px" />
  258. </el-form-item>
  259. <el-form-item label="账号" prop="account">
  260. <el-input v-model="searchForm.account" placeholder="请输入账号" clearable style="width: 180px" />
  261. </el-form-item>
  262. <el-form-item label="角色" prop="role">
  263. <el-select v-model="searchForm.role" placeholder="请选择角色" clearable style="width: 150px">
  264. <el-option v-for="item in roleOptions" :key="item.value" :label="item.label" :value="item.value" />
  265. </el-select>
  266. </el-form-item>
  267. <el-form-item label="状态" prop="enable">
  268. <el-select v-model="searchForm.enable" placeholder="请选择状态" clearable style="width: 120px">
  269. <el-option label="启用" :value="1" />
  270. <el-option label="禁用" :value="0" />
  271. </el-select>
  272. </el-form-item>
  273. <el-form-item>
  274. <el-button type="primary" @click="handleSearch">
  275. <el-icon class="el-icon--left"><Search /></el-icon>搜索
  276. </el-button>
  277. <el-button @click="handleReset">
  278. <el-icon class="el-icon--left"><RefreshRight /></el-icon>重置
  279. </el-button>
  280. </el-form-item>
  281. </el-form>
  282. </el-card>
  283. <!-- 操作按钮 -->
  284. <div class="toolbar">
  285. <el-button type="primary" @click="handleAdd">
  286. <el-icon class="el-icon--left"><Plus /></el-icon>新增
  287. </el-button>
  288. <el-button type="danger" :disabled="!selectedRows.length" @click="handleBatchDelete">
  289. <el-icon class="el-icon--left"><Delete /></el-icon>批量删除
  290. </el-button>
  291. </div>
  292. <!-- 表格 -->
  293. <el-card class="table-card" shadow="never">
  294. <el-table
  295. v-loading="loading"
  296. :data="tableData"
  297. border
  298. stripe
  299. @selection-change="handleSelectionChange"
  300. style="width: 100%"
  301. >
  302. <el-table-column type="selection" width="55" align="center" />
  303. <el-table-column prop="id" label="ID" width="80" align="center" />
  304. <el-table-column prop="username" label="用户名" min-width="120" align="center" />
  305. <el-table-column prop="account" label="账号" min-width="150" align="center" />
  306. <el-table-column prop="role" label="角色" width="120" align="center">
  307. <template #default="{ row }">
  308. <el-tag :type="row.role === 'admin' ? 'danger' : row.role === 'user' ? 'success' : 'info'">
  309. {{ roleOptions.find(t => t.value === row.role)?.label || row.role }}
  310. </el-tag>
  311. </template>
  312. </el-table-column>
  313. <el-table-column prop="enable" label="状态" width="100" align="center">
  314. <template #default="{ row }">
  315. <el-switch
  316. :model-value="row.enable"
  317. :active-value="1"
  318. :inactive-value="0"
  319. @change="(val: number) => handleToggleEnable(row, val)"
  320. />
  321. </template>
  322. </el-table-column>
  323. <el-table-column prop="avatar" label="头像" width="100" align="center">
  324. <template #default="{ row }">
  325. <el-avatar v-if="row.avatar" :src="row.avatar" :size="40" />
  326. <el-avatar v-else :size="40">{{ row.username?.charAt(0) || '?' }}</el-avatar>
  327. </template>
  328. </el-table-column>
  329. <el-table-column label="操作" width="180" align="center" fixed="right">
  330. <template #default="{ row }">
  331. <el-button type="primary" link @click="handleEdit(row)">
  332. <el-icon><Edit /></el-icon>编辑
  333. </el-button>
  334. <el-button type="danger" link @click="handleDelete(row)">
  335. <el-icon><Delete /></el-icon>删除
  336. </el-button>
  337. </template>
  338. </el-table-column>
  339. </el-table>
  340. <!-- 分页 -->
  341. <div class="pagination">
  342. <el-pagination
  343. v-model:current-page="pagination.pageNum"
  344. v-model:page-size="pagination.pageSize"
  345. :page-sizes="[10, 20, 50, 100]"
  346. :total="pagination.total"
  347. layout="total, sizes, prev, pager, next, jumper"
  348. @current-change="handlePageChange"
  349. @size-change="handleSizeChange"
  350. />
  351. </div>
  352. </el-card>
  353. <!-- 新增/编辑弹窗 -->
  354. <el-dialog v-model="dialogVisible" :title="dialogTitle" width="500px" draggable @closed="handleDialogClosed">
  355. <el-form ref="formRef" :model="userForm" :rules="formRules" label-width="80px">
  356. <el-form-item label="用户名" prop="username">
  357. <el-input v-model="userForm.username" placeholder="请输入用户名" />
  358. </el-form-item>
  359. <el-form-item label="账号" prop="account">
  360. <el-input v-model="userForm.account" placeholder="请输入账号" :disabled="isEdit" />
  361. </el-form-item>
  362. <el-form-item label="密码" :prop="isEdit ? '' : 'password'">
  363. <el-input v-model="userForm.password" type="password" placeholder="请输入密码" show-password />
  364. <span v-if="isEdit" class="tip">留空则不修改密码</span>
  365. </el-form-item>
  366. <el-form-item label="角色" prop="role">
  367. <el-select v-model="userForm.role" placeholder="请选择角色" style="width: 100%">
  368. <el-option v-for="item in roleOptions" :key="item.value" :label="item.label" :value="item.value" />
  369. </el-select>
  370. </el-form-item>
  371. </el-form>
  372. <template #footer>
  373. <el-button @click="dialogVisible = false">取消</el-button>
  374. <el-button type="primary" @click="handleSave">确定</el-button>
  375. </template>
  376. </el-dialog>
  377. </div>
  378. </template>
  379. <style scoped>
  380. .user-view {
  381. padding: 20px;
  382. }
  383. .search-card {
  384. margin-bottom: 16px;
  385. }
  386. .toolbar {
  387. margin-bottom: 16px;
  388. }
  389. .table-card {
  390. margin-bottom: 16px;
  391. }
  392. .pagination {
  393. display: flex;
  394. justify-content: flex-end;
  395. margin-top: 16px;
  396. }
  397. .tip {
  398. font-size: 12px;
  399. color: #909399;
  400. line-height: 1.4;
  401. }
  402. </style>