IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> Vue和Vue-Element-Admin(八):根据menu数组或router动态渲染侧边栏 -> 正文阅读

[JavaScript知识库]Vue和Vue-Element-Admin(八):根据menu数组或router动态渲染侧边栏

目录

最基本方式

?通过menu数组渲染

?通过router动态渲染


一般情况下menu动态获取的(通过登录权限获取),Vue-Element-Admin则是通过动态获取router和child渲染menu;

最基本方式

直接copy使用elementui的侧边栏,使得侧边栏100%高,header固定60,main是100%-60px,侧边栏的menu写死,如下代码,

    <template>
  <el-container style="height: 100%">
    <el-aside width="200px" style="background-color: rgb(238, 241, 246)">
      <el-menu :default-openeds="['1', '3']">
        <el-menu-item index="1-3">首页</el-menu-item>
        <el-submenu index="1">
          <template slot="title"
            ><i class="el-icon-message"></i>用户管理</template
          >
          <el-menu-item index="1-3">商品管理</el-menu-item>
          <el-submenu index="1-4">
            <template slot="title">权限权利</template>
            <el-menu-item-group>
              <el-menu-item index="1-4-1">admin用户</el-menu-item>
              <el-menu-item index="1-4-2">root用户</el-menu-item>
            </el-menu-item-group>
          </el-submenu>
        </el-submenu>
      </el-menu>
    </el-aside>
    <el-container style="height: 100%">
      <el-header> 我是header </el-header>
      <el-main>我是Main</el-main>
    </el-container>
  </el-container>
</template>

    <script>
export default {
  name: 'HelloWorld',
  data() {
    return {}
  }
}
</script>
    <style lang="less" scoped>
.el-header {
  text-align: center;
  background-color: rgb(75, 73, 73);
}
.el-aside {
  height: 100%;
  text-align: left;
  line-height: 200px;
}

.el-main {
  //height: calc(100% - 60px);
  background-color: white;
  text-align: center;
  line-height: 160px;
}
.el-menu {
  height: 100%;
}
</style>

效果图

?通过menu数组渲染

登录后从后端返回menu数组后动态渲染是很常规做法,这里menu写在data里,computed里面过滤有无child的menu数组,无chid直接el-menu-item标签,有标签el-submenu标签,除了child的属性写在slot里面,chid写在el-menu-item-group里,这样就能够根据menu数组结构渲染左侧边栏:

    <template>
  <el-container style="height: 100%">
    <el-aside width="200px" style="background-color: rgb(238, 241, 246)">
      <el-menu>
        <el-menu-item
          v-for="item in hasNoChild"
          @click="clickmenu(item)"
          :key="item.name"
          :index="item.path"
        >
          <i :class="'el-icon-' + item.icon"></i>
          <span slot="title">{{ item.label }}</span>
        </el-menu-item>

        <el-submenu
          v-for="item in hasChild"
          :key="item.name"
          :index="item.path"
        >
          <template slot="title">
            <i :class="'el-icon-' + item.icon"></i>
            <span slot="title">{{ item.label }}</span>
          </template>
          <el-menu-item-group
            v-for="subItem in item.children"
            :key="subItem.path"
          >
            <el-menu-item @click="clickMenu(subItem)" :index="subItem.name">{{
              subItem.label
            }}</el-menu-item>
          </el-menu-item-group>
        </el-submenu>
      </el-menu>
    </el-aside>
    <el-container style="height: 100%">
      <el-header> 我是header </el-header>
      <el-main>我是Main</el-main>
    </el-container>
  </el-container>
</template>

    <script>
