| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436 |
- <script setup lang="ts">
- import { getUserController } from '@/api/user-controller'
- import type { QueryByPageParams, UserDto, UserVo } from '@/api/models'
- import { ElMessage, ElMessageBox, type FormInstance, type FormRules, ElIcon } from 'element-plus'
- import { Plus, RefreshRight, Search, Delete, Edit } from '@element-plus/icons-vue'
- import { ref, reactive, computed } from 'vue'
- const api = getUserController()
- // 搜索表单
- const searchFormRef = ref<FormInstance>()
- const searchForm = reactive({
- username: '',
- account: '',
- role: '',
- enable: 1
- })
- // 用户表单(新增/编辑)
- const dialogVisible = ref(false)
- const dialogTitle = ref('新增用户')
- const formRef = ref<FormInstance>()
- const userForm = reactive<Partial<UserDto>>({
- id: undefined,
- username: '',
- account: '',
- password: '',
- role: ''
- })
- const formRules: FormRules = {
- username: [
- { required: true, message: '请输入用户名', trigger: 'blur' },
- { min: 2, max: 32, message: '用户名长度需在 2 到 32 个字符之间', trigger: 'blur' }
- ],
- account: [
- { required: true, message: '请输入账号', trigger: 'blur' },
- { min: 4, max: 20, message: '账号长度需在 4 到 20 个字符之间', trigger: 'blur' }
- ],
- password: [
- { required: true, message: '请输入密码', trigger: 'blur' },
- { min: 4, max: 20, message: '密码长度需在 4 到 20 个字符之间', trigger: 'blur' }
- ],
- role: [
- { required: true, message: '请选择角色', trigger: 'change' },
- { min: 4, max: 32, message: '角色长度需在 4 到 32 个字符之间', trigger: 'change' }
- ]
- }
- // 表格数据
- const tableData = ref<UserVo[]>([])
- const loading = ref(false)
- const selectedRows = ref<UserVo[]>([])
- // 分页
- const pagination = reactive({
- pageNum: 1,
- pageSize: 10,
- total: 0
- })
- // 角色选项
- const roleOptions = [
- { label: '管理员', value: 'admin' },
- { label: '普通用户', value: 'user' },
- // { label: '访客', value: 'guest' }
- ]
- // 是否编辑模式
- const isEdit = computed(() => !!userForm.id)
- // 搜索
- const handleSearch = () => {
- pagination.pageNum = 1
- fetchData()
- }
- // 重置搜索
- const handleReset = () => {
- searchFormRef.value?.resetFields()
- handleSearch()
- }
- // 获取数据
- const fetchData = async () => {
- loading.value = true
- try {
- const condition: QueryByPageParams = {
- username: searchForm.username || '',
- account: searchForm.account || '',
- role: searchForm.role || '',
- enable: searchForm.enable,
- pageNum: pagination.pageNum,
- pageSize: pagination.pageSize
- }
- const res = await api.queryByPage(condition)
- const pageData = res?.data
- tableData.value = pageData?.data || []
- pagination.total = pageData?.total || 0
- } catch (err) {
- ElMessage.error('获取数据失败')
- tableData.value = []
- } finally {
- loading.value = false
- }
- }
- // 翻页
- const handlePageChange = (page: number) => {
- pagination.pageNum = page
- fetchData()
- }
- // 每页条数变化
- const handleSizeChange = (size: number) => {
- pagination.pageSize = size
- pagination.pageNum = 1
- fetchData()
- }
- // 打开新增
- const handleAdd = () => {
- dialogTitle.value = '新增用户'
- Object.assign(userForm, { id: undefined, username: '', account: '', password: '', role: '' })
- dialogVisible.value = true
- }
- // 打开编辑
- const handleEdit = async (row: UserVo) => {
- dialogTitle.value = '编辑用户'
- try {
- const id = row.id || ''
- const res = await api.queryById(id)
- if (res?.code === 0 || res?.code === 200) {
- Object.assign(userForm, {
- id: res.data?.id,
- username: res.data?.username,
- account: res.data?.account,
- password: '',
- role: res.data?.role
- })
- dialogVisible.value = true
- }
- } catch {
- ElMessage.error('获取用户信息失败')
- }
- }
- // 保存
- const handleSave = async () => {
- await formRef.value?.validate(async (valid) => {
- if (!valid) return
- try {
- const baseData = {
- username: userForm.username!,
- account: userForm.account!,
- role: userForm.role!
- }
-
- if (isEdit.value) {
- const data: Partial<UserDto> = {
- id: userForm.id,
- ...baseData,
- ...(userForm.password ? { password: userForm.password } : {})
- }
- const res = await api.edit(data as UserDto)
- if (res?.code === 0 || res?.code === 200) {
- ElMessage.success('修改成功')
- dialogVisible.value = false
- fetchData()
- } else {
- ElMessage.error(res?.message || '修改失败')
- }
- } else {
- if (!userForm.password) {
- ElMessage.warning('请输入密码')
- return
- }
- const data: UserDto = {
- ...baseData,
- password: userForm.password!
- }
- const res = await api.add(data)
- if (res?.code === 0 || res?.code === 200) {
- ElMessage.success('新增成功')
- dialogVisible.value = false
- fetchData()
- } else {
- ElMessage.error(res?.message || '新增失败')
- }
- }
- } catch {
- ElMessage.error(isEdit.value ? '修改失败' : '新增失败')
- }
- })
- }
- // 对话框关闭回调,重置表单验证
- const handleDialogClosed = () => {
- formRef.value?.resetFields()
- }
- // 删除
- const handleDelete = (row: UserVo) => {
- ElMessageBox.confirm('确认删除该用户吗?', '提示', {
- type: 'warning'
- }).then(async () => {
- try {
- const ids = row.id ? [row.id] : []
- const res = await api.deleteById({ ids })
- if (res?.code === 0 || res?.code === 200) {
- ElMessage.success('删除成功')
- fetchData()
- } else {
- ElMessage.error(res?.message || '删除失败')
- }
- } catch {
- ElMessage.error('删除失败')
- }
- }).catch(() => {})
- }
- // 批量删除
- const handleBatchDelete = () => {
- if (!selectedRows.value.length) {
- ElMessage.warning('请选择要删除的数据')
- return
- }
- ElMessageBox.confirm(`确认删除选中的 ${selectedRows.value.length} 条数据吗?`, '提示', {
- type: 'warning'
- }).then(async () => {
- try {
- const ids = selectedRows.value.map(row => row.id).filter((id): id is string => id !== undefined)
- const res = await api.deleteById({ ids })
- if (res?.code === 0 || res?.code === 200) {
- ElMessage.success('删除成功')
- selectedRows.value = []
- fetchData()
- } else {
- ElMessage.error(res?.message || '删除失败')
- }
- } catch {
- ElMessage.error('删除失败')
- }
- }).catch(() => {})
- }
- // 切换启用状态
- const handleToggleEnable = async (row: UserVo, newEnable: number) => {
- const oldEnable = row.enable
- row.enable = newEnable
- try {
- const res = await api.updateUserStatus({ id: row.id!, enable: newEnable })
- if (res?.code === 0 || res?.code === 200) {
- ElMessage.success('状态更新成功')
- } else {
- row.enable = oldEnable
- ElMessage.error(res?.message || '状态更新失败')
- }
- } catch {
- row.enable = oldEnable
- ElMessage.error('状态更新失败')
- }
- }
- // 表格选择
- const handleSelectionChange = (rows: UserVo[]) => {
- selectedRows.value = rows
- }
- // 初始加载
- fetchData()
- </script>
- <template>
- <div class="user-view">
- <!-- 搜索区域 -->
- <el-card class="search-card" shadow="never">
- <el-form ref="searchFormRef" :model="searchForm" inline>
- <el-form-item label="用户名" prop="username">
- <el-input v-model="searchForm.username" placeholder="请输入用户名" clearable style="width: 180px" />
- </el-form-item>
- <el-form-item label="账号" prop="account">
- <el-input v-model="searchForm.account" placeholder="请输入账号" clearable style="width: 180px" />
- </el-form-item>
- <el-form-item label="角色" prop="role">
- <el-select v-model="searchForm.role" placeholder="请选择角色" clearable style="width: 150px">
- <el-option v-for="item in roleOptions" :key="item.value" :label="item.label" :value="item.value" />
- </el-select>
- </el-form-item>
- <el-form-item label="状态" prop="enable">
- <el-select v-model="searchForm.enable" placeholder="请选择状态" clearable style="width: 120px">
- <el-option label="启用" :value="1" />
- <el-option label="禁用" :value="0" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary" @click="handleSearch">
- <el-icon class="el-icon--left"><Search /></el-icon>搜索
- </el-button>
- <el-button @click="handleReset">
- <el-icon class="el-icon--left"><RefreshRight /></el-icon>重置
- </el-button>
- </el-form-item>
- </el-form>
- </el-card>
- <!-- 操作按钮 -->
- <div class="toolbar">
- <el-button type="primary" @click="handleAdd">
- <el-icon class="el-icon--left"><Plus /></el-icon>新增
- </el-button>
- <el-button type="danger" :disabled="!selectedRows.length" @click="handleBatchDelete">
- <el-icon class="el-icon--left"><Delete /></el-icon>批量删除
- </el-button>
- </div>
- <!-- 表格 -->
- <el-card class="table-card" shadow="never">
- <el-table
- v-loading="loading"
- :data="tableData"
- border
- stripe
- @selection-change="handleSelectionChange"
- style="width: 100%"
- >
- <el-table-column type="selection" width="55" align="center" />
- <el-table-column prop="id" label="ID" width="80" align="center" />
- <el-table-column prop="username" label="用户名" min-width="120" align="center" />
- <el-table-column prop="account" label="账号" min-width="150" align="center" />
- <el-table-column prop="role" label="角色" width="120" align="center">
- <template #default="{ row }">
- <el-tag :type="row.role === 'admin' ? 'danger' : row.role === 'user' ? 'success' : 'info'">
- {{ roleOptions.find(t => t.value === row.role)?.label || row.role }}
- </el-tag>
- </template>
- </el-table-column>
- <el-table-column prop="enable" label="状态" width="100" align="center">
- <template #default="{ row }">
- <el-switch
- :model-value="row.enable"
- :active-value="1"
- :inactive-value="0"
- @change="(val: number) => handleToggleEnable(row, val)"
- />
- </template>
- </el-table-column>
- <el-table-column prop="avatar" label="头像" width="100" align="center">
- <template #default="{ row }">
- <el-avatar v-if="row.avatar" :src="row.avatar" :size="40" />
- <el-avatar v-else :size="40">{{ row.username?.charAt(0) || '?' }}</el-avatar>
- </template>
- </el-table-column>
- <el-table-column label="操作" width="180" align="center" fixed="right">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleEdit(row)">
- <el-icon><Edit /></el-icon>编辑
- </el-button>
- <el-button type="danger" link @click="handleDelete(row)">
- <el-icon><Delete /></el-icon>删除
- </el-button>
- </template>
- </el-table-column>
- </el-table>
- <!-- 分页 -->
- <div class="pagination">
- <el-pagination
- v-model:current-page="pagination.pageNum"
- v-model:page-size="pagination.pageSize"
- :page-sizes="[10, 20, 50, 100]"
- :total="pagination.total"
- layout="total, sizes, prev, pager, next, jumper"
- @current-change="handlePageChange"
- @size-change="handleSizeChange"
- />
- </div>
- </el-card>
- <!-- 新增/编辑弹窗 -->
- <el-dialog v-model="dialogVisible" :title="dialogTitle" width="500px" draggable @closed="handleDialogClosed">
- <el-form ref="formRef" :model="userForm" :rules="formRules" label-width="80px">
- <el-form-item label="用户名" prop="username">
- <el-input v-model="userForm.username" placeholder="请输入用户名" />
- </el-form-item>
- <el-form-item label="账号" prop="account">
- <el-input v-model="userForm.account" placeholder="请输入账号" :disabled="isEdit" />
- </el-form-item>
- <el-form-item label="密码" :prop="isEdit ? '' : 'password'">
- <el-input v-model="userForm.password" type="password" placeholder="请输入密码" show-password />
- <span v-if="isEdit" class="tip">留空则不修改密码</span>
- </el-form-item>
- <el-form-item label="角色" prop="role">
- <el-select v-model="userForm.role" placeholder="请选择角色" style="width: 100%">
- <el-option v-for="item in roleOptions" :key="item.value" :label="item.label" :value="item.value" />
- </el-select>
- </el-form-item>
- </el-form>
- <template #footer>
- <el-button @click="dialogVisible = false">取消</el-button>
- <el-button type="primary" @click="handleSave">确定</el-button>
- </template>
- </el-dialog>
- </div>
- </template>
- <style scoped>
- .user-view {
- padding: 20px;
- }
- .search-card {
- margin-bottom: 16px;
- }
- .toolbar {
- margin-bottom: 16px;
- }
- .table-card {
- margin-bottom: 16px;
- }
- .pagination {
- display: flex;
- justify-content: flex-end;
- margin-top: 16px;
- }
- .tip {
- font-size: 12px;
- color: #909399;
- line-height: 1.4;
- }
- </style>
|