您现在的位置是:首页 > 技术教程 正文

Vue3 Element-plus el-menu无限级菜单组件封装

admin 阅读: 2024-04-01
后台-插件-广告管理-内容页头部广告(手机)

对于element中提供给我们的el-menu组件最多可以实现三层嵌套,如果多一层数据只能自己通过变量去加一层,如果加了两层、三层这种往往是行不通的,所以只能进行封装

效果图

 

 一、定义数据

MenuData.ts

  1. export default [
  2. {
  3. id: "1",
  4. name: "第一级菜单",
  5. level: '1',
  6. child: [
  7. {
  8. id: "11",
  9. name: "第二级菜单",
  10. level: '1-1',
  11. child: [
  12. {
  13. id: "111",
  14. name: "第三级菜单",
  15. level: '1-1-1',
  16. child: [
  17. {
  18. id: "1111",
  19. name: "第四级菜单",
  20. level: '1-1-1-1',
  21. child: [
  22. {
  23. id: "11111",
  24. name: "第五级菜单",
  25. level: '1-1-1-1-1',
  26. child: []
  27. }
  28. ]
  29. }
  30. ]
  31. }]
  32. }
  33. ]
  34. },
  35. {
  36. id: "2",
  37. name: "第一级同级菜单",
  38. level: '2',
  39. child: []
  40. }
  41. ]

二、封装组件 

封装思想:

 1.对本身组件进行循环使用,如果有子集使用本身组件 把child数据传给自己

 2.如果没有子集 使用 el-menu-item

以下代码对setup( )函数和setup语法糖分别做了实现 

setup语法糖

  1. <template>
  2. <el-menu
  3. :default-active="defaultActive"
  4. :unique-opened="true"
  5. class="el-menu-vertical-demo"
  6. >
  7. <template v-for="item in menu">
  8. <template v-if="item.child && item.child.length > 0">
  9. <el-sub-menu
  10. :key="item.id"
  11. :index="item.level"
  12. :disabled="item.meta?.disabled"
  13. :popper-append-to-body="false"
  14. >
  15. <template #title>
  16. <i :class="[item.meta?.icon]">i>
  17. <span> {{ generateSpaces(item.level) }} span>
  18. <span slot="title"> {{ item.name }}span>
  19. template>
  20. <MenuTree
  21. :menu="item.child"
  22. :defaultActive="defaultActive"
  23. @clickItem="clickItemHandle"
  24. />
  25. el-sub-menu>
  26. template>
  27. <template v-else>
  28. <el-menu-item
  29. :key="item.id"
  30. :index="item.level"
  31. :disabled="item.meta?.disabled"
  32. :popper-append-to-body="false"
  33. @click="clickItemHandle(item)"
  34. >
  35. <i :class="[item.meta?.icon]">i>
  36. <span> {{ generateSpaces(item.level) }} span>
  37. <span slot="title">{{ item.name }}span>
  38. el-menu-item>
  39. template>
  40. template>
  41. el-menu>
  42. template>
  43. <script lang="ts" name="MenuTree" setup>
  44. // 把下面代码变成setup语法糖的形式
  45. import type { PropType } from "vue";
  46. import type { MenuItem } from "@/types/lesson";
  47. // type 为了方便写成这样 可以根据自己项目设定type
  48. defineProps({
  49. menu: {
  50. type: Array as unknown as PropType,
  51. required: true,
  52. default: () => [],
  53. },
  54. defaultActive: {
  55. type: String as unknown as PropType,
  56. required: true,
  57. default: [],
  58. },
  59. });
  60. const emit = defineEmits(["update-active-path", "clickItem"]);
  61. // 返回的空格字符串 用于显示菜单层级
  62. const generateSpaces = (level: string) => {
  63. let str = "";
  64. level.split("") .filter((it) => it != "-") .forEach(() => {
  65. str += " ";
  66. });
  67. return str;
  68. };
  69. // 点击当前菜单项
  70. const clickItemHandle = (item: MenuItem) => {
  71. emit("clickItem", item);
  72. };
  73. script>
  74. <style scoped lang="less">
  75. .el-menu {
  76. width: 288px;
  77. }
  78. style>