export default {
  name: 'HelloWorld',
  data() {
    return {
      menu: [
        {
          path: '/home/main',
          name: 'main',
          label: '首页',
          icon: 's-home',
          url: 'home/index'
        },
        {
          path: '/home/mall',
          name: 'mall',
          label: '商品管理',
          icon: 'video-play',
          url: 'mall/index'
        },
        {
          path: '/home/user',
          name: 'user',
          label: '用户管理',
          icon: 'user',
          url: 'User/index'
        },
        {
          label: '其他',
          icon: 'location',
          name: 'other',
          path: 'nothing',
          children: [
            {
              path: '/page1',
              name: 'page1',
              label: '页面1',
              icon: 'setting',
              url: 'Other/PageOne.vue'
            },
            {
              path: '/page2',
              name: 'page2',
              label: '页面2',
              icon: 'setting',
              url: 'Other/PageTwo.vue'
            }
          ]
        }
      ]
    }
  },
  computed: {
    hasChild() {
      return this.menu.filter((item) => item.children)
    },
    hasNoChild() {
      return this.menu.filter((item) => !item.children)
    }
  },
  methods: {
    clickmenu(item) {
      this.$router.push({
        name: item.name
      })
    }
  }
}

?通过router动态渲染

Vue-Element-Admin封装了侧边栏展示,通过对应router的child个数来渲染menu,404等页面是不在layout下的,其他每个router必须是layout的child,sidebaritem.vue这样写的,如果这个路由只有1个child,那么就默认把你当第1层el-menu-item渲染,如果有多个非隐藏child,那就会被渲染成子menu;

<template>
  <div v-if="!item.hidden">
    <template v-if="hasOneShowingChild(item.children,item) && (!onlyOneChild.children||onlyOneChild.noShowingChildren)&&!item.alwaysShow">
      <app-link v-if="onlyOneChild.meta" :to="resolvePath(onlyOneChild.path)">
        <el-menu-item :index="resolvePath(onlyOneChild.path)" :class="{'submenu-title-noDropdown':!isNest}">
          <item :icon="onlyOneChild.meta.icon||(item.meta&&item.meta.icon)" :title="onlyOneChild.meta.title" />
        </el-menu-item>
      </app-link>
    </template>

    <el-submenu v-else ref="subMenu" :index="resolvePath(item.path)" popper-append-to-body>
      <template slot="title">
        <item v-if="item.meta" :icon="item.meta && item.meta.icon" :title="item.meta.title" />
      </template>
      <sidebar-item
        v-for="child in item.children"
        :key="child.path"
        :is-nest="true"
        :item="child"
        :base-path="resolvePath(child.path)"
        class="nest-menu"
      />
    </el-submenu>
  </div>
</template>

<script>
import path from 'path'
import { isExternal } from '@/utils/validate'
import Item from './Item'
import AppLink from './Link'
import FixiOSBug from './FixiOSBug'

export default {
  name: 'SidebarItem',
  components: { Item, AppLink },
  mixins: [FixiOSBug],
  props: {
    // route object
    item: {
      type: Object,
      required: true
    },
    isNest: {
      type: Boolean,
      default: false
    },
    basePath: {
      type: String,
      default: ''
    }
  },
  data() {
    // To fix https://github.com/PanJiaChen/vue-admin-template/issues/237
    // TODO: refactor with render function
    this.onlyOneChild = null
    return {}
  },
  methods: {
    hasOneShowingChild(children = [], parent) {
      const showingChildren = children.filter(item => {
        if (item.hidden) {
          return false
        } else {
          // Temp set(will be used if only has one showing child)
          this.onlyOneChild = item
          return true
        }
      })

      // When there is only one child router, the child router is displayed by default
      if (showingChildren.length === 1) {
        return true
      }

      // Show parent if there are no child router to display
      if (showingChildren.length === 0) {
        this.onlyOneChild = { ... parent, path: '', noShowingChildren: true }
        return true
      }

      return false
    },
    resolvePath(routePath) {
      if (isExternal(routePath)) {
        return routePath
      }
      if (isExternal(this.basePath)) {
        return this.basePath
      }
      return path.resolve(this.basePath, routePath)
    }
  }
}
</script>

?

  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2022-04-30 08:37:40  更:2022-04-30 08:38:55 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 2:12:10-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码