mirror of
https://gitcode.com/flipped-aurora/gin-vue-admin.git
synced 2025-12-30 03:42:26 +00:00
* feat(mcp): 新增gva_review工具并优化字典和代码生成逻辑 * fix: 调整mcp整体逻辑 * chore: 更新.gitignore,添加对本地配置文件的忽略 * feat(logo): 新增Logo组件并在多个页面中替换原有logo实现 * fix: 修复菜单 Logo 部分删除文本后显示异常的问题 * fix:添加字典列表搜索,支持中英文搜索.添加字典详情搜索 * style: 优化部分视觉样式 * feat: 增强错误预览组件的暗黑模式支持 * feat: 优化请求错误消息获取逻辑,增加状态文本优先级 * feat: 添加前端登录验证码静态验证逻辑 * feat: 添加开发环境启动脚本 * feat: 更新 SvgIcon 组件,支持本地图标和 Iconify 图标、移除未使用的 unocss 依赖 * fix:字典支持 tree 结构 * feat: 优化动态路由注册方式 * feat: 添加配置控制标签页keep-alive功能 * feat: 添加全局错误处理机制,捕获 Vue 和 JS 错误 * refactor: 移除API和菜单创建结果中的权限分配提醒,优化输出信息 * feat: 更新 reset.scss,优化全局样式重置,增强兼容性和可读性 * refactor(字典详情): 优化字典详情查询逻辑,移除预加载改为按需加载 * refactor(路由管理): 优化路由添加逻辑,增强路径处理和顶级路由注册 * refactor(系统配置): 将auto-migrate修改为disable-auto-migrate,保证用户升级的兼容性 * feat(utils): 优化字典数据递归查找功能并替换select为tree-select * fix(deps): 修复在字段类型为file生成搜索条件无法运行的bug * fix: 修复header的tools中icon不展示的问题 --------- Co-authored-by: piexlMax(奇淼 <qimiaojiangjizhao@gmail.com> Co-authored-by: Azir-11 <2075125282@qq.com> Co-authored-by: bypanghu <bypanghu@163.com> Co-authored-by: feitianbubu <feitianbubu@qq.com> Co-authored-by: 青菜白玉汤 <79054161+Azir-11@users.noreply.github.com> Co-authored-by: krank <emosick@qq.com>
503 lines
14 KiB
Vue
503 lines
14 KiB
Vue
<template>
|
|
<div v-loading.fullscreen.lock="fullscreenLoading">
|
|
<div class="flex gap-4 pt-2">
|
|
<div
|
|
class="flex-none w-64 bg-white text-slate-700 dark:text-slate-400 dark:bg-slate-900 rounded p-4"
|
|
>
|
|
<el-scrollbar style="height: calc(100vh - 300px)">
|
|
<el-tree
|
|
:data="categories"
|
|
node-key="id"
|
|
:props="defaultProps"
|
|
@node-click="handleNodeClick"
|
|
default-expand-all
|
|
>
|
|
<template #default="{ node, data }">
|
|
<div
|
|
class="w-36"
|
|
:class="
|
|
search.classId === data.ID ? 'text-blue-500 font-bold' : ''
|
|
"
|
|
>
|
|
{{ data.name }}
|
|
</div>
|
|
<el-dropdown>
|
|
<el-icon class="ml-3 text-right" v-if="data.ID > 0"
|
|
><MoreFilled
|
|
/></el-icon>
|
|
<el-icon class="ml-3 text-right mt-1" v-else><Plus /></el-icon>
|
|
<template #dropdown>
|
|
<el-dropdown-menu>
|
|
<el-dropdown-item @click="addCategoryFun(data)"
|
|
>添加分类</el-dropdown-item
|
|
>
|
|
<el-dropdown-item
|
|
@click="editCategory(data)"
|
|
v-if="data.ID > 0"
|
|
>编辑分类</el-dropdown-item
|
|
>
|
|
<el-dropdown-item
|
|
@click="deleteCategoryFun(data.ID)"
|
|
v-if="data.ID > 0"
|
|
>删除分类</el-dropdown-item
|
|
>
|
|
</el-dropdown-menu>
|
|
</template>
|
|
</el-dropdown>
|
|
</template>
|
|
</el-tree>
|
|
</el-scrollbar>
|
|
</div>
|
|
<div
|
|
class="flex-1 bg-white text-slate-700 dark:text-slate-400 dark:bg-slate-900"
|
|
>
|
|
<div class="gva-table-box mt-0 mb-0">
|
|
<warning-bar
|
|
title="点击“文件名”可以编辑;选择的类别即是上传的类别。"
|
|
/>
|
|
<div class="gva-btn-list gap-3">
|
|
<upload-common
|
|
:image-common="imageCommon"
|
|
:classId="search.classId"
|
|
@on-success="onSuccess"
|
|
/>
|
|
<cropper-image :classId="search.classId" @on-success="onSuccess" />
|
|
<QRCodeUpload :classId="search.classId" @on-success="onSuccess" />
|
|
<upload-image
|
|
:image-url="imageUrl"
|
|
:file-size="512"
|
|
:max-w-h="1080"
|
|
:classId="search.classId"
|
|
@on-success="onSuccess"
|
|
/>
|
|
<el-button type="primary" icon="upload" @click="importUrlFunc">
|
|
导入URL
|
|
</el-button>
|
|
<el-input
|
|
v-model="search.keyword"
|
|
class="w-72"
|
|
placeholder="请输入文件名或备注"
|
|
/>
|
|
<el-button type="primary" icon="search" @click="onSubmit"
|
|
>查询
|
|
</el-button>
|
|
</div>
|
|
|
|
<el-table :data="tableData">
|
|
<el-table-column align="left" label="预览" width="100">
|
|
<template #default="scope">
|
|
<CustomPic pic-type="file" :pic-src="scope.row.url" preview />
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column
|
|
align="left"
|
|
label="日期"
|
|
prop="UpdatedAt"
|
|
width="180"
|
|
>
|
|
<template #default="scope">
|
|
<div>{{ formatDate(scope.row.UpdatedAt) }}</div>
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column
|
|
align="left"
|
|
label="文件名/备注"
|
|
prop="name"
|
|
width="180"
|
|
>
|
|
<template #default="scope">
|
|
<div
|
|
class="cursor-pointer"
|
|
@click="editFileNameFunc(scope.row)"
|
|
>
|
|
{{ scope.row.name }}
|
|
</div>
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column
|
|
align="left"
|
|
label="链接"
|
|
prop="url"
|
|
min-width="300"
|
|
/>
|
|
<el-table-column align="left" label="标签" prop="tag" width="100">
|
|
<template #default="scope">
|
|
<el-tag
|
|
:type="
|
|
scope.row.tag?.toLowerCase() === 'jpg' ? 'info' : 'success'
|
|
"
|
|
disable-transitions
|
|
>{{ scope.row.tag }}
|
|
</el-tag>
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column align="left" label="操作" width="160">
|
|
<template #default="scope">
|
|
<el-button
|
|
icon="download"
|
|
type="primary"
|
|
link
|
|
@click="downloadFile(scope.row)"
|
|
>下载
|
|
</el-button>
|
|
<el-button
|
|
icon="delete"
|
|
type="primary"
|
|
link
|
|
@click="deleteFileFunc(scope.row)"
|
|
>删除
|
|
</el-button>
|
|
</template>
|
|
</el-table-column>
|
|
</el-table>
|
|
<div class="gva-pagination">
|
|
<el-pagination
|
|
:current-page="page"
|
|
:page-size="pageSize"
|
|
:page-sizes="[10, 30, 50, 100]"
|
|
:style="{ float: 'right', padding: '20px' }"
|
|
:total="total"
|
|
layout="total, sizes, prev, pager, next, jumper"
|
|
@current-change="handleCurrentChange"
|
|
@size-change="handleSizeChange"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 添加分类弹窗 -->
|
|
<el-dialog
|
|
v-model="categoryDialogVisible"
|
|
@close="closeAddCategoryDialog"
|
|
width="520"
|
|
:title="(categoryFormData.ID === 0 ? '添加' : '编辑') + '分类'"
|
|
draggable
|
|
>
|
|
<el-form
|
|
ref="categoryForm"
|
|
:rules="rules"
|
|
:model="categoryFormData"
|
|
label-width="80px"
|
|
>
|
|
<el-form-item label="上级分类">
|
|
<el-tree-select
|
|
v-model="categoryFormData.pid"
|
|
:data="categories"
|
|
check-strictly
|
|
:props="defaultProps"
|
|
:render-after-expand="false"
|
|
style="width: 240px"
|
|
/>
|
|
</el-form-item>
|
|
<el-form-item label="分类名称" prop="name">
|
|
<el-input
|
|
v-model.trim="categoryFormData.name"
|
|
placeholder="分类名称"
|
|
></el-input>
|
|
</el-form-item>
|
|
</el-form>
|
|
<template #footer>
|
|
<el-button @click="closeAddCategoryDialog">取消</el-button>
|
|
<el-button type="primary" @click="confirmAddCategory">确定</el-button>
|
|
</template>
|
|
</el-dialog>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup>
|
|
import {
|
|
getFileList,
|
|
deleteFile,
|
|
editFileName,
|
|
importURL
|
|
} from '@/api/fileUploadAndDownload'
|
|
import { downloadImage } from '@/utils/downloadImg'
|
|
import CustomPic from '@/components/customPic/index.vue'
|
|
import UploadImage from '@/components/upload/image.vue'
|
|
import UploadCommon from '@/components/upload/common.vue'
|
|
import { CreateUUID, formatDate } from '@/utils/format'
|
|
import WarningBar from '@/components/warningBar/warningBar.vue'
|
|
|
|
import { ref } from 'vue'
|
|
import { ElMessage, ElMessageBox } from 'element-plus'
|
|
import {
|
|
addCategory,
|
|
deleteCategory,
|
|
getCategoryList
|
|
} from '@/api/attachmentCategory'
|
|
import CropperImage from '@/components/upload/cropper.vue'
|
|
import QRCodeUpload from '@/components/upload/QR-code.vue'
|
|
|
|
defineOptions({
|
|
name: 'Upload'
|
|
})
|
|
|
|
const fullscreenLoading = ref(false)
|
|
const path = ref(import.meta.env.VITE_BASE_API)
|
|
|
|
const imageUrl = ref('')
|
|
const imageCommon = ref('')
|
|
|
|
const page = ref(1)
|
|
const total = ref(0)
|
|
const pageSize = ref(10)
|
|
const search = ref({
|
|
keyword: null,
|
|
classId: 0
|
|
})
|
|
const tableData = ref([])
|
|
|
|
// 分页
|
|
const handleSizeChange = (val) => {
|
|
pageSize.value = val
|
|
getTableData()
|
|
}
|
|
|
|
const handleCurrentChange = (val) => {
|
|
page.value = val
|
|
getTableData()
|
|
}
|
|
|
|
const onSubmit = () => {
|
|
search.value.classId = 0
|
|
page.value = 1
|
|
getTableData()
|
|
}
|
|
|
|
// 查询
|
|
const getTableData = async () => {
|
|
const table = await getFileList({
|
|
page: page.value,
|
|
pageSize: pageSize.value,
|
|
...search.value
|
|
})
|
|
if (table.code === 0) {
|
|
tableData.value = table.data.list
|
|
total.value = table.data.total
|
|
page.value = table.data.page
|
|
pageSize.value = table.data.pageSize
|
|
}
|
|
}
|
|
getTableData()
|
|
|
|
const deleteFileFunc = async (row) => {
|
|
ElMessageBox.confirm('此操作将永久删除文件, 是否继续?', '提示', {
|
|
confirmButtonText: '确定',
|
|
cancelButtonText: '取消',
|
|
type: 'warning'
|
|
})
|
|
.then(async () => {
|
|
const res = await deleteFile(row)
|
|
if (res.code === 0) {
|
|
ElMessage({
|
|
type: 'success',
|
|
message: '删除成功!'
|
|
})
|
|
if (tableData.value.length === 1 && page.value > 1) {
|
|
page.value--
|
|
}
|
|
await getTableData()
|
|
}
|
|
})
|
|
.catch(() => {
|
|
ElMessage({
|
|
type: 'info',
|
|
message: '已取消删除'
|
|
})
|
|
})
|
|
}
|
|
|
|
const downloadFile = (row) => {
|
|
if (row.url.indexOf('http://') > -1 || row.url.indexOf('https://') > -1) {
|
|
downloadImage(row.url, row.name)
|
|
} else {
|
|
downloadImage(path.value + '/' + row.url, row.name)
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 编辑文件名或者备注
|
|
* @param row
|
|
* @returns {Promise<void>}
|
|
*/
|
|
const editFileNameFunc = async (row) => {
|
|
ElMessageBox.prompt('请输入文件名或者备注', '编辑', {
|
|
confirmButtonText: '确定',
|
|
cancelButtonText: '取消',
|
|
inputPattern: /\S/,
|
|
inputErrorMessage: '不能为空',
|
|
inputValue: row.name
|
|
})
|
|
.then(async ({ value }) => {
|
|
row.name = value
|
|
// console.log(row)
|
|
const res = await editFileName(row)
|
|
if (res.code === 0) {
|
|
ElMessage({
|
|
type: 'success',
|
|
message: '编辑成功!'
|
|
})
|
|
await getTableData()
|
|
}
|
|
})
|
|
.catch(() => {
|
|
ElMessage({
|
|
type: 'info',
|
|
message: '取消修改'
|
|
})
|
|
})
|
|
}
|
|
|
|
/**
|
|
* 导入URL
|
|
*/
|
|
const importUrlFunc = () => {
|
|
ElMessageBox.prompt('格式:文件名|链接或者仅链接。', '导入', {
|
|
confirmButtonText: '确定',
|
|
cancelButtonText: '取消',
|
|
inputType: 'textarea',
|
|
inputPlaceholder:
|
|
'我的图片|https://my-oss.com/my.png\nhttps://my-oss.com/my_1.png',
|
|
inputPattern: /\S/,
|
|
inputErrorMessage: '不能为空'
|
|
})
|
|
.then(async ({ value }) => {
|
|
let data = value.split('\n')
|
|
let importData = []
|
|
data.forEach((item) => {
|
|
let oneData = item.trim().split('|')
|
|
let url, name
|
|
if (oneData.length > 1) {
|
|
name = oneData[0].trim()
|
|
url = oneData[1]
|
|
} else {
|
|
url = oneData[0].trim()
|
|
let str = url.substring(url.lastIndexOf('/') + 1)
|
|
name = str.substring(0, str.lastIndexOf('.'))
|
|
}
|
|
if (url) {
|
|
importData.push({
|
|
name: name,
|
|
url: url,
|
|
classId: search.value.classId,
|
|
tag: url.substring(url.lastIndexOf('.') + 1),
|
|
key: CreateUUID()
|
|
})
|
|
}
|
|
})
|
|
|
|
const res = await importURL(importData)
|
|
if (res.code === 0) {
|
|
ElMessage({
|
|
type: 'success',
|
|
message: '导入成功!'
|
|
})
|
|
await getTableData()
|
|
}
|
|
})
|
|
.catch(() => {
|
|
ElMessage({
|
|
type: 'info',
|
|
message: '取消导入'
|
|
})
|
|
})
|
|
}
|
|
|
|
const onSuccess = () => {
|
|
search.value.keyword = null
|
|
page.value = 1
|
|
getTableData()
|
|
}
|
|
|
|
const defaultProps = {
|
|
children: 'children',
|
|
label: 'name',
|
|
value: 'ID'
|
|
}
|
|
|
|
const categories = ref([])
|
|
const fetchCategories = async () => {
|
|
const res = await getCategoryList()
|
|
let data = {
|
|
name: '全部分类',
|
|
ID: 0,
|
|
pid: 0,
|
|
children: []
|
|
}
|
|
if (res.code === 0) {
|
|
categories.value = res.data || []
|
|
categories.value.unshift(data)
|
|
}
|
|
}
|
|
|
|
const handleNodeClick = (node) => {
|
|
search.value.keyword = null
|
|
search.value.classId = node.ID
|
|
page.value = 1
|
|
getTableData()
|
|
}
|
|
|
|
const categoryDialogVisible = ref(false)
|
|
const categoryFormData = ref({
|
|
ID: 0,
|
|
pid: 0,
|
|
name: ''
|
|
})
|
|
|
|
const categoryForm = ref(null)
|
|
const rules = ref({
|
|
name: [
|
|
{ required: true, message: '请输入分类名称', trigger: 'blur' },
|
|
{ max: 20, message: '最多20位字符', trigger: 'blur' }
|
|
]
|
|
})
|
|
|
|
const addCategoryFun = (category) => {
|
|
categoryDialogVisible.value = true
|
|
categoryFormData.value.ID = 0
|
|
categoryFormData.value.pid = category.ID
|
|
}
|
|
|
|
const editCategory = (category) => {
|
|
categoryFormData.value = {
|
|
ID: category.ID,
|
|
pid: category.pid,
|
|
name: category.name
|
|
}
|
|
categoryDialogVisible.value = true
|
|
}
|
|
|
|
const deleteCategoryFun = async (id) => {
|
|
const res = await deleteCategory({ id: id })
|
|
if (res.code === 0) {
|
|
ElMessage.success({ type: 'success', message: '删除成功' })
|
|
await fetchCategories()
|
|
}
|
|
}
|
|
|
|
const confirmAddCategory = async () => {
|
|
categoryForm.value.validate(async (valid) => {
|
|
if (valid) {
|
|
const res = await addCategory(categoryFormData.value)
|
|
if (res.code === 0) {
|
|
ElMessage({ type: 'success', message: '操作成功' })
|
|
await fetchCategories()
|
|
closeAddCategoryDialog()
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
const closeAddCategoryDialog = () => {
|
|
categoryDialogVisible.value = false
|
|
categoryFormData.value = {
|
|
ID: 0,
|
|
pid: 0,
|
|
name: ''
|
|
}
|
|
}
|
|
|
|
fetchCategories()
|
|
</script>
|