在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/...

相关推荐
IT_陈寒23 分钟前
Spring Boot 3.2 性能翻倍秘诀:这5个配置优化让你的应用起飞🚀
前端·人工智能·后端
b***676435 分钟前
【JavaEE】Spring Web MVC
前端·spring·java-ee
Mintopia40 分钟前
🧭 Claude Code 用户工作区最佳实践指南
前端·人工智能·claude
Mintopia1 小时前
🌐 多用户并发请求下的 WebAIGC 服务稳定性技术保障
javascript·人工智能·自动化运维
一 乐2 小时前
健身达人小程序|基于java+vue健身达人小程序的系统设计与实现(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·小程序
笑醉踏歌行4 小时前
NVM 在安装老版本 Node环境时,无法安装 NPM的问题
前端·npm·node.js
YUJIANYUE4 小时前
Gemini一次成型龙跟随鼠标html5+canvas特效
前端·计算机外设·html5
abiao19814 小时前
npm WARN ERESOLVE overriding peer dependency
前端·npm·node.js
TechExplorer3654 小时前
禁用 npm 更新检查
前端·npm·node.js
行云流水6267 小时前
uniapp pinia实现数据持久化插件
前端·javascript·uni-app