logo 范 · 拾光录
网址收集 关于作者 Github Gitee
杂文随笔5
Hexo博客:基础使用Hexo博客:Next主题Hexo博客:Next进阶使用Hexo博客:Next高级配置基于Node的WIKI管理
前端知识16
HTML常用知识CSS常用知识CSS美化checkbox复选框JavaScript常用知识JavaScript格式化时间戳JavaScript窗口宽高处理JavaScript黑夜主题切换实现方案JavaScript数字转大写简易图片查看器TypeScript基础知识Threejs基础三要素Threejs网格辅助和轨道控制器Threejs物体绘制Electron基础使用Nodejs基础知识animate.css页面动画
Vue框架19
Vite的使用及扩展Vue3父子组件Vue3使用Marked解析MarkdownMermaid图表生成库初始化页面加载动画Axios表单提交二维码解决方案NProgress加载进度条Vue3动态菜单实现Vue3使用ECharts图表Vue3处理Excel导入导出keep-alive页面缓存及setup问题Element:文件上传Element:结合Pinia实现动态菜单Element:图片上传组件Element:自定义统一弹窗组件Element:表格自定义指令控制按钮显示(鉴权)可视化大屏使用缩放适配分辨率
UniApp15
UniApp的基础使用封装网络请求工具及文件上传uni-app的开发记录微信小程序分享原生文件上传Pinia取消滚动条(兼容小程序)tabbar消息数量显示scroll-view上滑到底部加载数据状态栏高度动态设配数据共享与传递uview-plus导航栏实现背景融合Wot UIWot UI实现顶部背景图融合uni-app x
Java基础知识10
基础知识面向对象Lambda表达式常用API常用知识积累try-with-resource注解反射多线程经纬度距离计算
SpringBoot31
application配置Maven创建聚合项目全局异常处理锁机制项目启动初始化数据方式邮件功能集成原生定时任务异步集成阿里云OSS阿里OSS预签名上传基于hutool读excelJSR303WebSocketWebSocket版AI接口流式调用Smart-Doc接口文档生成器application配置信息加密雪花算法工具AOP实现请求参数脱敏思路JWT生成Token及工具类SpringBoot默认JSON与对象转换若依框架:安装使用若依框架:优化和调整文件上传若依框架:管理后台页面优化若依框架:后端接口代码优化SpringAISpringBoot实现AI接口流式调用服务启动时创建MySQL连接自建项目工程树形结构处理工具微信支付代码微信手机号登录
SpringMVC14
跨域处理拦截器RESTful风格伪前后端分离Jackson转换器调整Thymeleaf基于拦截器做权限校验AOP打印接口请求响应日志AOP打印接口请求响应耗时文件上传和回显POST请求加解密实现(AES)POST请求加解密实现(RSA+AES)参数动态校验实现方案真实IP和归属地
MyBatis8
MyBatis基本使用与配置Mapper使用相关MaBatis多数据源配置MyBatisPlus数据统计类处理方案MyBatisPlus条件查询正向工程的实现(H2)mybatis-plus-join
SpringCloud15
Netflix:微服务与搭建Netflix:服务的消费与提供Netflix:EurekaNetflix:ActuatorNetflix:RibbonNetflix:FeignNetflix:HystrixNetflix:ZuulAlibaba:简介与搭建Alibaba:Nacos注册中心Alibaba:RibbonAlibaba:OpenFeignAlibaba:Nacos配置中心Alibaba:GetewayAlibaba:Sentinel
MySQL6
MySQL基础知识MySQL多表查询与事务MySQL常用函数及解决方案MySQL视图MySQL索引安装MySQL
Redis7
Redis介绍和安装Redis配置文件Redis持久化Redis集群Redis语法基础Redis相关问题及解决方案SpringBoot集成Redis使用记录
MongoDB10
Linux安装MongoDBMongoDB基础语法MongoTemplate及SpringBoot配置MongoTemplate中Update操作MongoTemplate中聚合查询MongoTemplate日期归档示例项目使用相关知识归纳地理位置存储与距离查询MongoDB副本集与事务获取类名和属性名工具类
其他数据库1
H2数据库
Python编程6
Python基础知识Python语法yolo目标检测OpenCV的使用及树莓派平台condauv
工具集合13
IDEAMavenGradleGitNginx安装Nginx配置VSCodeJMeter压测DockerOllamaRustFSPicGoObs录制
Linux知识11
Linux常用命令Jar启动脚本VirtualBox安装CentOSVirtualBox安装Ubuntu树莓派安装及使用frp内网穿透ArchLinux:基础系统安装ArchLInux:图形化界面安装ArchLinux:常用软件ArchLinux:深度优化ArchLinux:Niri
创意设计2
Blender:入门知识UI设计基础知识
AI相关9
Claude CodeHermes AgentOpenAI基本使用OpenAI工具调用OpenAI记忆管理OpenAI推理执行OpenAI开发框架Langchainllama.cpp

基于Vue3+ElementPlus+Router+Store实现动态菜单(支持1-3级),应用于后台管理系统中,根据管理员的权限值自动生成相应的菜单进行展示,是每个管理后台系统最基础的功能,适用于大部分后台管理系统

