【代码优化】系统菜单管理使用虚拟列表优化加载速度

pull/678/head
puhui999 2025-01-23 16:32:01 +08:00
parent c50f59d181
commit 5832fd687f
1 changed files with 144 additions and 66 deletions

View File

@ -67,87 +67,159 @@
<!-- 列表 -->
<ContentWrap>
<el-table
v-if="refreshTable"
v-loading="loading"
:data="list"
:default-expand-all="isExpandAll"
row-key="id"
>
<el-table-column :show-overflow-tooltip="true" label="菜单名称" prop="name" width="250" />
<el-table-column align="center" label="图标" prop="icon" width="100">
<template #default="scope">
<Icon :icon="scope.row.icon" />
</template>
</el-table-column>
<el-table-column label="排序" prop="sort" width="60" />
<el-table-column :show-overflow-tooltip="true" label="权限标识" prop="permission" />
<el-table-column :show-overflow-tooltip="true" label="组件路径" prop="component" />
<el-table-column :show-overflow-tooltip="true" label="组件名称" prop="componentName" />
<el-table-column label="状态" prop="status">
<template #default="scope">
<el-switch
class="ml-4px"
v-model="scope.row.status"
v-hasPermi="['system:menu:update']"
:active-value="CommonStatusEnum.ENABLE"
:inactive-value="CommonStatusEnum.DISABLE"
:loading="menuStatusUpdating[scope.row.id]"
@change="(val) => handleStatusChanged(scope.row, val as number)"
/>
</template>
</el-table-column>
<el-table-column align="center" label="操作">
<template #default="scope">
<el-button
v-hasPermi="['system:menu:update']"
link
type="primary"
@click="openForm('update', scope.row.id)"
>
修改
</el-button>
<el-button
v-hasPermi="['system:menu:create']"
link
type="primary"
@click="openForm('create', undefined, scope.row.id)"
>
新增
</el-button>
<el-button
v-hasPermi="['system:menu:delete']"
link
type="danger"
@click="handleDelete(scope.row.id)"
>
删除
</el-button>
</template>
</el-table-column>
</el-table>
<el-auto-resizer>
<template #default="{ width }">
<el-table-v2
v-model:expanded-row-keys="expandedRowKeys"
:columns="columns"
:data="list"
:expand-column-key="columns[0].key"
:height="1000"
:width="width"
fixed
row-key="id"
/>
</template>
</el-auto-resizer>
</ContentWrap>
<!-- 表单弹窗添加/修改 -->
<MenuForm ref="formRef" @success="getList" />
</template>
<script lang="ts" setup>
<script lang="tsx" setup>
import { DICT_TYPE, getIntDictOptions } from '@/utils/dict'
import { handleTree } from '@/utils/tree'
import * as MenuApi from '@/api/system/menu'
import { MenuVO } from '@/api/system/menu'
import MenuForm from './MenuForm.vue'
import { CACHE_KEY, useCache } from '@/hooks/web/useCache'
import DictTag from '@/components/DictTag/src/DictTag.vue'
import { Icon } from '@/components/Icon'
import { ElButton, TableV2FixedDir } from 'element-plus'
import { checkPermi } from '@/utils/permission'
import { CommonStatusEnum } from '@/utils/constants'
import { CACHE_KEY, useCache } from '@/hooks/web/useCache'
defineOptions({ name: 'SystemMenu' })
//
const columns = [
{
key: 'name',
title: '菜单名称',
dataKey: 'name',
width: 250,
fixed: TableV2FixedDir.LEFT
},
{
key: 'icon',
title: '图标',
dataKey: 'icon',
width: 100,
align: 'center',
cellRenderer: ({ cellData: icon }) => <Icon icon={icon} />
},
{
key: 'sort',
title: '排序',
dataKey: 'sort',
width: 60
},
{
key: 'permission',
title: '权限标识',
dataKey: 'permission',
width: 300
},
{
key: 'component',
title: '组件路径',
dataKey: 'component',
width: 500
},
{
key: 'componentName',
title: '组件名称',
dataKey: 'componentName',
width: 200
},
{
key: 'status',
title: '状态',
dataKey: 'status',
width: 60,
fixed: TableV2FixedDir.RIGHT,
cellRenderer: ({ rowData }) => {
//
if (!checkPermi(['system:menu:update'])) {
return <DictTag type={DICT_TYPE.COMMON_STATUS} value={rowData.status} />
}
// ElSwitch
return (
<ElSwitch
v-model={rowData.status}
active-value={CommonStatusEnum.ENABLE}
inactive-value={CommonStatusEnum.DISABLE}
loading={menuStatusUpdating[rowData.id]}
class="ml-4px"
onChange={(val) => handleStatusChanged(rowData, val)}
/>
)
}
},
{
key: 'operations',
title: '操作',
align: 'center',
width: 160,
fixed: TableV2FixedDir.RIGHT,
cellRenderer: ({ rowData }) => {
//
const buttons = []
//
if (checkPermi(['system:menu:update'])) {
buttons.push(
<ElButton key="edit" link type="primary" onClick={() => openForm('update', rowData.id)}>
修改
</ElButton>
)
}
if (checkPermi(['system:menu:create'])) {
buttons.push(
<ElButton
key="create"
link
type="primary"
onClick={() => openForm('create', undefined, rowData.id)}
>
新增
</ElButton>
)
}
if (checkPermi(['system:menu:delete'])) {
buttons.push(
<ElButton key="delete" link type="danger" onClick={() => handleDelete(rowData.id)}>
删除
</ElButton>
)
}
// null
if (buttons.length === 0) {
return null
}
//
return <>{buttons}</>
}
}
]
const { wsCache } = useCache()
const { t } = useI18n() //
const message = useMessage() //
const loading = ref(true) //
const list = ref<any>([]) //
const list = ref<any[]>([]) //
const queryParams = reactive({
name: undefined,
status: undefined
@ -156,6 +228,9 @@ const queryFormRef = ref() // 搜索的表单
const isExpandAll = ref(false) //
const refreshTable = ref(true) //
//
const expandedRowKeys = ref<number[]>([])
/** 查询列表 */
const getList = async () => {
loading.value = true
@ -186,11 +261,14 @@ const openForm = (type: string, id?: number, parentId?: number) => {
/** 展开/折叠操作 */
const toggleExpandAll = () => {
refreshTable.value = false
if (!isExpandAll.value) {
//
expandedRowKeys.value = list.value.map((item) => item.id)
} else {
//
expandedRowKeys.value = []
}
isExpandAll.value = !isExpandAll.value
nextTick(() => {
refreshTable.value = true
})
}
/** 刷新菜单缓存按钮操作 */