feat:添加admin相关接口

This commit is contained in:
2025-11-13 15:05:54 +08:00
parent 11cb40c86a
commit 067e0539ba
58 changed files with 18736 additions and 273 deletions
+21 -12
View File
@@ -15,6 +15,11 @@
<el-button type="primary" @click="handleAdd">
<el-icon><plus /></el-icon>新增域名
</el-button>
<el-button type="success" @click="getList">
<el-icon><Refresh /></el-icon>刷新
</el-button>
<el-button type="danger" :disabled="!selectedRows.length" @click="handleBatchDelete">
<el-icon><delete /></el-icon>批量删除
</el-button>
@@ -30,10 +35,10 @@
style="width: 100%"
>
<el-table-column type="selection" width="55" />
<el-table-column prop="Id" label="ID" width="80" />
<el-table-column prop="Domain" label="域名" min-width="200" >
<el-table-column prop="id" label="ID" width="80" />
<el-table-column prop="domain" label="域名" min-width="200" >
<template #default="{ row }">
<el-link :href="`http://${row.Domain}`" target="_blank" type="primary">{{ row.Domain }}</el-link>
<el-link :href="`http://${row.domain}`" target="_blank" type="primary">{{ row.domain }}</el-link>
</template>
</el-table-column>
<el-table-column prop="CreatedAt" label="创建时间" width="180" :formatter="parseCreatedAt" />
@@ -223,9 +228,14 @@ const handleDelete = (row) => {
type: 'warning'
}).then(async () => {
try {
await deleteDomain(row.Id)
ElMessage.success('删除成功')
getList()
const res = await deleteDomain({domain_id: row.id})
console.log(res)
if (res.code === 200) {
ElMessage.success('删除成功')
getList()
} else {
ElMessage.error(res.data.message || '删除失败')
}
} catch (error) {
console.error('删除域名失败:', error)
ElMessage.error('删除失败')
@@ -239,10 +249,11 @@ const handleBatchDelete = () => {
ElMessage.warning('请选择要删除的域名')
return
}
const ids = selectedRows.value.map(item => item.Id)
const ids = selectedRows.value.map(item => item.id)
const domains = selectedRows.value.map(item => item.domain).join('、')
console.log('id数据:',ids)
ElMessageBox.confirm(`确认删除以下域名吗?\n${domains}`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
@@ -257,9 +268,7 @@ const handleBatchDelete = () => {
console.error('批量删除域名失败:', error)
ElMessage.error('批量删除失败')
}
await batchDeleteDomain(ids)
ElMessage.success('批量删除成功')
getList()
} catch (error) {
console.error('批量删除域名失败:', error)
ElMessage.error('批量删除失败')
+778
View File
@@ -0,0 +1,778 @@
<template>
<div class="permission-admin-container">
<!-- 搜索和操作栏 -->
<el-card class="filter-container" shadow="never">
<el-form :inline="true" :model="queryParams" class="search-form">
<el-form-item label="类型">
<el-select v-model="queryParams.owner_type" placeholder="请选择类型" clearable style="width: 150px" @change="handleOwnerTypeChange">
<el-option label="用户" value="user" />
<el-option label="组" value="group" />
</el-select>
</el-form-item>
<el-form-item label="用户" v-if="queryParams.owner_type === 'user'">
<!-- <el-select v-model="queryParams.user_id" placeholder="请选择用户" clearable filterable style="width: 200px">
<el-option v-for="item in userOptions" :key="item.UserId" :label="`${item.UserName} (ID: ${item.UserId})`" :value="item.UserId" />
</el-select> -->
<div class="user_selector-inline">
<el-tag v-if="queryParams.user_id" type="primary" closable @close="clearQueryUser" style="margin-right: 8px;">
{{ getQueryUserName() }}
</el-tag>
<el-button type="primary" plain @click="openQueryUserSelector" size="default">
<el-icon><User /></el-icon>
{{ queryParams.user_id ? '重新选择' : '选择用户' }}
</el-button>
</div>
</el-form-item>
<el-form-item label="管理员组" v-if="queryParams.owner_type === 'group'">
<el-select v-model="queryParams.admin_group_id" placeholder="请选择管理员组" clearable filterable style="width: 200px">
<el-option v-for="item in adminGroupOptions" :key="item.id" :label="`${item.name} (ID: ${item.id})`" :value="item.id" />
</el-select>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="handleQuery">
<el-icon><Search /></el-icon>查询
</el-button>
<el-button @click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
<div class="action-bar">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>分配权限
</el-button>
<el-button type="success" @click="fetchAdminPermissionList">
<el-icon><Refresh/></el-icon>刷新
</el-button>
</div>
</el-card>
<!-- 管理员权限列表 -->
<el-card class="table-container" shadow="never">
<el-table
v-loading="loading"
:data="adminPermissionList"
style="width: 100%"
>
<el-table-column prop="id" label="ID" width="80" />
<el-table-column label="拥有者类型" width="120">
<template #default="{ row }">
<el-tag :type="row.ownerType === 'user' ? 'primary' : 'success'">
{{ row.ownerType === 'user' ? '用户' : '组' }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="拥有者" width="180">
<template #default="{ row }">
<span v-if="row.ownerType === 'user'">用户ID: {{ row.userId }}</span>
<span v-else>管理员组ID: {{ row.groupId }}</span>
</template>
</el-table-column>
<el-table-column prop="permissionId" label="路径权限ID" width="120" />
<el-table-column label="权限路径" min-width="200" show-overflow-tooltip>
<template #default="{ row }">
{{ row.permission?.path || '-' }}
</template>
</el-table-column>
<el-table-column label="权限名称" width="150" show-overflow-tooltip>
<template #default="{ row }">
{{ row.permission?.name || '-' }}
</template>
</el-table-column>
<el-table-column prop="weight" label="权重" width="100" />
<el-table-column label="权限类型" width="120">
<template #default="{ row }">
<el-tag :type="getPermissionTypeTag(row.permissionType)">
{{ getPermissionTypeText(row.permissionType) }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="过期时间" width="180" show-overflow-tooltip>
<template #default="{ row }">
{{ formatDate(row.expireAt) }}
</template>
</el-table-column>
<el-table-column label="操作" width="200" fixed="right">
<template #default="{ row }">
<el-button type="primary" link @click="handleEdit(row)">编辑</el-button>
<el-button type="danger" link @click="handleDelete(row)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<el-pagination
v-model:current-page="queryParams.page"
v-model:page-size="queryParams.count"
:page-sizes="[10, 20, 50, 100]"
layout="total, sizes, prev, pager, next, jumper"
:total="total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
background
class="pagination"
/>
</el-card>
<!-- 用户选择弹窗 -->
<el-dialog
v-model="userSelectorVisible"
title="选择用户"
width="800px"
class="user-selector-dialog"
>
<!-- 搜索栏 -->
<div class="selector-search">
<el-input
v-model="userSearchParams.key"
placeholder="搜索用户名或ID"
clearable
@keyup.enter="searchUsers"
style="width: 300px; margin-right: 12px"
>
<template #prefix>
<el-icon><Search /></el-icon>
</template>
</el-input>
<el-button type="primary" @click="searchUsers">
<el-icon><Search /></el-icon>
搜索
</el-button>
<el-button @click="resetUserSearch">重置</el-button>
</div>
<!-- 用户表格 -->
<el-table
v-loading="userSelectorLoading"
:data="userSelectorList"
highlight-current-row
@current-change="handleUserSelectChange"
style="width: 100%; margin-top: 16px"
:height="400"
>
<el-table-column type="index" label="序号" width="60" />
<el-table-column prop="UserId" label="用户ID" width="100" />
<el-table-column prop="UserName" label="用户名" min-width="150" />
<el-table-column prop="Email" label="邮箱" min-width="180" />
<el-table-column label="状态" width="100">
<template #default="{ row }">
<el-tag :type="row.Status === 1 ? 'success' : 'danger'" size="small">
{{ row.Status === 1 ? '正常' : '禁用' }}
</el-tag>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<el-pagination
v-model:current-page="userSearchParams.page"
v-model:page-size="userSearchParams.count"
:page-sizes="[10, 20, 50, 100]"
layout="total, sizes, prev, pager, next, jumper"
:total="userSelectorTotal"
@size-change="handleUserSelectorSizeChange"
@current-change="handleUserSelectorPageChange"
background
class="selector-pagination"
/>
<template #footer>
<el-button @click="userSelectorVisible = false">取消</el-button>
<el-button type="primary" @click="confirmUserSelection" :disabled="!selectedUserTemp">
确定选择
</el-button>
</template>
</el-dialog>
<!-- 分配权限对话框 -->
<el-dialog
v-model="dialogVisible"
title="分配管理员权限"
width="700px"
>
<el-form
ref="permissionFormRef"
:model="permissionForm"
:rules="permissionRules"
label-width="140px"
>
<el-form-item label="权限绑定类型" prop="owner_type">
<el-select v-model="permissionForm.owner_type" placeholder="请选择权限绑定类型" style="width: 100%" @change="handleFormOwnerTypeChange" :disabled="permissionForm.id">
<el-option label="用户" value="user" />
<el-option label="组" value="group" />
</el-select>
<div class="form-tip">如果是 user 则填写 user_id如果是 group 则填写 admin_group_id</div>
</el-form-item>
<el-form-item label="用户" prop="user_id" v-if="permissionForm.owner_type === 'user'" >
<div class="user_selector-inline">
<el-tag v-if="permissionForm.user_id" type="primary" closable @close="clearFormUser" style="margin-right: 8px;">
{{ getFormUserName() }}
</el-tag>
<el-button type="primary" plain @click="openFormUserSelector" size="default" :disabled="permissionForm.user_id">
<el-icon><User /></el-icon>
{{ permissionForm.user_id ? '重新选择' : '选择用户' }}
</el-button>
</div>
</el-form-item>
<el-form-item label="管理员组" prop="admin_group_id" v-if="permissionForm.owner_type === 'group'">
<el-select v-model="permissionForm.admin_group_id" placeholder="请选择管理员组" filterable style="width: 100%">
<el-option v-for="item in adminGroupOptions" :key="item.id" :label="`${item.name} (ID: ${item.id})`" :value="item.id" />
</el-select>
</el-form-item>
<el-form-item label="路径权限" prop="permission_id">
<div style="display: flex; gap: 8px;">
<el-select
v-model="permissionForm.permission_id"
placeholder="请选择路径权限"
filterable
style="flex: 1"
:loading="permissionLoading"
>
<el-option
v-for="item in permissionOptions"
:key="item.id"
:value="item.id"
>
<div style="display: flex; justify-content: space-between; align-items: center;">
<span>
<el-tag v-if="item.method" :type="getMethodTag(item.method)" size="small" style="margin-right: 8px;">{{ item.method }}</el-tag>
{{ item.path }}
</span>
<span style="color: #999; font-size: 12px; margin-left: 12px;">{{ item.note || item.name || `ID: ${item.id}` }}</span>
</div>
</el-option>
</el-select>
<el-button @click="fetchPermissionList" :loading="permissionLoading" :icon="Refresh">刷新</el-button>
</div>
<div class="form-tip"> {{ permissionOptions.length }} 个路径权限可选</div>
</el-form-item>
<el-form-item label="权重" prop="weight">
<el-input-number v-model="permissionForm.weight" placeholder="请输入权重" :min="0" style="width: 100%" />
<div class="form-tip">权重默认 10</div>
</el-form-item>
<el-form-item label="权限类型" prop="permission_type">
<el-select v-model="permissionForm.permission_type" placeholder="请选择权限类型" style="width: 100%">
<el-option label="无权限 (none)" :value="0" />
<el-option label="禁止 (prohibit)" :value="1" />
<el-option label="读取 (read)" :value="2" />
<el-option label="写入 (write)" :value="3" />
<el-option label="全部 (all)" :value="4" />
</el-select>
<div class="form-tip">0 none / 1 prohibit / 2 read / 3 write / 4 all</div>
</el-form-item>
<el-form-item label="过期时间" prop="expire_at">
<el-date-picker
v-model="permissionForm.expire_at"
type="datetime"
placeholder="请选择过期时间"
value-format="YYYY-MM-DD HH:mm:ss"
style="width: 100%"
/>
<div class="form-tip">权限过期时间 不填写则不会过期</div>
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="submitForm">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Plus, Search, Refresh, User } from '@element-plus/icons-vue'
import {
getPermissionListByAdmin,
addPermissionAdmin,
updatePermissionAdmin,
deletePermissionAdmin,
getPermissionList
} from '@/api/admin/Permission'
import { getUserList } from '@/api/admin/user'
import { getAdminGroupList } from '@/api/admin/group'
import { formatDate ,timeToTimestamp} from '@/utils/tool'
const selectorType = ref('query')
const userSelectorVisible = ref(false)
const userSelectorList = ref([])
const userSelectorTotal = ref(0)
const userSearchParams = reactive({
key: '',
page: 1,
count: 10
})
const selectedUserTemp = ref(null)
const userSelectorLoading = ref(false)
const UserOptions = ref([])
// 查询参数
const queryParams = reactive({
owner_type: '',
user_id: undefined,
admin_group_id: undefined,
page: 1,
count: 10
})
// 清除查询用户
const clearQueryUser = () => {
queryParams.user_id = undefined
}
// 获取查询用户名称
const getQueryUserName = () => {
const user = UserOptions.value.find(u => u.UserId === queryParams.user_id)
return user ? `${user.UserName} (ID: ${user.UserId})` : `用户ID: ${queryParams.user_id}`
}
// 表单:清除用户
const clearFormUser = () => {
permissionForm.user_id = undefined
}
// 表单:获取显示名称
const getFormUserName = () => {
const user = UserOptions.value.find(u => u.UserId === permissionForm.user_id)
return user ? `${user.UserName} (ID: ${user.UserId})` : `用户ID: ${permissionForm.user_id}`
}
// 确认用户选择
const confirmUserSelection = () => {
if (!selectedUserTemp.value) {
ElMessage.warning('请选择一个用户')
return
}
if (selectorType.value === 'query') {
// 查询表单选择,仅影响查询条件
queryParams.user_id = selectedUserTemp.value.UserId
} else if (selectorType.value === 'form') {
// 表单选择,仅影响表单
permissionForm.user_id = selectedUserTemp.value.UserId
}
// 将选中的用户添加到 UserOptions 中(如果不存在)
if (!UserOptions.value.find(u => u.UserId === selectedUserTemp.value.UserId)) {
UserOptions.value.push(selectedUserTemp.value)
}
userSelectorVisible.value = false
ElMessage.success('用户选择成功')
}
// 打开查询用户选择器
const openQueryUserSelector = () => {
selectorType.value = 'query'
userSelectorVisible.value = true
selectedUserTemp.value = null
userSearchParams.key = ''
userSearchParams.page = 1
fetchUserSelectorList()
}
// 打开表单用户选择器
const openFormUserSelector = () => {
selectorType.value = 'form'
userSelectorVisible.value = true
selectedUserTemp.value = null
userSearchParams.key = ''
userSearchParams.page = 1
fetchUserSelectorList()
}
// 用户选择变化
const handleUserSelectChange = (row) => {
selectedUserTemp.value = row
}
// 搜索用户
const searchUsers = () => {
userSearchParams.page = 1
fetchUserSelectorList()
}
// 用户选择器分页
const handleUserSelectorSizeChange = (size) => {
userSearchParams.count = size
fetchUserSelectorList()
}
const handleUserSelectorPageChange = (page) => {
userSearchParams.page = page
fetchUserSelectorList()
}
// 获取用户选择器列表
const fetchUserSelectorList = async () => {
userSelectorLoading.value = true
try {
const res = await getUserList(userSearchParams)
console.log('用户选择器列表:', res.data)
if (res.data.code === 200) {
userSelectorList.value = res.data.data?.data || []
userSelectorTotal.value = res.data.data?.all_count || 0
}
} catch (error) {
console.error('获取用户列表失败:', error)
ElMessage.error('获取用户列表失败')
} finally {
userSelectorLoading.value = false
}
}
// 管理员权限表单
const permissionForm = reactive({
id: undefined,
user_id: undefined,
admin_group_id: undefined,
owner_type: '',
permission_id: undefined,
weight: 10,
permission_type: 4,
expire_at: ''
})
const permissionRules = {
owner_type: [
{ required: true, message: '请选择权限的绑定类型', trigger: 'change' }
],
permission_id: [
{ required: true, message: '请输入权限ID', trigger: 'blur' }
],
permission_type: [
{ required: true, message: '请选择权限类型', trigger: 'change' }
]
}
// 状态数据
const loading = ref(false)
const adminPermissionList = ref([])
const total = ref(0)
const dialogVisible = ref(false)
const permissionFormRef = ref(null)
// const userOptions = ref([])
// const UserOptions = ref([])
const adminGroupOptions = ref([])
const permissionOptions = ref([])
const permissionLoading = ref(false)
// 获取方法标签颜色
const getMethodTag = (method) => {
const tagMap = {
'GET': 'success',
'POST': 'primary',
'PUT': 'warning',
'DELETE': 'danger',
'PATCH': 'info'
}
return tagMap[method?.toUpperCase()] || 'info'
}
// 获取权限类型标签颜色
const getPermissionTypeTag = (type) => {
const tagMap = {
0: 'info', // none
1: 'danger', // prohibit
2: 'warning', // read
3: 'primary', // write
4: 'success' // all
}
return tagMap[type] || 'info'
}
// 获取权限类型文本
const getPermissionTypeText = (type) => {
const textMap = {
0: '无权限',
1: '禁止',
2: '读取',
3: '写入',
4: '全部'
}
return textMap[type] || '未知'
}
// 获取管理员权限列表
const fetchAdminPermissionList = async () => {
if (!queryParams.owner_type) {
ElMessage.warning('请先选择类型')
return
}
loading.value = true
try {
const params = { ...queryParams }
// 清除空参数
Object.keys(params).forEach(key => {
if (params[key] === '' || params[key] === null || params[key] === undefined) {
delete params[key]
}
})
const res = await getPermissionListByAdmin(params)
console.log('管理员权限列表数据:', res.data)
if (res.data.code === 200) {
adminPermissionList.value = res.data.data || []
total.value = res.data.data.length || 0
} else {
ElMessage.error(res.data.message || '获取管理员权限列表失败')
}
} catch (error) {
console.error('获取管理员权限列表失败:', error)
ElMessage.error('获取管理员权限列表失败')
} finally {
loading.value = false
}
}
// 查询
const handleQuery = () => {
queryParams.page = 1
fetchAdminPermissionList()
}
// 重置查询
const resetQuery = () => {
queryParams.user_id = undefined
queryParams.admin_group_id = undefined
queryParams.owner_type = ''
queryParams.page = 1
adminPermissionList.value = []
total.value = 0
}
// 类型变化时清空关联字段
const handleOwnerTypeChange = () => {
if (queryParams.owner_type === 'user') {
queryParams.admin_group_id = undefined
} else if (queryParams.owner_type === 'group') {
queryParams.user_id = undefined
}
}
// 表单类型变化时清空关联字段
const handleFormOwnerTypeChange = () => {
if (permissionForm.owner_type === 'user') {
permissionForm.admin_group_id = undefined
} else if (permissionForm.owner_type === 'group') {
permissionForm.user_id = undefined
}
}
// 分页
const handleSizeChange = (size) => {
queryParams.count = size
fetchAdminPermissionList()
}
const handleCurrentChange = (page) => {
queryParams.page = page
fetchAdminPermissionList()
}
// 分配权限
const handleAdd = () => {
dialogVisible.value = true
Object.assign(permissionForm, {
id: undefined,
user_id: undefined,
admin_group_id: undefined,
owner_type: '',
permission_id: undefined,
weight: 10,
permission_type: 4,
expire_at: ''
})
permissionFormRef.value?.resetFields()
}
// 编辑权限
const handleEdit = (row) => {
// 处理过期时间
let expireTime = ''
if (row.expireAt && row.expireAt !== '0001-01-01T00:00:00Z' && row.expireAt !== null) {
const date = new Date(row.expireAt)
if (!isNaN(date.getTime())) {
expireTime = date.toISOString().slice(0, 19).replace('T', ' ')
}
}
Object.assign(permissionForm, {
id: row.id,
user_id: row.userId,
admin_group_id: row.groupId,
owner_type: row.ownerType,
permission_id: row.permissionId,
weight: row.weight,
permission_type: row.permissionType,
expire_at: expireTime
})
dialogVisible.value = true
}
// 删除权限
const handleDelete = (row) => {
ElMessageBox.confirm(`确认删除该权限吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
try {
const res = await deletePermissionAdmin({ id: row.id })
if (res.data.code === 200) {
ElMessage.success('删除成功')
fetchAdminPermissionList()
}
} catch (error) {
console.error('删除失败:', error)
ElMessage.error(error.response?.data?.message || '删除失败')
}
}).catch(() => {})
}
// 提交表单
const submitForm = () => {
permissionFormRef.value?.validate(async (valid) => {
if (valid) {
try {
const submitData = {
owner_type: permissionForm.owner_type,
permission_id: Number(permissionForm.permission_id),
weight: Number(permissionForm.weight),
permission_type: Number(permissionForm.permission_type)
}
// 根据 owner_type 添加对应的 ID
if (permissionForm.owner_type === 'user') {
submitData.user_id = Number(permissionForm.user_id)
} else if (permissionForm.owner_type === 'group') {
submitData.admin_group_id = Number(permissionForm.admin_group_id)
}
// 将过期时间转换为时间戳(秒)
if (permissionForm.expire_at) {
submitData.expire_at = timeToTimestamp(permissionForm.expire_at)
}
// 如果是编辑,添加ID
if (permissionForm.id) {
submitData.id = permissionForm.id
}
console.log('提交管理员权限数据:', submitData)
let res
if (permissionForm.id) {
res = await updatePermissionAdmin(submitData)
} else {
res = await addPermissionAdmin(submitData)
}
if (res.data.code === 200) {
ElMessage.success(permissionForm.id ? '修改成功' : '分配成功')
dialogVisible.value = false
fetchAdminPermissionList()
} else {
ElMessage.error(res.data.message || '操作失败')
}
} catch (error) {
console.error('操作失败:', error)
ElMessage.error(error.response?.data?.message || '操作失败')
}
}
})
}
// 获取用户列表
const fetchUserList = async () => {
try {
const res = await getUserList({
page: 1,
count: 10000,
key: ''
})
if (res.data.code === 200) {
UserOptions.value = res.data.data?.data || []
}
} catch (error) {
console.error('获取用户列表失败:', error)
}
}
// 获取管理员组列表
const fetchAdminGroupList = async () => {
try {
const res = await getAdminGroupList({
page: 1,
count: 1000
})
if (res.data.code === 200) {
adminGroupOptions.value = res.data.data?.data || []
}
} catch (error) {
console.error('获取管理员组列表失败:', error)
}
}
// 获取路径权限列表
const fetchPermissionList = async () => {
permissionLoading.value = true
try {
const res = await getPermissionList({
page: 1,
count: 10000
})
if (res.data.code === 200) {
permissionOptions.value = res.data.data?.list || []
console.log('路径权限列表加载成功,共', permissionOptions.value.length, '条')
if (dialogVisible.value) {
ElMessage.success(`已加载 ${permissionOptions.value.length} 个路径权限`)
}
} else {
ElMessage.error(res.data.message || '获取路径权限列表失败')
}
} catch (error) {
console.error('获取路径权限列表失败:', error)
ElMessage.error('获取路径权限列表失败')
} finally {
permissionLoading.value = false
}
}
// 初始化
onMounted(() => {
fetchUserList()
fetchAdminGroupList()
fetchPermissionList()
})
</script>
<style scoped>
.permission-admin-container {
padding: 0;
}
.filter-container {
margin-bottom: 20px;
border-radius: 8px;
}
.search-form {
margin-bottom: 15px;
}
.action-bar {
display: flex;
gap: 12px;
}
.table-container {
border-radius: 8px;
}
.pagination {
margin-top: 24px;
justify-content: flex-end;
}
.form-tip {
font-size: 12px;
color: #999;
margin-top: 4px;
line-height: 1.5;
}
</style>
+281
View File
@@ -0,0 +1,281 @@
<template>
<div class="permission-route-container">
<!-- 搜索和操作栏 -->
<el-card class="filter-container" shadow="never">
<el-form :inline="true" :model="queryParams" class="search-form">
<el-form-item label="关键词">
<el-input v-model="queryParams.key" placeholder="请输入关键词搜索" clearable style="width: 250px" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="handleQuery">
<el-icon><Search /></el-icon>查询
</el-button>
<el-button @click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
<div class="action-bar">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>新增路由权限
</el-button>
<el-button type="success" @click="fetchPermissionList">
<el-icon><Refresh /></el-icon>刷新
</el-button>
</div>
</el-card>
<!-- 路由权限列表 -->
<el-card class="table-container" shadow="never">
<el-table
v-loading="loading"
:data="permissionList"
style="width: 100%"
>
<el-table-column prop="id" label="ID" width="80" />
<el-table-column prop="name" label="权限名称" min-width="200" />
<el-table-column prop="path" label="路由路径" min-width="300" />
<el-table-column prop="note" label="说明" min-width="250" show-overflow-tooltip />
<el-table-column label="操作" width="200" fixed="right">
<template #default="{ row }">
<el-button type="primary" link @click="handleEdit(row)">编辑</el-button>
<el-button type="danger" link @click="handleDelete(row)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<el-pagination
v-model:current-page="queryParams.page"
v-model:page-size="queryParams.count"
:page-sizes="[10, 20, 50, 100]"
layout="total, sizes, prev, pager, next, jumper"
:total="total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
background
class="pagination"
/>
</el-card>
<!-- 路由权限表单对话框 -->
<el-dialog
v-model="dialogVisible"
:title="dialogType === 'add' ? '新增路由权限' : '编辑路由权限'"
width="600px"
>
<el-form
ref="permissionFormRef"
:model="permissionForm"
:rules="permissionRules"
label-width="120px"
>
<el-form-item label="权限名称" prop="name">
<el-input v-model="permissionForm.name" placeholder="请输入权限名称" />
</el-form-item>
<el-form-item label="路由路径" prop="path">
<el-input v-model="permissionForm.path" placeholder="请输入路由路径,如: /api/v1/admin/..." />
</el-form-item>
<el-form-item label="说明" prop="note">
<el-input v-model="permissionForm.note" type="textarea" :rows="3" placeholder="请输入说明" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="submitForm">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Plus, Search } from '@element-plus/icons-vue'
import {
getPermissionList,
addPermissionInfo,
updatePermissionInfo,
deletePermissionInfo
} from '@/api/admin/Permission'
// 查询参数
const queryParams = reactive({
key: '',
page: 1,
count: 10
})
// 路由权限表单
const permissionForm = reactive({
id: undefined,
path: '',
name: '',
note: ''
})
const permissionRules = {
path: [
{ required: true, message: '请输入路由路径', trigger: 'blur' }
],
name: [
{ required: true, message: '请输入权限名称', trigger: 'blur' }
]
}
// 状态数据
const loading = ref(false)
const permissionList = ref([])
const total = ref(0)
const dialogVisible = ref(false)
const dialogType = ref('add')
const permissionFormRef = ref(null)
// 获取路由权限列表
const fetchPermissionList = async () => {
loading.value = true
try {
const res = await getPermissionList(queryParams)
console.log('路由权限列表数据:', res.data)
if (res.data.code === 200) {
permissionList.value = res.data.data.list || []
total.value = res.data.data.all_count || 0
}
} catch (error) {
console.error('获取路由权限列表失败:', error)
ElMessage.error('获取路由权限列表失败')
} finally {
loading.value = false
}
}
// 查询
const handleQuery = () => {
queryParams.page = 1
fetchPermissionList()
}
// 重置查询
const resetQuery = () => {
queryParams.key = ''
queryParams.page = 1
fetchPermissionList()
}
// 分页
const handleSizeChange = (size) => {
queryParams.count = size
fetchPermissionList()
}
const handleCurrentChange = (page) => {
queryParams.page = page
fetchPermissionList()
}
// 新增路由权限
const handleAdd = () => {
dialogType.value = 'add'
dialogVisible.value = true
Object.assign(permissionForm, {
id: undefined,
path: '',
name: '',
note: ''
})
permissionFormRef.value?.resetFields()
}
// 编辑路由权限
const handleEdit = (row) => {
dialogType.value = 'edit'
dialogVisible.value = true
Object.assign(permissionForm, {
id: row.id,
path: row.path,
name: row.name,
note: row.note || ''
})
}
// 删除路由权限
const handleDelete = (row) => {
ElMessageBox.confirm(`确认删除路由权限 ${row.name} 吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
try {
const res = await deletePermissionInfo({ id: row.id })
if (res.data.code === 200) {
ElMessage.success('删除成功')
fetchPermissionList()
}
} catch (error) {
console.error('删除失败:', error)
ElMessage.error(error.response?.data?.message || '删除失败')
}
}).catch(() => {})
}
// 提交表单
const submitForm = () => {
permissionFormRef.value?.validate(async (valid) => {
if (valid) {
try {
console.log('提交路由权限数据:', permissionForm)
let res
if (dialogType.value === 'add') {
res = await addPermissionInfo(permissionForm)
} else {
res = await updatePermissionInfo(permissionForm)
}
if (res.data.code === 200) {
ElMessage.success(dialogType.value === 'add' ? '新增成功' : '修改成功')
dialogVisible.value = false
fetchPermissionList()
}
} catch (error) {
console.error('操作失败:', error)
ElMessage.error(error.response?.data?.message || '操作失败')
}
}
})
}
// 初始化
onMounted(() => {
fetchPermissionList()
})
</script>
<style scoped>
.permission-route-container {
padding: 0;
}
.filter-container {
margin-bottom: 20px;
border-radius: 8px;
}
.search-form {
margin-bottom: 15px;
}
.action-bar {
display: flex;
gap: 12px;
}
.table-container {
border-radius: 8px;
}
.pagination {
margin-top: 24px;
justify-content: flex-end;
}
</style>
+535
View File
@@ -0,0 +1,535 @@
<template>
<div class="setting-container">
<!-- 搜索和操作栏 -->
<el-card class="filter-container" shadow="never">
<el-form :inline="true" :model="queryParams" class="search-form">
<el-form-item label="配置组">
<el-select v-model="queryParams.group_id" placeholder="请选择配置组" clearable style="width: 200px" @change="handleQuery">
<el-option
v-for="group in groupList"
:key="group.id"
:label="group.name"
:value="group.id"
/>
</el-select>
</el-form-item>
<el-form-item label="关键词筛选">
<el-input v-model="queryParams.key" placeholder="请输入关键词" clearable style="width: 200px" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="handleQuery">
<el-icon><Search /></el-icon>查询
</el-button>
<el-button @click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
<div class="action-bar">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>新增配置
</el-button>
<el-button type="danger" :disabled="!selectedRows.length" @click="handleBatchDelete">
<el-icon><Delete /></el-icon>批量删除
</el-button>
</div>
</el-card>
<!-- 配置列表 -->
<el-card class="table-container" shadow="never">
<el-table
v-loading="loading"
:data="settingList"
@selection-change="handleSelectionChange"
style="width: 100%"
>
<el-table-column type="selection" width="55" />
<el-table-column prop="id" label="ID" width="80" />
<el-table-column prop="name" label="名称" min-width="150" />
<el-table-column prop="value" label="值" min-width="200" show-overflow-tooltip>
<template #default="{ row }">
<span v-if="row.type === 'bool'">{{ row.value ? '' : '' }}</span>
<span v-else>{{ row.value }}</span>
</template>
</el-table-column>
<el-table-column prop="type" label="类型" width="100">
<template #default="{ row }">
<el-tag :type="getTypeColor(row.type)">
{{ row.type || '未知' }}
</el-tag>
</template>
</el-table-column>
<el-table-column prop="settingGroupID" label="配置组" width="150" />
<el-table-column label="是否开放" width="100">
<template #default="{ row }">
<el-switch
v-model="row.open"
@change="handleToggleOpen(row)"
:disabled="toggleLoading === row.id"
/>
</template>
</el-table-column>
<el-table-column prop="note" label="备注" min-width="200" show-overflow-tooltip />
<el-table-column label="创建时间" width="180">
<template #default="{ row }">
{{ formatDate(row.CreatedAt) }}
</template>
</el-table-column>
<el-table-column label="操作" width="200" fixed="right">
<template #default="{ row }">
<el-button type="primary" link @click="handleEdit(row)">编辑</el-button>
<el-button type="danger" link @click="handleDelete(row)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<el-pagination
v-model:current-page="queryParams.page"
v-model:page-size="queryParams.count"
:page-sizes="[10, 20, 50, 100]"
layout="total, sizes, prev, pager, next, jumper"
:total="total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
background
class="pagination"
/>
</el-card>
<!-- 配置表单对话框 -->
<el-dialog
v-model="dialogVisible"
:title="dialogTitle"
width="600px"
destroy-on-close
>
<el-form
ref="settingFormRef"
:model="settingForm"
:rules="settingRules"
label-width="120px"
>
<el-form-item label="配置组" prop="settingGroupID">
<el-select v-model="settingForm.settingGroupID" placeholder="请选择配置组" style="width: 100%">
<el-option
v-for="group in groupList"
:key="group.id"
:label="group.name"
:value="group.id"
/>
</el-select>
</el-form-item>
<el-form-item label="名称" prop="name">
<el-input v-model="settingForm.name" placeholder="请输入配置名称" />
</el-form-item>
<el-form-item label="类型" prop="type">
<el-select v-model="settingForm.type" placeholder="请选择类型" style="width: 100%" @change="handleTypeChange">
<el-option label="字符串 (string)" value="string" />
<el-option label="整数 (int)" value="int" />
<el-option label="浮点数 (float)" value="float" />
<el-option label="布尔值 (bool)" value="bool" />
</el-select>
</el-form-item>
<el-form-item label="值" prop="value">
<el-input
v-if="settingForm.type === 'string'"
v-model="settingForm.value"
type="textarea"
:rows="3"
placeholder="请输入配置值"
/>
<el-input-number
v-else-if="settingForm.type === 'int'"
v-model="settingForm.value"
:controls="false"
placeholder="请输入整数"
style="width: 100%"
/>
<el-input-number
v-else-if="settingForm.type === 'float'"
v-model="settingForm.value"
:controls="false"
:precision="2"
placeholder="请输入浮点数"
style="width: 100%"
/>
<el-switch
v-else-if="settingForm.type === 'bool'"
v-model="settingForm.value"
/>
<el-input
v-else
v-model="settingForm.value"
placeholder="请输入配置值"
/>
</el-form-item>
<el-form-item label="是否开放访问">
<el-switch v-model="settingForm.open" />
<span style="margin-left: 10px; color: #909399; font-size: 12px;">
开启后允许公开访问
</span>
</el-form-item>
<el-form-item label="备注" prop="note">
<el-input
v-model="settingForm.note"
type="textarea"
:rows="3"
placeholder="请输入备注信息"
/>
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="submitForm">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Search, Plus, Delete } from '@element-plus/icons-vue'
import {
getSettingList,
getSettingInfo,
createSetting,
updateSetting,
setSettingOpen,
deleteSetting
} from '@/api/admin/setting'
import { getSettingGroupList } from '@/api/admin/setting'
// 查询参数
const queryParams = reactive({
group_id: undefined,
key: '',
page: 1,
count: 10
})
// 配置表单
const settingForm = reactive({
id: undefined,
name: '',
value: '',
type: 'string',
settingGroupID: undefined,
open: false,
note: ''
})
const settingRules = {
name: [
{ required: true, message: '请输入配置名称', trigger: 'blur' }
],
value: [
{ required: true, message: '请输入配置值', trigger: 'blur' }
],
type: [
{ required: true, message: '请选择配置类型', trigger: 'change' }
],
settingGroupID: [
{ required: true, message: '请选择配置组', trigger: 'change' }
]
}
// 状态数据
const loading = ref(false)
const settingList = ref([])
const groupList = ref([])
const total = ref(0)
const selectedRows = ref([])
const dialogVisible = ref(false)
const dialogTitle = ref('新增配置')
const settingFormRef = ref(null)
const toggleLoading = ref(null)
// 格式化日期时间
const formatDate = (dateString) => {
if (!dateString) return '-'
const date = new Date(dateString)
const year = date.getFullYear()
const month = String(date.getMonth() + 1).padStart(2, '0')
const day = String(date.getDate()).padStart(2, '0')
const hours = String(date.getHours()).padStart(2, '0')
const minutes = String(date.getMinutes()).padStart(2, '0')
const seconds = String(date.getSeconds()).padStart(2, '0')
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`
}
// 获取类型颜色
const getTypeColor = (type) => {
const colorMap = {
'string': 'primary',
'int': 'success',
'float': 'warning',
'bool': 'info'
}
return colorMap[type] || ''
}
// 获取配置组列表
const fetchGroupList = async () => {
try {
const res = await getSettingGroupList({ page: 1, count: 1000 })
if (res.data.code === 200) {
groupList.value = res.data.data.data || []
}
} catch (error) {
console.error('获取配置组列表失败:', error)
}
}
// 获取配置列表
const fetchSettingList = async () => {
loading.value = true
try {
const params = { ...queryParams }
if (!params.group_id) {
delete params.group_id
}
const res = await getSettingList(params)
console.log('配置列表数据:', res.data)
if (res.data.code === 200) {
settingList.value = res.data.data.data || []
total.value = res.data.data.all_count || 0
}
} catch (error) {
console.error('获取配置列表失败:', error)
ElMessage.error('获取配置列表失败')
} finally {
loading.value = false
}
}
// 查询
const handleQuery = () => {
queryParams.page = 1
fetchSettingList()
}
// 重置查询
const resetQuery = () => {
queryParams.group_id = undefined
queryParams.key = ''
queryParams.page = 1
fetchSettingList()
}
// 选择项变化
const handleSelectionChange = (selection) => {
selectedRows.value = selection
}
// 分页
const handleSizeChange = (size) => {
queryParams.count = size
fetchSettingList()
}
const handleCurrentChange = (page) => {
queryParams.page = page
fetchSettingList()
}
// 类型变化
const handleTypeChange = (type) => {
// 根据类型重置值
if (type === 'bool') {
settingForm.value = false
} else if (type === 'int' || type === 'float') {
settingForm.value = 0
} else {
settingForm.value = ''
}
}
// 新增配置
const handleAdd = () => {
dialogTitle.value = '新增配置'
Object.assign(settingForm, {
id: undefined,
name: '',
value: '',
type: 'string',
setting_group_id: undefined,
open: false,
note: ''
})
dialogVisible.value = true
}
// 编辑配置
const handleEdit = async (row) => {
dialogTitle.value = '编辑配置'
try {
const res = await getSettingInfo({ id: row.id })
console.log('配置详情数据:', res)
if (res.data.code === 200) {
const data = res.data.data
Object.assign(settingForm, {
id: data.id,
name: data.name || '',
value: data.value,
type: data.type || 'string',
settingGroupID: data.settingGroupID,
open: data.open || false,
note: data.note || ''
})
console.log('配置详情数据:', settingForm)
// 根据类型转换值
if (data.type === 'bool') {
settingForm.value = data.value === true || data.value === 'true' || data.value === 1
} else if (data.type === 'int') {
settingForm.value = parseInt(data.value) || 0
} else if (data.type === 'float') {
settingForm.value = parseFloat(data.value) || 0
}
dialogVisible.value = true
}
} catch (error) {
console.error('获取配置详情失败:', error)
ElMessage.error('获取配置详情失败')
}
}
// 切换开放状态
const handleToggleOpen = async (row) => {
toggleLoading.value = row.id
try {
const res = await setSettingOpen({
id: row.id,
open: row.open
})
if (res.data.code === 200) {
ElMessage.success('修改成功')
} else {
// 恢复原状态
row.open = !row.open
ElMessage.error(res.data.message || '修改失败')
}
} catch (error) {
// 恢复原状态
row.open = !row.open
console.error('修改失败:', error)
ElMessage.error(error.response?.data?.message || '修改失败')
} finally {
toggleLoading.value = null
}
}
// 删除配置
const handleDelete = (row) => {
ElMessageBox.confirm(`确认删除配置 "${row.name}" 吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
try {
const res = await deleteSetting({ id: row.id })
console.log('删除配置响应:', res.data)
if (res.data.code === 200) {
ElMessage.success('删除成功')
fetchSettingList()
}
} catch (error) {
console.error('删除失败:', error)
ElMessage.error(error.response?.data?.message || '删除失败')
}
}).catch(() => {})
}
// 批量删除
const handleBatchDelete = () => {
if (selectedRows.value.length === 0) {
ElMessage.warning('请至少选择一条记录')
return
}
ElMessageBox.confirm(`确认删除选中的 ${selectedRows.value.length} 条记录吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
try {
const deletePromises = selectedRows.value.map(row =>
deleteSetting({ id: row.id })
)
await Promise.all(deletePromises)
ElMessage.success('批量删除成功')
fetchSettingList()
} catch (error) {
console.error('批量删除失败:', error)
ElMessage.error('批量删除失败')
}
}).catch(() => {})
}
// 提交表单
const submitForm = () => {
settingFormRef.value?.validate(async (valid) => {
if (valid) {
try {
const submitData = {
name: settingForm.name,
value: String(settingForm.value),
type: settingForm.type,
setting_group_id: settingForm.settingGroupID,
open: settingForm.open,
note: settingForm.note
}
if (settingForm.id) {
submitData.id = settingForm.id
}
console.log('提交配置数据:', submitData)
const res = settingForm.id
? await updateSetting(submitData)
: await createSetting(submitData)
if (res.data.code === 200) {
ElMessage.success(settingForm.id ? '修改成功' : '创建成功')
dialogVisible.value = false
fetchSettingList()
}
} catch (error) {
console.error('提交失败:', error)
ElMessage.error(error.response?.data?.message || '提交失败')
}
}
})
}
// 初始化
onMounted(() => {
fetchGroupList()
fetchSettingList()
})
</script>
<style scoped>
.setting-container {
padding: 0;
}
.filter-container {
margin-bottom: 20px;
border-radius: 8px;
}
.search-form {
margin-bottom: 15px;
}
.action-bar {
display: flex;
gap: 12px;
}
.table-container {
border-radius: 8px;
}
.pagination {
margin-top: 24px;
justify-content: flex-end;
}
</style>
+344
View File
@@ -0,0 +1,344 @@
<template>
<div class="setting-group-container">
<!-- 搜索和操作栏 -->
<el-card class="filter-container" shadow="never">
<el-form :inline="true" :model="queryParams" class="search-form">
<el-form-item label="关键词筛选">
<el-input v-model="queryParams.key" placeholder="请输入关键词" clearable style="width: 200px" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="handleQuery">
<el-icon><Search /></el-icon>查询
</el-button>
<el-button @click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
<div class="action-bar">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>新增配置组
</el-button>
<el-button type="danger" :disabled="!selectedRows.length" @click="handleBatchDelete">
<el-icon><Delete /></el-icon>批量删除
</el-button>
</div>
</el-card>
<!-- 配置组列表 -->
<el-card class="table-container" shadow="never">
<el-table
v-loading="loading"
:data="groupList"
@selection-change="handleSelectionChange"
style="width: 100%"
>
<el-table-column type="selection" width="55" />
<el-table-column prop="id" label="ID" width="80" />
<el-table-column prop="name" label="名称" min-width="200" />
<el-table-column prop="note" label="备注" min-width="250" show-overflow-tooltip />
<el-table-column label="创建时间" width="180">
<template #default="{ row }">
{{ formatDate(row.CreatedAt) }}
</template>
</el-table-column>
<el-table-column label="更新时间" width="180">
<template #default="{ row }">
{{ formatDate(row.UpdatedAt) }}
</template>
</el-table-column>
<el-table-column label="操作" width="200" fixed="right">
<template #default="{ row }">
<el-button type="primary" link @click="handleEdit(row)">编辑</el-button>
<el-button type="danger" link @click="handleDelete(row)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<el-pagination
v-model:current-page="queryParams.page"
v-model:page-size="queryParams.count"
:page-sizes="[10, 20, 50, 100]"
layout="total, sizes, prev, pager, next, jumper"
:total="total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
background
class="pagination"
/>
</el-card>
<!-- 配置组表单对话框 -->
<el-dialog
v-model="dialogVisible"
:title="dialogTitle"
width="500px"
destroy-on-close
>
<el-form
ref="groupFormRef"
:model="groupForm"
:rules="groupRules"
label-width="100px"
>
<el-form-item label="名称" prop="name">
<el-input v-model="groupForm.name" placeholder="请输入配置组名称" />
</el-form-item>
<el-form-item label="备注" prop="note">
<el-input
v-model="groupForm.note"
type="textarea"
:rows="3"
placeholder="请输入备注信息"
/>
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="submitForm">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Search, Plus, Delete } from '@element-plus/icons-vue'
import {
getSettingGroupList,
getSettingGroupInfo,
createSettingGroup,
updateSettingGroup,
deleteSettingGroup
} from '@/api/admin/setting'
// 查询参数
const queryParams = reactive({
key: '',
page: 1,
count: 10
})
// 配置组表单
const groupForm = reactive({
id: undefined,
name: '',
note: ''
})
const groupRules = {
name: [
{ required: true, message: '请输入配置组名称', trigger: 'blur' }
]
}
// 状态数据
const loading = ref(false)
const groupList = ref([])
const total = ref(0)
const selectedRows = ref([])
const dialogVisible = ref(false)
const dialogTitle = ref('新增配置组')
const groupFormRef = ref(null)
// 格式化日期时间
const formatDate = (dateString) => {
if (!dateString) return '-'
const date = new Date(dateString)
const year = date.getFullYear()
const month = String(date.getMonth() + 1).padStart(2, '0')
const day = String(date.getDate()).padStart(2, '0')
const hours = String(date.getHours()).padStart(2, '0')
const minutes = String(date.getMinutes()).padStart(2, '0')
const seconds = String(date.getSeconds()).padStart(2, '0')
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`
}
// 获取配置组列表
const fetchGroupList = async () => {
loading.value = true
try {
const res = await getSettingGroupList(queryParams)
console.log('配置组列表数据:', res.data)
if (res.data.code === 200) {
groupList.value = res.data.data.data || []
total.value = res.data.data.all_count || 0
}
} catch (error) {
console.error('获取配置组列表失败:', error)
ElMessage.error('获取配置组列表失败')
} finally {
loading.value = false
}
}
// 查询
const handleQuery = () => {
queryParams.page = 1
fetchGroupList()
}
// 重置查询
const resetQuery = () => {
queryParams.key = ''
queryParams.page = 1
fetchGroupList()
}
// 选择项变化
const handleSelectionChange = (selection) => {
selectedRows.value = selection
}
// 分页
const handleSizeChange = (size) => {
queryParams.count = size
fetchGroupList()
}
const handleCurrentChange = (page) => {
queryParams.page = page
fetchGroupList()
}
// 新增配置组
const handleAdd = () => {
dialogTitle.value = '新增配置组'
Object.assign(groupForm, {
id: undefined,
name: '',
note: ''
})
dialogVisible.value = true
}
// 编辑配置组
const handleEdit = async (row) => {
dialogTitle.value = '编辑配置组'
try {
const res = await getSettingGroupInfo({ setting_group_id: row.id })
console.log('配置组详情数据:', res.data)
if (res.data.code === 200) {
Object.assign(groupForm, {
id: res.data.data.id,
name: res.data.data.name || '',
note: res.data.data.note || ''
})
dialogVisible.value = true
}
} catch (error) {
console.error('获取配置组详情失败:', error)
ElMessage.error('获取配置组详情失败')
}
}
// 删除配置组
const handleDelete = (row) => {
ElMessageBox.confirm(`确认删除配置组 "${row.name}" 吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
try {
const res = await deleteSettingGroup({ setting_group_id: row.id })
console.log('删除配置组响应:', res.data)
if (res.data.code === 200) {
ElMessage.success('删除成功')
fetchGroupList()
}
} catch (error) {
console.error('删除失败:', error)
ElMessage.error(error.response?.data?.message || '删除失败')
}
}).catch(() => {})
}
// 批量删除
const handleBatchDelete = () => {
if (selectedRows.value.length === 0) {
ElMessage.warning('请至少选择一条记录')
return
}
ElMessageBox.confirm(`确认删除选中的 ${selectedRows.value.length} 条记录吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
try {
const deletePromises = selectedRows.value.map(row =>
deleteSettingGroup({ setting_group_id: row.id })
)
await Promise.all(deletePromises)
ElMessage.success('批量删除成功')
fetchGroupList()
} catch (error) {
console.error('批量删除失败:', error)
ElMessage.error('批量删除失败')
}
}).catch(() => {})
}
// 提交表单
const submitForm = () => {
groupFormRef.value?.validate(async (valid) => {
if (valid) {
try {
const submitData = {
name: groupForm.name,
note: groupForm.note
}
if (groupForm.id) {
submitData.id = groupForm.id
}
console.log('提交配置组数据:', submitData)
const res = groupForm.id
? await updateSettingGroup(submitData)
: await createSettingGroup(submitData)
if (res.data.code === 200) {
ElMessage.success(groupForm.id ? '修改成功' : '创建成功')
dialogVisible.value = false
fetchGroupList()
}
} catch (error) {
console.error('提交失败:', error)
ElMessage.error(error.response?.data?.message || '提交失败')
}
}
})
}
// 初始化
onMounted(() => {
fetchGroupList()
})
</script>
<style scoped>
.setting-group-container {
padding: 0;
}
.filter-container {
margin-bottom: 20px;
border-radius: 8px;
}
.search-form {
margin-bottom: 15px;
}
.action-bar {
display: flex;
gap: 12px;
}
.table-container {
border-radius: 8px;
}
.pagination {
margin-top: 24px;
justify-content: flex-end;
}
</style>
+771
View File
@@ -0,0 +1,771 @@
<template>
<div class="system-file-container">
<!-- 搜索和操作栏 -->
<el-card class="filter-container" shadow="never">
<el-form :inline="true" :model="queryParams" class="search-form">
<el-form-item label="关键词筛选">
<el-input v-model="queryParams.key" placeholder="请输入关键词" clearable style="width: 200px" />
</el-form-item>
<el-form-item label="筛选用户">
<el-input-number v-model="queryParams.user_id" placeholder="请输入用户ID" :controls="false" clearable style="width: 150px" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="handleQuery">
<el-icon><Search /></el-icon>查询
</el-button>
<el-button @click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
<div class="action-bar">
<el-button type="primary" @click="handleUpload">
<el-icon><Upload /></el-icon>上传文件
</el-button>
<el-button type="success" @click="fetchFileList">
<el-icon><Refresh/></el-icon>刷新
</el-button>
<el-button type="danger" :disabled="!selectedRows.length" @click="handleBatchDelete">
<el-icon><Delete /></el-icon>批量删除
</el-button>
</div>
</el-card>
<!-- 文件列表 -->
<el-card class="table-container" shadow="never">
<el-table
v-loading="loading"
:data="fileList"
@selection-change="handleSelectionChange"
style="width: 100%"
>
<el-table-column type="selection" width="55" />
<el-table-column prop="id" label="ID" width="80" />
<el-table-column prop="realName" label="真实文件名" min-width="200" />
<el-table-column prop="saveName" label="保存名称" min-width="150" />
<el-table-column prop="savePath" label="保存路径" min-width="250" show-overflow-tooltip />
<el-table-column prop="size" label="文件大小" width="120">
<template #default="{ row }">
{{ formatFileSize(row.size) }}
</template>
</el-table-column>
<el-table-column prop="type" label="文件类型" width="120">
<template #default="{ row }">
<el-tag :type="getFileTypeColor(row.type)">
{{ row.type || '未知' }}
</el-tag>
</template>
</el-table-column>
<el-table-column prop="userId" label="用户ID" width="100" />
<el-table-column label="是否公开" width="100">
<template #default="{ row }">
<el-tag :type="row.openDow ? 'success' : 'info'">
{{ row.openDow ? '公开' : '私有' }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="创建时间" width="180">
<template #default="{ row }">
{{ formatDate(row.CreatedAt) }}
</template>
</el-table-column>
<el-table-column label="操作" width="200" fixed="right">
<template #default="{ row }">
<el-button type="primary" link @click="handleView(row)">查看</el-button>
<el-button type="success" link @click="handleDownload(row)">下载</el-button>
<el-button type="warning" link @click="handleEdit(row)">编辑</el-button>
<el-button type="danger" link @click="handleDelete(row)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<el-pagination
v-model:current-page="queryParams.page"
v-model:page-size="queryParams.count"
:page-sizes="[10, 20, 50, 100]"
layout="total, sizes, prev, pager, next, jumper"
:total="total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
background
class="pagination"
/>
</el-card>
<!-- 文件详情对话框 -->
<el-dialog
v-model="detailDialogVisible"
title="文件详情"
width="700px"
destroy-on-close
>
<div v-if="fileDetail" class="file-detail-container">
<!-- 文件预览区域 -->
<div class="file-preview-section">
<div class="preview-label">文件预览</div>
<div class="preview-content">
<el-image
v-if="isImageFile(fileDetail.type) && fileDetail.url"
:src="fileDetail.url"
fit="contain"
style="max-width: 100%; max-height: 400px; border-radius: 8px;"
:preview-src-list="[fileDetail.url]"
:initial-index="0"
>
<template #error>
<div class="image-error">
<el-icon size="40"><Picture /></el-icon>
<div>图片加载失败</div>
</div>
</template>
</el-image>
<div v-else class="file-icon-large">
<el-icon size="80"><Document /></el-icon>
<div class="file-type-text">{{ fileDetail.type || '未知类型' }}</div>
</div>
</div>
</div>
<!-- 文件信息 -->
<el-descriptions :column="2" border class="file-info-descriptions">
<el-descriptions-item label="文件ID" label-align="right">{{ fileDetail.id }}</el-descriptions-item>
<el-descriptions-item label="用户ID" label-align="right">{{ fileDetail.userId }}</el-descriptions-item>
<el-descriptions-item label="真实文件名" label-align="right" :span="2">{{ fileDetail.realName }}</el-descriptions-item>
<el-descriptions-item label="保存名称" label-align="right">{{ fileDetail.saveName }}</el-descriptions-item>
<el-descriptions-item label="文件类型" label-align="right">
<el-tag :type="getFileTypeColor(fileDetail.type)">{{ fileDetail.type || '未知' }}</el-tag>
</el-descriptions-item>
<el-descriptions-item label="文件大小" label-align="right">{{ formatFileSize(fileDetail.size) }}</el-descriptions-item>
<el-descriptions-item label="是否公开" label-align="right">
<el-tag :type="fileDetail.openDow ? 'success' : 'info'">
{{ fileDetail.openDow ? '公开访问' : '私有' }}
</el-tag>
</el-descriptions-item>
<el-descriptions-item label="保存路径" label-align="right" :span="2">
<span class="file-path">{{ fileDetail.savePath }}</span>
</el-descriptions-item>
<el-descriptions-item label="文件URL" label-align="right" :span="2">
<el-link :href="fileDetail.url" target="_blank" type="primary" v-if="fileDetail.url">
点击查看文件
</el-link>
<span v-else style="color: #909399;">无URL</span>
</el-descriptions-item>
<el-descriptions-item label="创建时间" label-align="right">{{ formatDate(fileDetail.CreatedAt) }}</el-descriptions-item>
<el-descriptions-item label="更新时间" label-align="right">{{ formatDate(fileDetail.UpdatedAt) }}</el-descriptions-item>
<el-descriptions-item label="备注" label-align="right" :span="2">{{ fileDetail.content || '无' }}</el-descriptions-item>
</el-descriptions>
</div>
</el-dialog>
<!-- 文件编辑对话框 -->
<el-dialog
v-model="editDialogVisible"
title="编辑文件信息"
width="500px"
>
<el-form
ref="fileFormRef"
:model="fileForm"
:rules="fileRules"
label-width="120px"
>
<el-form-item label="文件ID">
<el-input v-model="fileForm.file_id" disabled />
</el-form-item>
<el-form-item label="用户ID" prop="user_id">
<el-input-number v-model="fileForm.user_id" placeholder="请输入用户ID" :controls="false" style="width: 100%" />
</el-form-item>
<el-form-item label="是否允许公开">
<el-switch v-model="fileForm.open_dow" />
<span style="margin-left: 10px; color: #909399; font-size: 12px;">
开启后允许公开访问
</span>
</el-form-item>
</el-form>
<template #footer>
<el-button @click="editDialogVisible = false">取消</el-button>
<el-button type="primary" @click="submitEditForm">确定</el-button>
</template>
</el-dialog>
<!-- 文件上传对话框 -->
<el-dialog
v-model="uploadDialogVisible"
title="上传文件"
width="600px"
>
<el-form
ref="uploadFormRef"
:model="uploadForm"
label-width="120px"
>
<el-form-item label="上传类型" prop="update_type">
<el-select v-model="uploadForm.update_type" placeholder="请选择上传类型" style="width: 100%">
<el-option label="工单文件" value="work_order" />
<el-option label="封面" value="cover" />
</el-select>
</el-form-item>
<el-form-item label="是否开放下载">
<el-switch v-model="uploadForm.open_down" />
<span style="margin-left: 10px; color: #909399; font-size: 12px;">
开启后允许公开下载
</span>
</el-form-item>
<el-form-item label="上传文件">
<el-upload
ref="uploadRef"
:http-request="handleCustomUpload"
:on-success="handleUploadSuccess"
:on-error="handleUploadError"
:on-remove="handleRemoveFile"
:on-change="handleFileChange"
:before-upload="beforeUpload"
:file-list="uploadFileList"
:auto-upload="false"
drag
multiple
>
<el-icon class="el-icon--upload"><upload-filled /></el-icon>
<div class="el-upload__text">
将文件拖到此处<em>点击上传</em>
</div>
<template #tip>
<div class="el-upload__tip">
支持 jpg/png/gif/pdf/doc/docx 文件且不超过 10MB
</div>
</template>
</el-upload>
</el-form-item>
</el-form>
<template #footer>
<el-button @click="handleCloseUpload">取消</el-button>
<el-button type="primary" @click="handleSubmitUpload">确定上传</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Upload, Delete, Search, Document, VideoPlay, Folder, UploadFilled, Picture } from '@element-plus/icons-vue'
import { getFileList, getFileDetail, updateFile, deleteFile, uploadFile } from '@/api/admin/file'
// 查询参数
const queryParams = reactive({
key: '',
user_id: undefined,
page: 1,
count: 10
})
// 文件表单
const fileForm = reactive({
file_id: undefined,
user_id: undefined,
open_dow: false
})
const fileRules = {
user_id: [
{ required: true, message: '请输入用户ID', trigger: 'blur' }
]
}
// 状态数据
const loading = ref(false)
const fileList = ref([])
const fileDetail = ref(null)
const total = ref(0)
const selectedRows = ref([])
const detailDialogVisible = ref(false)
const editDialogVisible = ref(false)
const uploadDialogVisible = ref(false)
const fileFormRef = ref(null)
const uploadRef = ref(null)
const uploadFormRef = ref(null)
// 上传表单
const uploadForm = reactive({
update_type: 'work_order',
open_down: false
})
// 上传文件列表
const uploadFileList = ref([])
// 判断是否为图片文件
const isImageFile = (type) => {
const imageTypes = ['cover', 'image', 'avatar', 'photo', 'picture']
return imageTypes.includes(type?.toLowerCase())
}
// 获取文件类型颜色
const getFileTypeColor = (type) => {
if (isImageFile(type)) return 'success'
const colorMap = {
'document': 'primary',
'video': 'warning',
'audio': 'info',
'file': ''
}
return colorMap[type?.toLowerCase()] || 'info'
}
// 格式化日期时间
const formatDate = (dateString) => {
if (!dateString) return '-'
const date = new Date(dateString)
const year = date.getFullYear()
const month = String(date.getMonth() + 1).padStart(2, '0')
const day = String(date.getDate()).padStart(2, '0')
const hours = String(date.getHours()).padStart(2, '0')
const minutes = String(date.getMinutes()).padStart(2, '0')
const seconds = String(date.getSeconds()).padStart(2, '0')
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`
}
// 格式化文件大小
const formatFileSize = (size) => {
if (size < 1024) return size + ' B'
if (size < 1024 * 1024) return (size / 1024).toFixed(2) + ' KB'
if (size < 1024 * 1024 * 1024) return (size / (1024 * 1024)).toFixed(2) + ' MB'
return (size / (1024 * 1024 * 1024)).toFixed(2) + ' GB'
}
// 获取文件列表
const fetchFileList = async () => {
loading.value = true
try {
const res = await getFileList(queryParams)
console.log('文件列表数据:', res.data)
if (res.data.code === 200) {
fileList.value = res.data.data.list || []
total.value = res.data.data.all_count || 0
}
} catch (error) {
console.error('获取文件列表失败:', error)
ElMessage.error('获取文件列表失败')
} finally {
loading.value = false
}
}
// 查询
const handleQuery = () => {
queryParams.page = 1
fetchFileList()
}
// 重置查询
const resetQuery = () => {
queryParams.key = ''
queryParams.user_id = undefined
queryParams.page = 1
fetchFileList()
}
// 选择项变化
const handleSelectionChange = (selection) => {
selectedRows.value = selection
}
// 分页
const handleSizeChange = (size) => {
queryParams.count = size
fetchFileList()
}
const handleCurrentChange = (page) => {
queryParams.page = page
fetchFileList()
}
// 查看文件详情
const handleView = async (row) => {
try {
const res = await getFileDetail({ file_id: row.id })
console.log('文件详情数据:', res.data)
if (res.data.code === 200) {
fileDetail.value = res.data.data.data
fileDetail.value.url = res.data.data.url
detailDialogVisible.value = true
}
} catch (error) {
console.error('获取文件详情失败:', error)
ElMessage.error('获取文件详情失败')
}
}
// 下载文件
const handleDownload = async (row) => {
try {
// 先获取文件详情以获取完整URL
const res = await getFileDetail({ file_id: row.id })
if (res.data.code === 200 && res.data.data.url) {
const link = document.createElement('a')
link.href = res.data.data.url
link.download = row.realName
link.target = '_blank'
link.click()
ElMessage.success('开始下载文件')
} else {
ElMessage.error('获取文件下载链接失败')
}
} catch (error) {
console.error('下载文件失败:', error)
ElMessage.error('下载文件失败')
}
}
// 编辑文件
const handleEdit = (row) => {
Object.assign(fileForm, {
file_id: row.id,
user_id: row.userId || undefined,
open_dow: row.openDow || false
})
editDialogVisible.value = true
}
// 删除文件
const handleDelete = (row) => {
ElMessageBox.confirm(`确认删除文件 ${row.realName} 吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
try {
const res = await deleteFile({ file_id: row.id })
console.log('删除文件响应:', res.data)
if (res.data.code === 200) {
ElMessage.success('删除成功')
fetchFileList()
}
} catch (error) {
console.error('删除失败:', error)
ElMessage.error(error.response?.data?.message || '删除失败')
}
}).catch(() => {})
}
// 批量删除
const handleBatchDelete = () => {
console.log("批量选择的值:",selectedRows.value)
if (selectedRows.value.length === 0) {
ElMessage.warning('请至少选择一条记录')
return
}
ElMessageBox.confirm(`确认删除选中的 ${selectedRows.value.length} 条记录吗?`, '警告', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async() => {
try{
const deleteMap = selectedRows.value.map(f => deleteFile({file_id:f.id}))
//等待所有删除完毕
await Promise.all(deleteMap)
ElMessage.success('批量删除成功')
//刷新文件列表
fetchFileList()
}catch(error){
console.error('删除失败:', error)
ElMessage.error(error.response?.data?.message || '删除失败')
}
}).catch(() => {})
}
// 上传文件
const handleUpload = () => {
uploadForm.update_type = 'work_order'
uploadForm.open_down = false
uploadFileList.value = []
uploadDialogVisible.value = true
}
// 关闭上传对话框
const handleCloseUpload = () => {
uploadDialogVisible.value = false
uploadFileList.value = []
}
// 文件列表变化
const handleFileChange = (file, fileList) => {
console.log('文件列表变化:', file, fileList)
uploadFileList.value = fileList
}
// 移除文件
const handleRemoveFile = (file, fileList) => {
console.log('移除文件:', file, fileList)
uploadFileList.value = fileList
}
// 提交上传
const handleSubmitUpload = () => {
if (uploadFileList.value.length === 0) {
ElMessage.warning('请至少选择一个文件')
return
}
// 触发所有待上传文件的上传
const filesToUpload = uploadFileList.value.filter(file =>
file.status !== 'success' && file.status !== 'uploading'
)
if (filesToUpload.length === 0) {
ElMessage.info('所有文件已上传完成')
return
}
// 逐个提交文件
uploadRef.value?.submit()
}
// 上传前检查(只做提示,不阻止文件添加到列表)
const beforeUpload = (file) => {
const isValidType = ['image/jpeg', 'image/png', 'image/gif', 'application/pdf', 'application/msword', 'application/vnd.openxmlformats-officedocument.wordprocessingml.document'].includes(file.type)
const isLt10M = file.size / 1024 / 1024 < 10
console.log('beforeUpload', file)
if (!isValidType) {
ElMessage.warning(`文件 ${file.name} 格式不符合要求(仅支持 JPG/PNG/GIF/PDF/DOC/DOCX`)
}
if (!isLt10M) {
ElMessage.warning(`文件 ${file.name} 大小超过 10MB`)
}
// 允许文件添加到列表,在上传时再进行验证
return true
}
// 自定义上传方法
const handleCustomUpload = async (options) => {
const { file, onSuccess, onError } = options
console.log('开始上传文件:', file)
// 在上传前进行验证
const isValidType = ['image/jpeg', 'image/png', 'image/gif', 'application/pdf', 'application/msword', 'application/vnd.openxmlformats-officedocument.wordprocessingml.document'].includes(file.type)
const isLt10M = file.size / 1024 / 1024 < 10
if (!isValidType) {
const error = new Error(`文件 ${file.name} 格式不符合要求(仅支持 JPG/PNG/GIF/PDF/DOC/DOCX`)
// 标记为校验类错误,on-error 中不再弹 error 提示
error.isValidation = true
onError(error, file)
return
}
if (!isLt10M) {
const error = new Error(`文件 ${file.name} 大小超过 10MB`)
error.isValidation = true
onError(error, file)
return
}
try {
const formData = new FormData()
// 根据 API 文档,字段名应该是 files(复数)
formData.append('files', file)
// 添加文件名列表(虽然 API 文档说是数组,但实际传递时直接传字符串)
formData.append('file_names', file.name)
// 添加上传类型
if (uploadForm.update_type) {
formData.append('update_type', uploadForm.update_type)
}
// 添加是否开放下载
formData.append('open_down', uploadForm.open_down ? '1' : '0')
console.log('上传参数:', {
files: file.name,
file_names: [file.name],
update_type: uploadForm.update_type,
open_down: uploadForm.open_down
})
const res = await uploadFile(formData)
console.log('上传响应:', res)
// 根据返回码严格区分成功和失败
if (res && res.data && res.data.code === 200) {
onSuccess(res.data.data, file)
} else {
const errorMsg = res?.data?.message || res?.data?.msg || '上传失败'
const error = new Error(errorMsg)
onError(error, file)
}
} catch (error) {
console.error('上传文件失败:', error)
const err = new Error(error?.response?.data?.message || error?.message || '上传失败')
onError(err, file)
}
}
// 上传成功
const handleUploadSuccess = (response, file, fileList) => {
console.log('上传成功文件:', file)
console.log('上传成功文件列表:',fileList)
// 成功回调只会在 code === 200 时触发
// ElMessage.success(`文件 ${file.name} 上传成功`)
// 更新文件列表状态
uploadFileList.value = fileList
// 如果所有文件都上传成功,关闭对话框并刷新列表
const allSuccess = fileList.every(f => f.status === 'success')
const uploadList = fileList.some(f => f.status === 'uploading')
if (allSuccess && !uploadList && fileList.length > 0) {
ElMessage.success(`已成功上传${fileList.length}个文件`)
setTimeout(() => {
uploadDialogVisible.value = false
uploadFileList.value = []
fetchFileList()
}, 1000)
}
}
// 上传失败
const handleUploadError = (error, file, fileList) => {
console.error('上传失败:', error, file, fileList)
// 对校验类错误仅在 beforeUpload 中提示过一次 warning,这里不再重复报错
if (error?.isValidation) return
ElMessage.error(error?.message || '上传失败,请检查网络连接或联系管理员')
}
// 提交编辑表单
const submitEditForm = () => {
fileFormRef.value?.validate(async (valid) => {
if (valid) {
try {
const submitData = {
file_id: fileForm.file_id,
user_id: Number(fileForm.user_id),
open_dow: fileForm.open_dow
}
console.log('提交文件信息数据:', submitData)
const res = await updateFile(submitData)
if (res.data.code === 200) {
ElMessage.success('修改成功')
editDialogVisible.value = false
fetchFileList()
}
} catch (error) {
console.error('修改失败:', error)
ElMessage.error(error.response?.data?.message || '修改失败')
}
}
})
}
// 初始化
onMounted(() => {
fetchFileList()
})
</script>
<style scoped>
.system-file-container {
padding: 0;
}
.filter-container {
margin-bottom: 20px;
border-radius: 8px;
}
.search-form {
margin-bottom: 15px;
}
.action-bar {
display: flex;
gap: 12px;
}
.table-container {
border-radius: 8px;
}
.file-icon {
display: flex;
align-items: center;
justify-content: center;
width: 60px;
height: 60px;
background-color: #f5f7fa;
border-radius: 4px;
}
.pagination {
margin-top: 24px;
justify-content: flex-end;
}
.file-detail-container {
padding: 10px 0;
}
.file-preview-section {
margin-bottom: 24px;
}
.preview-label {
font-size: 14px;
font-weight: 500;
color: #303133;
margin-bottom: 12px;
}
.preview-content {
display: flex;
justify-content: center;
align-items: center;
background-color: #f5f7fa;
border-radius: 8px;
padding: 20px;
min-height: 200px;
}
.file-icon-large {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
color: #909399;
gap: 12px;
}
.file-type-text {
font-size: 14px;
color: #606266;
}
.image-error {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
color: #909399;
gap: 8px;
}
.file-info-descriptions {
margin-top: 16px;
}
.file-path {
font-family: 'Courier New', monospace;
font-size: 12px;
color: #606266;
word-break: break-all;
}
:deep(.el-descriptions__label) {
width: 120px;
}
</style>