Router路由

{  
  path: '/home',
  component: () => import('../views/Home.vue'),
  redirect: '/home/dataShow',
  children: [
    /**
     * 说明:
     *    menuLevel 为导航级别,按顺序写
     */
    {
      path: 'dataShow',
      name: '数据概况',
      menuLevel: 1,
      adminTypes: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12],
      component: () => import('../components/DataShow.vue'),
      icon: 'histogram',
    },
    {
      path: 'system',
      name: '系统管理',
      menuLevel: 1,
      icon: 'cpu',
    },
    {
      path: 'system/adminUser',
      name: '管理员管理',
      menuLevel: 2,
      component: () => import('../components/system/AminUser.vue'),
    },
    {
      path: 'update',
      name: '更新管理',
      menuLevel: 2,
    },
    {
      path: 'system/appUpdate',
      name: 'APP更新管理',
      menuLevel: 3,
      component: () => import('../components/system/AppUpdate.vue'),
    },
    {
      path: 'system/PCUpdate',
      name: 'PC更新管理',
      menuLevel: 3,
      component: () => import('../components/system/PCUpdate.vue'),
    },
    // --------------------------分------割-------------------------------
    {
      path: 'system/breakRouter',
      menuLevel: 4,
    },
    ...other
  ]
  },

菜单组件

<!--菜单-->
<template>
  <div class="home-menu">
    <el-menu
        :default-active="store.state.menuIndex"
        router
        background-color="#304156"
        text-color="#ddd"
        active-text-color="#409eff"
        :collapse="store.state.menuCollapse">
      <menuTree :menuList="store.state.menuTree"></menuTree>
    </el-menu>
  </div>
</template>

<script setup>
import store from '../../store'
import router from '../../router'
import menuTree from './MenuTree.vue'

menu()

// 菜单计算(将路由转为为菜单树)
function menu() {
  // 获取所有菜单列表
  let menus = router.options.routes[0].children
  let oneItem = {}
  let twoItem = {}
  let oneMenu = []
  let twoMenu = []
  let threeMenu = []
  for (let item of menus) {
    // 分割
    if (item.menuLevel === 4) {
      break
    }
    item.path = '/home/' + item.path
    if (item.menuLevel === 1) {
      if (threeMenu.length > 0) {
        twoItem.ch = threeMenu
        twoMenu.push(twoItem)
        twoItem = {}
        threeMenu = []
      }
      if (twoMenu.length > 0) {
        oneItem.ch = twoMenu
        oneMenu.push(oneItem)
        oneItem = {}
        twoMenu = []
      }
      if (item.component) {
        // 该级别可点击,校验是否有权限
        menuShowRule(item, () => {
          // 没有下级,直接添加进最后数据
          oneMenu.push(item)
        })
      } else {
        // 不可点击
        oneItem = item
      }
    } else if (item.menuLevel === 2) {
      if (threeMenu.length > 0) {
        twoItem.ch = threeMenu
        twoMenu.push(twoItem)
        twoItem = {}
        threeMenu = []
      }
      if (item.component) {
        // 该级别可点击,校验是否有权限
        menuShowRule(item, () => {
          // 没有下级,直接添加进数据
          twoMenu.push(item)
        })
      } else {
        // 不可点击
        twoItem = item
      }
      twoItem = item
    } else if (item.menuLevel === 3) {
      menuShowRule(item, () => {
        threeMenu.push(item)
      })
    }
  }
  if (threeMenu.length > 0) {
    twoItem.ch = threeMenu
    twoMenu.push(twoItem)
  }
  if (twoMenu.length > 0) {
    oneItem.ch = twoMenu
    oneMenu.push(oneItem)
  }
  // 保存菜单
  store.commit('setMenuTree', oneMenu)
}

/**
 * 定义是否显示的规则
 */
function menuShowRule(item, fun) {
  // 自己部门的权限
  if (item.adminTypes.includes(store.state.adminType)) {
    fun()
  }
}
</script>

<style scoped lang="sass">
// 菜单高度
.el-menu-item
  height: 45px

.el-sub-menu
  height: 45px

// 去除菜单边框
.el-menu
  border-right: none
</style>

递归显示组件

<template>
  <template v-for="menu in menuList">
    <!--没有子菜单-->
    <el-menu-item v-if="menu.component" :index="menu.path" @click="store.commit('setMenuIndex', menu.path)">
      <el-icon>
        <component :is="menu.icon"></component>
      </el-icon>
      <span>{{ menu.name }}</span>
    </el-menu-item>
    <!--有子菜单-->
    <el-sub-menu
        v-else
        :index="menu.path">
      <template #title>
        <el-icon>
          <component :is="menu.icon"></component>
        </el-icon>
        <span>{{ menu.name }}</span>
      </template>
      <!--递归调用自身-->
      <MenuTree :menuList="menu.ch"></MenuTree>
    </el-sub-menu>
  </template>
</template>

<script setup>
import store from '../../store'

defineProps({
  menuList: {
    type: Array,
    required: false,
  }
})
</script>

<style scoped>

</style>
Router路由
菜单组件
递归显示组件