setup函数

  1. <template>
  2. <el-menu :default-active="defaultActive" :unique-opened="true" class="el-menu-vertical-demo" >
  3. <template v-for="item in menu">
  4. <template v-if="item.child && item.child.length > 0">
  5. <el-sub-menu
  6. :key="item.id"
  7. :index="item.level"
  8. :disabled="item.meta?.disabled"
  9. :popper-append-to-body="false"
  10. >
  11. <template #title>
  12. <i :class="[item.meta?.icon]">i>
  13. <span> {{ generateSpaces(item.level) }} span>
  14. <span slot="title"> {{ item.name }}span>
  15. template>
  16. <MenuTree :menu="item.child" :defaultActive="defaultActive" @clickItem="clickItemHandle" />
  17. el-sub-menu>
  18. template>
  19. <template v-else>
  20. <el-menu-item
  21. :key="item.id"
  22. :index="item.level"
  23. :disabled="item.meta?.disabled"
  24. :popper-append-to-body="false"
  25. @click="clickItemHandle(item)"
  26. >
  27. <i :class="[item.meta?.icon]">i>
  28. <span> {{ generateSpaces(item.level) }} span>
  29. <span slot="title">{{ item.name }}span>
  30. el-menu-item>
  31. template>
  32. template>
  33. el-menu>
  34. template>
  35. <script lang="ts">
  36. import { defineComponent, toRefs } from 'vue';
  37. import type { PropType } from 'vue'
  38. import type {MenuItem} from '@/types/lesson'
  39. export default defineComponent({
  40. name: 'MenuTree',
  41. props: {
  42. menu: {
  43. type: Array as unknown as PropType,
  44. required: true,
  45. default: () => [],
  46. },
  47. defaultActive: {
  48. type: String as unknown as PropType,
  49. required: true,
  50. default: '',
  51. },
  52. },
  53. emits: ['update-active-path','clickItem'],
  54. setup(props, context) {
  55. const { menu, defaultActive } = toRefs(props);
  56. const generateSpaces = (level:string) => {
  57. let str = ''
  58. level.split('').filter(it=>it!='-').forEach(() => {
  59. str += ' '
  60. })
  61. return str
  62. }
  63. const clickItemHandle = (item:MenuItem) => {
  64. context.emit('clickItem', item)
  65. }
  66. return {
  67. clickItemHandle,
  68. menu,
  69. defaultActive,
  70. generateSpaces,
  71. }
  72. },
  73. });
  74. script>
  75. <style scoped lang="less">
  76. .el-menu {
  77. width: 288px;
  78. }
  79. style>

 type就不补充了 可根据自己项目定义,可临时改成any

三、使用组件

  1. <template>
  2. <MenuTree
  3. :menu="menuList"
  4. :defaultActive="defaultActive"
  5. @clickItem="handleMenuClick"
  6. :update-click="handleMenuClick"
  7. />
  8. template>
  9. <script setup lang="ts">
  10. import MenuTree from "./components/MenuTree.vue";
  11. import type {MenuItem} from '@/types/lesson'
  12. import menuData from './MenuData'
  13. const defaultActive = ref(''); // "1-1-1-1" 默认选中的数据
  14. const menuList = ref(menuData)
  15. const handleMenuClick = (item:MenuItem) => {
  16. console.log('父组件',item);
  17. };
  18. script>

补充default-active变量,如果一开始想默认点开第一层的数据 就需要找规律啦

拿到所有的level,通过接口方式返给你 自己平铺拿到所有的level也好 

例如数据格式:

  1. let arr = [ "1-1",
  2. "1-1-1",
  3. "1-1-1-1",
  4. "1-1-1-2",
  5. "1-1-1-3",
  6. "1-1-1-4",
  7. "1-1-1-5",
  8. "1-1-1-6",
  9. "1-1-2",
  10. "1-1-2-1"
  11. ]

 想要的结果就是 最长且相同数字最多的元素 1-1-1-1

arr.sort((a,b)=> b.split('-').length - a.split('-').length)[0]

使用split防止有些字符串是10、11 两位数字的

标签:
声明

1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,请转载时务必注明文章作者和来源,不尊重原创的行为我们将追究责任;3.作者投稿可能会经我们编辑修改或补充。

在线投稿:投稿 站长QQ:1888636

后台-插件-广告管理-内容页尾部广告(手机)
关注我们

扫一扫关注我们,了解最新精彩内容

搜索
排行榜