Ver código fonte

# feat:用户管理模块的增删改查

yang yi 1 semana atrás
pai
commit
24d708772a
4 arquivos alterados com 446 adições e 2 exclusões
  1. 1 1
      src/util/axios-instance.ts
  2. 426 1
      src/view/UserView.vue
  3. 4 0
      tsconfig.app.json
  4. 15 0
      vite.config.ts

+ 1 - 1
src/util/axios-instance.ts

@@ -34,7 +34,7 @@ axiosInstance.interceptors.request.use(
 axiosInstance.interceptors.response.use(
     (response: AxiosResponse) => {
         // 根据后端返回格式调整
-        if (response.data.code !== 0) {
+        if (response.data.code !== 0 && response.data.code !== 200) {
             return Promise.reject(new Error(response.data.message || '请求失败'));
         }
         return response.data;

+ 426 - 1
src/view/UserView.vue

@@ -1,11 +1,436 @@
 <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>

+ 4 - 0
tsconfig.app.json

@@ -2,6 +2,10 @@
   "extends": "@vue/tsconfig/tsconfig.dom.json",
   "compilerOptions": {
     "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
+    "baseUrl": ".",
+    "paths": {
+      "@/*": ["./src/*"]
+    },
 
     /* Linting */
     "strict": true,

+ 15 - 0
vite.config.ts

@@ -1,7 +1,22 @@
 import { defineConfig } from 'vite'
 import vue from '@vitejs/plugin-vue'
+import { fileURLToPath, URL } from 'node:url'
 
 // https://vite.dev/config/
 export default defineConfig({
   plugins: [vue()],
+  resolve: {
+    alias: {
+      '@': fileURLToPath(new URL('./src', import.meta.url))
+    }
+  },
+  server: {
+    proxy: {
+      '/api': {
+        target: 'http://localhost:8080',
+        changeOrigin: true,
+        rewrite: (path) => path.replace(/^\/api/, '')
+      }
+    }
+  }
 })