在Vue3 jsx 语法中使用 v-slots 插槽 和 component动态组件,实现一个递归遍历循环渲染的多层级菜单

前言

相信大家在使用vue3开发项目过程中肯定用到过 slots 插槽和 component 动态组件来封装一些业务组件吧,但大多数情况下都是在template 模板语法中去使用的。下面给大家分享一下我个人在项目开发过程中在vue3 jsx语法中去使用 v-slotscomponent 去封装一个递归循环的管理系统多层级菜单

首先看看在 template 模板语法中 slots 和component 是如何使用的

1.原template语法中使用 slot 插槽是通过 #title#icon 方式实现

2.动态组件通过component 方式实现

javascript 复制代码
<template>
  <a-menu>
    <a-sub-menu key="sub1" @titleClick="titleClick">
      <template #icon>
         <component :is="QqOutlined "></component >
      </template>
      <template #title>Navigation One</template>
      <a-menu-item-group key="g1">
        <template #icon>
          <component :is="QqOutlined "></component >
        </template>
        <template #title>Item 1</template>
        <a-menu-item key="1">Option 1</a-menu-item>
        <a-menu-item key="2">Option 2</a-menu-item>
      </a-menu-item-group>
      <a-menu-item-group key="g2" title="Item 2">
        <a-menu-item key="3">Option 3</a-menu-item>
        <a-menu-item key="4">Option 4</a-menu-item>
      </a-menu-item-group>
    </a-sub-menu>
  </a-menu>
</template>

而在jsx语法中插槽和动态组件的使用方法则有很大的不同

1.使用 v-slots 接收插槽参数 default 为默认插槽titleicon为ui组件的作用域插槽

2.动态的实现需要使用 hresolveComponent 这两个组合式api

javascript 复制代码
   const slot = {
       title: () => <span> {menu.meta.title}</span>,
       icon: () => (
         <span>{menu.meta.icon ? h(resolveComponent(menu.meta.icon)) : ""} </span>
       ),
     };
     <a-menu
       mode="inline"
       theme="dark"
       forceSubMenuRender={true}
     >
     <a-sub-menu key={menu.name} v-slots={slot}>
        <a-menu-item key={menu.name}>
           菜单内容
       </a-menu-item>
       </a-sub-menu>
     </a-menu>

完整的代码deom,实现一个menu递归遍历循环渲染的菜单

javascript 复制代码
<script>
import { useStore } from "vuex";
import { useRoute } from "vue-router";
import { computed, ref, watch, h, resolveComponent, defineComponent } from "vue";
export default defineComponent({
  setup() {
    const store = useStore();
    const route = useRoute();
    //当前选中项
    const selectedKeys = ref([route.name]);
    //当前展开项
    const openKeys = ref([route.meta.partentName || ""]);
    const menuTrees = computed(() => store.state.permission.addRouters);
    watch(route, (n, o) => {
      selectedKeys.value = [n.name];
    });

    // 生成 subMenu 菜单
    const renderSubMenu = (menu) => {
      const menuItem = [];
      menu.children.forEach((item) => menuItem.push(renderMenu(item)));
      const slot = {
        title: () => <span> {menu.meta.title}</span>,
        icon: () => (
          <span>{menu.meta.icon ? h(resolveComponent(menu.meta.icon)) : ""} </span>
        ),
      };
      return (
        <a-sub-menu key={menu.name} v-slots={slot}>
          {menuItem}
        </a-sub-menu>
      );
    };
    // 生成  menu-item 菜单
    const renderMenuItem = (menu) => {
      const slot = {
        icon: () => (
          <span>{menu.meta.icon ? h(resolveComponent(menu.meta.icon)) : ""} </span>
        ),
      };
      return (
        <a-menu-item key={menu.name} v-slots={slot}>
          <router-link to={menu.path}>{menu.meta.title}</router-link>
        </a-menu-item>
      );
    };

    const renderMenu = (menu) => {
      return menu.children ? renderSubMenu(menu) : renderMenuItem(menu);
    };

    const menuList = menuTrees.value.map((menu) => {
      return renderMenu(menu);
    });
    return () => (
      <a-menu
        mode="inline"
        theme="dark"
        forceSubMenuRender={true}
        v-model:selectedKeys={selectedKeys.value}
        v-model:openKeys={openKeys.value}
      >
        {menuList}
      </a-menu>
    );
  },
});
</script>

效果图:

如果当前脚手架不支持 jsx语法的编译 需要安装支持 jsx语法的包
javascript 复制代码
npm i @vitejs/plugin-vue-jsx -s

完整的项目代码地址可参考:gitee.com/ZHANG_6666/...

相关推荐
AndrewHZ1 分钟前
【GIS数据处理】什么是Cesium?从零入门Web端三维地理可视化工具
javascript·gis·web开发·cesium·gis前端·三维地理可视化
GISer_Jing4 分钟前
Next.js 15 全栈开发实战指南
开发语言·javascript·ecmascript
+VX:Fegn08954 分钟前
计算机毕业设计|基于springboot + vue服装商城系统(源码+数据库+文档)
数据库·vue.js·spring boot
JIngJaneIL5 分钟前
基于Java在线考试管理系统(源码+数据库+文档)
java·开发语言·数据库·vue.js·spring boot
凌康ACG6 分钟前
c++使用quickjs执行JavaScript
javascript·c++·quickjs
JIngJaneIL10 分钟前
基于Java音乐管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot
软件技术NINI11 分钟前
前端面试题
前端
接着奏乐接着舞15 分钟前
react hooks
前端·javascript·react.js
踢球的打工仔16 分钟前
前端html(3)
前端·算法·html
IDOlaoluo16 分钟前
nginx-sticky怎么用 Nginx 负载均衡添加 sticky 模块完整步骤
前端·chrome