vue2根据路由动态生成menu菜单

1、需要先获得路由数据

路由数据可以是routes.js中的路由数据,也可以是服务返回的数组

route.js中的数据如下

[ 复制代码
    {
        // 骨架路由/顶层路由
        path: '/',
        redirect: '/home',
        component: () => import('@/layout/index.vue'),
        meta: {
            hidden: true,
            title: '权限管理',
            icon: 'lock',
        },
        children: [
            {
                path: '/home',
                component: () => import('@/views/home/index.vue'),
                meta: {
                    hidden: true,
                    title: '权限管理',
                    icon: 'lock',
                },
            },
        ]
    },
    {
        path: '/name',
        component: () => import('@/layout/index.vue'),
        redirect: '/setting/base',
        meta: {
            hidden: true,
            title: '权限管理',
            icon: 'lock',
        },
        children: [
            {
                path: '/setting/base',
                component: () => import('@/views/product/attr'),
                meta: {
                    hidden: true,
                    title: '权限管理',
                    icon: 'lock',
                },
            }
        ]
    },
    {
        path: '/acl',
        component: () => import('@/layout/index.vue'),
        name: 'Acl',
        redirect: '/acl/user',
        meta: {
            hidden: true,
            title: '权限管理',
            icon: 'lock',
        },
        children: [
            {
                path: '/acl/user',
                component: () => import('@/views/acl/user/index.vue'),
                name: 'User',
                meta: {
                    hidden: true,
                    title: '用户管理',
                    icon: 'user',
                },
            },
            {
                path: '/acl/role',
                component: () => import('@/views/acl/role/index.vue'),
                name: 'Role',
                meta: {
                    hidden: true,
                    title: '角色管理',
                    icon: 'userFilled',
                },
            },
            {
                path: '/acl/permission',
                component: () => import('@/views/acl/permission/index.vue'),
                name: 'Permission',
                meta: {
                    hidden: true,
                    title: '菜单管理',
                    icon: 'monitor',
                },
            },
        ],
    },
    {
        path: '/product',
        name: 'Product',
        component: () => import('@/layout/index.vue'),
        meta: {
            title: '商品管理',
            icon: 'goods',
            hidden: true,
        },
        redirect: '/product/trademark',
        children: [
            {
                path: '/product/trademark',
                name: 'Trademark',
                component: () => import('@/views/product/trademark/index.vue'),
                meta: {
                    title: '品牌管理',
                    icon: 'collectionTag',
                    hidden: true,
                }
            },
            {
                path: '/product/attr',
                name: 'Attr',
                component: () => import('@/views/product/attr/index.vue'),
                meta: {
                    title: '属性管理',
                    icon: 'goods',
                    hidden: true,
                }
            },
        ]
    },
    {

        path: '/login',
        name: 'login',
        component: () => import('@/views/login/index.vue'),
        meta: {
            title: '属性管理',
            icon: 'SwitchFilled',
            hidden: false,
        }
    },
    {
        // 404路由
        path: '*',
        name: '404',
        component: () => import('@/views/404/index.vue'),
        meta: {
            title: '属性管理',
            icon: 'SwitchFilled',
            hidden: false,
        }
    }
]

其中layout为骨架组件,大部分需要展示的组件都是layout的子组件,在layout中进行布局页面,设置三个组件

  1. menu菜单
  2. tabbar导航栏 --> 面包屑
  3. main 内容展示区域
js 复制代码
<router-view></router-view>
我们将路由出口标签放在main组件中

meta为路由元信息,可以设置属性来控制路由的显示与隐藏和icon图标

进入正题

先展示一下组件嵌套关系,切记el-menu标签(element-ui)是放在layout中的index.vue中的,因为这里需要使用到递归组件,需要传递props属性

  1. menu组件中引入route.js中的路由数据。
  2. menu组件中使用el-menu标签,中的子标签。
javascript 复制代码
import routes from "@/xxx/xx/routes.js"

<div>
    <template v-for="item in menuList">
      <!-- 路由没有子路由 -->
      <template v-if="!item.children">
        <!-- 通过路由元信息:meta,控制路由是否动态生成 -->
        <el-menu-item :index="item.path" :key="item.id" v-if="item.meta.hidden">
          <!-- icon图标 -->
          <i :class="item.icon"></i>
          <!-- menu菜单名 -->
          <span slot="title">{{ item.meta.title }}</span>
        </el-menu-item>
      </template>

      <!-- 路由有子路由路由 且只有一个子路由,这个可有可无 -->
        
      <template v-if="item.children && item.children.length == 1">
        <el-menu-item :index="item.path" :key="item.id" v-if="item.meta.hidden">
          <!-- icon图标 -->
          <i :class="item.icon"></i>
          <!-- menu菜单名 -->
          <span slot="title">{{ item.meta.title }}</span>
        </el-menu-item>
      </template>

      <!-- 路由有子路由路由 且有多个子路由 -->
      <template v-if="item.children && item.children.length > 1">
        <el-submenu :index="item.path" :key="item.id" v-if="item.meta.hidden">
          <template slot="title">
            <!-- icon图标 -->
            <i :class="item.icon"></i>
            <!-- menu菜单名 -->
            <span slot="title">{{ item.meta.title }}</span>
          </template>

          <!-- submenu里面的内容通过组件递归生成 -->
          <menuInfo :menuList="item.children"></menuInfo>
        </el-submenu>
      </template>
    </template>
  </div>

<script>
export default {
  name: "menuInfo",
  data() {
    return {};
  },
  props: ["menuList"],
};
</script>

其中在vue2中template标签在使用v-for时不能绑定关键字key,那么我们就把key设置到el-menu-item标签身上。一般情况下不使用el-menu-item-group标签,el-menu-item标签够用。 menulist为父组件传递进来的数据。

组件递归需要注意的点:

  1. 递归组件必须有name属性,并且name的值必须和父组件引入组件时命名相同
  2. 如果在vue3中,setup写法中没有name属性,需要使用vue2的形式重命名才能进行递归
相关推荐
dy17172 小时前
element-plus表格默认展开有子的数据
前端·javascript·vue.js
2501_915918416 小时前
Web 前端可视化开发工具对比 低代码平台、可视化搭建工具、前端可视化编辑器与在线可视化开发环境的实战分析
前端·低代码·ios·小程序·uni-app·编辑器·iphone
程序员的世界你不懂7 小时前
【Flask】测试平台开发,新增说明书编写和展示功能 第二十三篇
java·前端·数据库
索迪迈科技7 小时前
网络请求库——Axios库深度解析
前端·网络·vue.js·北京百思可瑞教育·百思可瑞教育
gnip7 小时前
JavaScript二叉树相关概念
前端
attitude.x8 小时前
PyTorch 动态图的灵活性与实用技巧
前端·人工智能·深度学习
β添砖java8 小时前
CSS3核心技术
前端·css·css3
空山新雨(大队长)8 小时前
HTML第八课:HTML4和HTML5的区别
前端·html·html5
猫头虎-前端技术8 小时前
浏览器兼容性问题全解:CSS 前缀、Grid/Flex 布局兼容方案与跨浏览器调试技巧
前端·css·node.js·bootstrap·ecmascript·css3·媒体