iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么使用SpringBoot+Vue实现动态菜单
  • 108
分享到

怎么使用SpringBoot+Vue实现动态菜单

2023-07-02 18:07:28 108人浏览 薄情痞子
摘要

本篇内容介绍了“怎么使用SpringBoot+Vue实现动态菜单”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 整体思路效果图:最终菜单

本篇内容介绍了“怎么使用SpringBoot+Vue实现动态菜单”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

1. 整体思路

效果图:

怎么使用SpringBoot+Vue实现动态菜单

最终菜单显示效果类似上图,我把这里的菜单分为了四类:

有父有子:像系统管理那种,既有父菜单,又有子菜单。

只有一个一级菜单,这种又细分为三种情况:

  • 普通的菜单,点击之后在右边主页面打开某个功能页面。

  • 一个超链接,但不是外链,是一个在当前系统中打开的外部网页,点击之后,会在右边的主页面中新开一个选项卡,这个选项卡中显示的是一个外部网页(本质上是通过 iframe 标签引入的一个外部网页)。

  • 一个超链接,并且还是一个外链,点击之后,直接在浏览器中打开一个新的选项卡,新的选项卡中展示一个外部链接。

整体上来说,就分为这四种情况。其中 1、2.1、2.3 应该都好理解,2.2 有的小伙伴可能不清楚,我给大家截个图看下就知道了:

怎么使用SpringBoot+Vue实现动态菜单

四种菜单对应的 JSON 格式分别如下:

有父有子:

{ "name": "Monitor", "path": "/monitor", "hidden": false, "redirect": "noRedirect", "component": "Layout", "alwaysshow": true, "meta": {  "title": "系统监控",  "icon": "monitor",  "noCache": false,  "link": null }, "children": [{  "name": "Online",  "path": "online",  "hidden": false,  "component": "monitor/online/index",  "meta": {   "title": "在线用户",   "icon": "online",   "noCache": false,   "link": null  } }, {  "name": "Job",  "path": "job",  "hidden": false,  "component": "monitor/job/index",  "meta": {   "title": "定时任务",   "icon": "job",   "noCache": false,   "link": null  } }]}

只有一个一级菜单,且一级菜单点击后是一个功能页面:

{ "path": "/", "hidden": false, "component": "Layout", "children": [{  "name": "Role",  "path": "role",  "hidden": false,  "component": "system/role/index",  "meta": {   "title": "角色管理",   "icon": "peoples",   "noCache": false,   "link": null  } }]}

只有一个一级菜单,且一级菜单点击之后在当前系统中一个新的选项卡里打开一个网页:

{    "name": "Http://www.javaboy.org",    "path": "/",    "hidden": false,    "component": "Layout",    "meta": {        "title": "TienChin健身官网",        "icon": "guide",        "noCache": false,        "link": null    },    "children": [        {            "name": "Www.javaboy.org",            "path": "www.javaboy.org",            "hidden": false,            "component": "InnerLink",            "meta": {                "title": "TienChin健身官网",                "icon": "guide",                "noCache": false,                "link": "http://www.javaboy.org"            }        }    ]}

只有一个一级菜单,且一级菜单点击之后在浏览器打开一个新的选项卡:

{    "name": "Http://www.javaboy.org",    "path": "http://www.javaboy.org",    "hidden": false,    "component": "Layout",    "meta": {        "title": "TienChin健身官网",        "icon": "guide",        "noCache": false,        "link": "http://www.javaboy.org"    }}

根据以上四种不同的 jsON,我们总结出以下规律:

  • 父组件都是 Layout,这里的 Layout 就相当于我们 vhr 中的 Home 组件,也就是整个页面的框架

  • 如果想在当前系统中,新开选项卡打开一个功能项,那么这个菜单项必然有 children,即使 children 中只有一项菜单。

  • 如果菜单项是一个外链,那么这个菜单项就不需要有 children 了。

  • 某种程度上,我们其实可以将 2、3 归为一类,毕竟 3 只是展示内容的组件固定为 InnerLink,2 则视情况而定。

  • 整体上,可以点击的菜单的 path 都是父菜单的 path + 子菜单的 path,如果菜单项有父有子,那就正常拼接就行了;如果只有一个子菜单,那么父菜单的 path 就是 /;如果是一个外链,那就只有父菜单的 path 了。

好了,这就是动态菜单的整体设计。

2. 前端渲染

接下来我们再来看一看前端的菜单渲染,前端的动态菜单渲染位于 tienchin-ui/src/layout/components/Sidebar/SidebarItem.vue 文件中:

<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, onlyOneChild.query)">        <el-menu-item :index="resolvePath(onlyOneChild.path)" :class="{ 'submenu-title-noDropdown': !isNest }">          <svg-icon :icon-class="onlyOneChild.meta.icon || (item.meta && item.meta.icon)"/>          <template #title><span class="menu-title" :title="hasTitle(onlyOneChild.meta.title)">{{ onlyOneChild.meta.title }}</span></template>        </el-menu-item>      </app-link>    </template>    <el-sub-menu v-else ref="subMenu" :index="resolvePath(item.path)" popper-append-to-body>      <template v-if="item.meta" #title>        <svg-icon :icon-class="item.meta && item.meta.icon" />        <span class="menu-title" :title="hasTitle(item.meta.title)">{{ item.meta.title }}</span>      </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-sub-menu>  </div></template>

这里涉及到几个方法,具体的方法细节我就不贴出来了,主要和大家说下实现思路。

  • 先看整体上,这个菜单要是非隐藏的,隐藏的菜单,那么直接一级菜单及其下的子菜单就都不渲染了。

  • 渲染整体上分两块,上面的 template 主要是渲染只有一个子菜单的情况,也就是第一小节的 2、3、4 三种情况,下面的渲染正常的有父有子的情况,也就是第一小节的菜单 1。

  • hasOneShowingChild 主要是判断这个菜单项是否只有一个需要渲染的子菜单,如果有多个子菜单,但是大部分都是隐藏,只有一个需要渲染出来,那也算只有一个子菜单,如果一个菜单项都没有子菜单,那也算一个子菜单,只不过这个子菜单就是他自身,对应第一小节第 4 种情况。在判断的过程中,将唯一需要渲染的菜单的数据赋值给 onlyOneChild 变量,那么最终,如果当前菜单项只有一个子菜单,且这个子菜单没有子菜单(或者有子菜单但是子菜单不用显示),并且当前菜单也不是必须要渲染的,那就将 onlyOneChild 的数据渲染出来。

  • 对于普通的有父有子的情况,渲染的时候,通过 el-sub-menu 标签进行渲染,但是注意子项是 sidebar-item,sidebar-item 其实就是当前项!换言之,这里的渲染其实还用到了递归(直到没有 children 的时候结束),这样即便菜单有三级四级五级等等,只要不嫌难看,都是可以渲染出来的。

3. 后端菜单生成

3.1 菜单表

首先我们来看看菜单表的定义,也就是 sys_menu

CREATE TABLE `sys_menu` (  `menu_id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '菜单ID',  `menu_name` varchar(50) COLLATE utf8mb4_unicode_ci NOT NULL COMMENT '菜单名称',  `parent_id` bigint(20) DEFAULT '0' COMMENT '父菜单ID',  `order_num` int(4) DEFAULT '0' COMMENT '显示顺序',  `path` varchar(200) COLLATE utf8mb4_unicode_ci DEFAULT '' COMMENT '路由地址',  `component` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '组件路径',  `query` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '路由参数',  `is_frame` int(1) DEFAULT '1' COMMENT '是否为外链(0是 1否)',  `is_cache` int(1) DEFAULT '0' COMMENT '是否缓存(0缓存 1不缓存)',  `menu_type` char(1) COLLATE utf8mb4_unicode_ci DEFAULT '' COMMENT '菜单类型(M目录 C菜单 F按钮)',  `visible` char(1) COLLATE utf8mb4_unicode_ci DEFAULT '0' COMMENT '菜单状态(0显示 1隐藏)',  `status` char(1) COLLATE utf8mb4_unicode_ci DEFAULT '0' COMMENT '菜单状态(0正常 1停用)',  `perms` varchar(100) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '权限标识',  `icon` varchar(100) COLLATE utf8mb4_unicode_ci DEFAULT '#' COMMENT '菜单图标',  `create_by` varchar(64) COLLATE utf8mb4_unicode_ci DEFAULT '' COMMENT '创建者',  `create_time` datetime DEFAULT NULL COMMENT '创建时间',  `update_by` varchar(64) COLLATE utf8mb4_unicode_ci DEFAULT '' COMMENT '更新者',  `update_time` datetime DEFAULT NULL COMMENT '更新时间',  `remark` varchar(500) COLLATE utf8mb4_unicode_ci DEFAULT '' COMMENT '备注',  PRIMARY KEY (`menu_id`)) ENGINE=InnoDB AUTO_INCREMENT=3054 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='菜单权限表';

其实这里很多字段都和我们 vhr 项目项目很相似,我也就不重复啰嗦了,我这里主要和小伙伴们说一个字段,那就是 menu_type

menu_type 表示一个菜单字段的类型,一个菜单有三种类型,分别是目录(M)、菜单(C)以及按钮(F)。这里所说的目录,相当于我们在 vhr 中所说的一级菜单,菜单相当于我们在 vhr 中所说的二级菜单。

当用户从前端登录成功后,要去动态加载的菜单的时候,就查询 M 和 C 类型的数据即可,F 类型的数据不是菜单项,查询的时候直接过滤掉即可,通过 menu_type 这个字段可以轻松的过滤掉 F 类型的数据。小伙伴们想想,F 类型的数据过滤掉之后,剩下的数据不就是一级菜单和二级菜单了,那不就和 vhr 又一样了么!

怎么使用SpringBoot+Vue实现动态菜单

在 vhr 中,考虑到菜单就是只有两级:一级菜单和二级菜单,一级菜单是目录,二级菜单是则是具体的菜单项,没有三级菜单!所以在 vhr 中,查询菜单的时候我直接用了一个一对多的查询,将一级菜单做一的一方,二级菜单做多的一方,这样比较省事。当然灵活度差一点,所以在 TienChin 项目中,这块还是用上了递归。

3.2 菜单接口

当用户登录成功之后,会自动请求 /getRouters 接口来获取菜单信息,我们一起来看下:

@GetMapping("getRouters")public ajaxResult getRouters() {    Long userId = SecurityUtils.getUserId();    List<SysMenu> menus = menuService.selectMenuTreeByUserId(userId);    return AjaxResult.success(menuService.buildMenus(menus));}

这里的查询实际上分为两个步骤:

  • 根据用户 id 查询到所有的菜单信息,这一步的查询实际上是比较容易的,就单纯的多张表联合在一起,然后过滤出和当前用户相关并且菜单类型为 M 或者 C 的菜单(类型为 F 的表示按钮,就不要了),查询到菜单信息之后,然后进行一个递归操作,将菜单数据的层级排列出来。

  • menuService.buildMenus 这一步则是将菜单数据专为前端所需要的路由数据。

一共就这两个步骤,我们来逐一进行分析。

先来看查询菜单数据。

@Overridepublic List<SysMenu> selectMenuTreeByUserId(Long userId) {    List<SysMenu> menus = null;    if (SecurityUtils.isAdmin(userId)) {        menus = menuMapper.selectMenuTreeAll();    } else {        menus = menuMapper.selectMenuTreeByUserId(userId);    }    return getChildPerms(menus, 0);}public List<SysMenu> getChildPerms(List<SysMenu> list, int parentId) {    List<SysMenu> returnList = new ArrayList<SysMenu>();    for (Iterator<SysMenu> iterator = list.iterator(); iterator.hasNext(); ) {        SysMenu t = (SysMenu) iterator.next();        // 一、根据传入的某个父节点ID,遍历该父节点的所有子节点        if (t.getParentId() == parentId) {            recursionFn(list, t);            returnList.add(t);        }    }    return returnList;}private void recursionFn(List<SysMenu> list, SysMenu t) {    // 得到子节点列表    List<SysMenu> childList = getChildList(list, t);    t.setChildren(childList);    for (SysMenu tChild : childList) {        if (hasChild(list, tChild)) {            recursionFn(list, tChild);        }    }}private List<SysMenu> getChildList(List<SysMenu> list, SysMenu t) {    List<SysMenu> tlist = new ArrayList<SysMenu>();    Iterator<SysMenu> it = list.iterator();    while (it.hasNext()) {        SysMenu n = (SysMenu) it.next();        if (n.getParentId().longValue() == t.getMenuId().longValue()) {            tlist.add(n);        }    }    return tlist;}private boolean hasChild(List<SysMenu> list, SysMenu t) {    return getChildList(list, t).size() > 0;}

这里一共涉及到五个关键方法,我们来逐一进行分析:

  • selectMenuTreeByUserId:这个方法的执行比较容易,如果当前用户是管理员,那就不用加过滤条件了,直接查询出所有的类型为 M 和 C 的菜单项即可。

  • getChildPerms:这个方法主要是将前面查询出来的菜单数据进行重组,本来都是一个集合中的数据,现在在该方法中处理成树状,处理的核心逻辑就是调用 recursionFn 方法将之进行递归。

  • recursionFn:这是最为关键的递归方法了,首先调用 getChildList 获取当前菜单项的 children,然后将获取到的 children 设置给当前菜单项,最后还要遍历获取到的 children,如果这个 children 也是有子菜单的,则继续调用 recursionFn 方法进行处理。

  • getChildList:这个是查询某一个菜单的子菜单,这个很容易,如果某一个菜单的 parentId 是当前菜单的 id,那么这个菜单就是当前菜单的子菜单。

  • hasChild:这个是判断给定的菜单是否有子菜单,这个逻辑就比较简单了。

好啦,这个就是整个的查询逻辑,整体上来说是比较容易的,就是查询 M 和 C 类型的菜单,然后再做一个递归操作,将菜单数据变成一个树状数据。

但是因为 SysMenu 和前后端所需要的路由数据的字段名称对不上,并且格式参数等都不符合前端的要求,所以还需要再做一个转换,这就是 menuService.buildMenus 所做的事情了:

@Overridepublic List<RouterVo> buildMenus(List<SysMenu> menus) {    List<RouterVo> routers = new LinkedList<RouterVo>();    for (SysMenu menu : menus) {        RouterVo router = new RouterVo();        router.setHidden("1".equals(menu.getVisible()));        router.setName(getRouteName(menu));        router.setPath(getRouterPath(menu));        router.setComponent(getComponent(menu));        router.setQuery(menu.getQuery());        router.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon(), StringUtils.equals("1", menu.getIsCache()), menu.getPath()));        List<SysMenu> cMenus = menu.getChildren();        if (!cMenus.isEmpty() && cMenus.size() > 0 && UserConstants.TYPE_DIR.equals(menu.getMenuType())) {            router.setAlwaysShow(true);            router.setRedirect("noRedirect");            router.setChildren(buildMenus(cMenus));        } else if (isMenuFrame(menu)) {            router.setMeta(null);            List<RouterVo> childrenList = new ArrayList<RouterVo>();            RouterVo children = new RouterVo();            children.setPath(menu.getPath());            children.setComponent(menu.getComponent());            children.setName(StringUtils.capitalize(menu.getPath()));            children.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon(), StringUtils.equals("1", menu.getIsCache()), menu.getPath()));            children.setQuery(menu.getQuery());            childrenList.add(children);            router.setChildren(childrenList);        } else if (menu.getParentId().intValue() == 0 && isInnerLink(menu)) {            router.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon()));            router.setPath("/");            List<RouterVo> childrenList = new ArrayList<RouterVo>();            RouterVo children = new RouterVo();            String routerPath = innerLinkReplaceEach(menu.getPath());            children.setPath(routerPath);            children.setComponent(UserConstants.INNER_LINK);            children.setName(StringUtils.capitalize(routerPath));            children.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon(), menu.getPath()));            childrenList.add(children);            router.setChildren(childrenList);        }        routers.add(router);    }    return routers;}

从这个方法的执行逻辑上我们可以看到,这里的菜单数据一共分为了四种情况,其实刚好就和我们第一小节所介绍的情况相对应。

整体上来看,分支语句外面设置了组件的最基本的属性。三个分支语句:

  • 第一个分支,处理普通的有父有子的情况。

  • 第二个分支,处理第一小节第二种情况。

  • 第三个分支,处理第一小节第三种情况。

  • 如果三个分支都没进去,那就是第一小节的第四种情况,以及各个子菜单的情况了。

好了,基于这样大的思路,再来看各个属性的具体设置,就很容易了。

  • 首先是可见性 hidden,这个没啥好说的。

  • 接下来是菜单的 name 属性,name 属性分为了两种情况:路由的 name 属性是菜单表中的 path 字段值且首字母大写(菜单 1、3、4);如果在一级菜单中,出现了一个菜单 C(本来这一级别只有 M),并且还不是外链,那么就设置菜单的 name 为空字符串(相当于此时不需要 name 属性了,对应菜单 2 的情况)。

  • 接下来是路由的 path,设置 path 的时候也分好种情况,松哥对照着代码来和大家说一下:

public String getRouterPath(SysMenu menu) {    String routerPath = menu.getPath();    // 内链打开外网方式    if (menu.getParentId().intValue() != 0 && isInnerLink(menu)) {        routerPath = innerLinkReplaceEach(routerPath);    }    // 非外链并且是一级目录(类型为目录)    if (0 == menu.getParentId().intValue() && UserConstants.TYPE_DIR.equals(menu.getMenuType())            && UserConstants.NO_FRAME.equals(menu.getIsFrame())) {        routerPath = "/" + menu.getPath();    }    // 非外链并且是一级目录(类型为菜单)    else if (isMenuFrame(menu)) {        routerPath = "/";    }    return routerPath;}

a. 首先获取从数据库中查询到的 path 属性。b. 如果当前组件不是一级菜单,并且是在内部组件中展示,那么除去这个 path 里边的 http 或者 https(对应菜单 3 的 children 的情况)。c. 如果当前组件是一级菜单并且是 M 型并且不是外链,那么就在原有的 path 上加上 / 前缀(对应菜单 1 的一级菜单的 path 情况)。d. 如果当前组件是一级菜单,且是 C 型菜单,那么设置 path 为 /(对应菜单 2、3 中一级菜单的 path 情况)。e. 其他情况,菜单都是从数据库查到什么返回什么。

接下来是设置前端 component,这个菜单项用哪个 component 组件显示出来。

public String getComponent(SysMenu menu) {    String component = UserConstants.LAYOUT;    if (StringUtils.isNotEmpty(menu.getComponent()) && !isMenuFrame(menu)) {        component = menu.getComponent();    } else if (StringUtils.isEmpty(menu.getComponent()) && menu.getParentId().intValue() != 0 && isInnerLink(menu)) {        component = UserConstants.INNER_LINK;    } else if (StringUtils.isEmpty(menu.getComponent()) && isParentView(menu)) {        component = UserConstants.PARENT_VIEW;    }    return component;}

a. 首先默认的组件是 Layout(菜单1、2、3、4 的一级菜单)。b. 如果配置的时候就有 component,并且当前菜单项也不是外链,那么就使用配置的 component(菜单 1、2 的子菜单情况)。c. 如果不是一级菜单(是一个子菜单),并且是一个在当前系统展示的外链,那么就使用 InnerLink 这个组件(这个组件中有一个 iframe 标签可以把外链展示出来,如菜单 4 的子菜单情况)。d. 如果配置的时候没有设置组件并且菜单类型是 M(二级菜单中还有三级菜单的情况),那么就设置显示组件为 ParentView。

component 就分为这几种情况。

接下来就是 query 和 meta 这两个参数就没啥好说的。

接下来就是三个分支的情况了。

“怎么使用springBoot+Vue实现动态菜单”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: 怎么使用SpringBoot+Vue实现动态菜单

本文链接: https://www.lsjlt.com/news/343580.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么使用SpringBoot+Vue实现动态菜单
    本篇内容介绍了“怎么使用SpringBoot+Vue实现动态菜单”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 整体思路效果图:最终菜单...
    99+
    2023-07-02
  • SpringBoot+Vue实现动态菜单的思路梳理
    目录1. 整体思路2. 前端渲染3. 后端菜单生成3.1 菜单表3.2 菜单接口关于 Spring Boot + Vue3 的动态菜单,松哥之前已经写了两篇文章了,这两篇文章主要是从...
    99+
    2024-04-02
  • vue实现三级联动动态菜单
    本文实例为大家分享了vue实现三级联动动态菜单的具体代码,供大家参考,具体内容如下 三级联动动态菜单展示:一级菜单选中,生成二级菜单数据,二级菜单选中,生成三级菜单数据(根据上一级菜...
    99+
    2024-04-02
  • vue+quasar使用递归实现动态多级菜单
    本文实例为大家分享了vue+quasar使用递归实现动态多级菜单的具体代码,供大家参考,具体内容如下 效果图: 菜单初始化入口 menu.vue,初始化侧边栏菜单组建,<my...
    99+
    2024-04-02
  • vue动态菜单怎么设置
    随着前端开发技术的不断发展,许多前端框架也一直在不断完善和发展,其中Vue框架以其简单易学、高效便捷的特性被越来越多的开发者所青睐。在Vue开发中,经常需要使用动态菜单,那么Vue动态菜单怎么设置呢?下面本文将为大家讲解一下Vue动态菜单的...
    99+
    2023-05-20
  • DjangoVue实现动态菜单和动态权限
    目录用户与用户组的架构设计动态菜单和权限的设计思路与实现Vue 端如何实现动态路由Django 端如何实现动态权限随着前后端分离架构的流行,在 web 应用中,RESTful API...
    99+
    2024-04-02
  • 怎么使用JS实现三级联动菜单
    这篇“怎么使用JS实现三级联动菜单”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“怎么使用J...
    99+
    2024-04-02
  • 怎么使用Vue组件tree实现树形菜单
    本篇内容主要讲解“怎么使用Vue组件tree实现树形菜单”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么使用Vue组件tree实现树形菜单”吧!vue 编写的树形菜单,小巧实用,支持vue1....
    99+
    2023-07-04
  • 使用vue-element-admin框架从后端动态获取菜单功能的实现
    目录2、详解2.1、新增asyncRoutes路由2.2、新建permission.js文件2.3、在vuex中注册permission模块2.4、在getters中增加路由状态2....
    99+
    2024-04-02
  • vue怎么实现菜单栏滑动切换特效
    现代网页设计强调用户体验和交互性,让用户可以方便快捷地完成自己的目的。为了实现这一目标,很多网站都会采用滑动效果,例如点击菜单栏上的某个链接,网页就会平滑地滑动到相应位置,有效地提供了用户体验,提高了网站可用性。Vue.js作为一种流行的J...
    99+
    2023-05-14
  • css如何实现动态二级菜单
    这篇文章将为大家详细讲解有关css如何实现动态二级菜单,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。动态实现简单的二级菜单当鼠标放到一级标签上时,鼠标会变成小手的形状 展示二级菜单,源码如下,复制即可直接...
    99+
    2023-06-08
  • 利用ASP.Net Core中的Razor实现动态菜单
    目录准备1.实现思路2.编码2.1  建立渲染内容填充方法2.2  建立分部视图2.3 调用分布视图准备 1.框架 .netcore  版本 ...
    99+
    2024-04-02
  • Vue使用element-ui实现菜单导航
    本文实例为大家分享了Vue使用element-ui实现菜单导航的具体代码,供大家参考,具体内容如下 效果图 目录截图 安装vue-router 和 element-ui vue-...
    99+
    2024-04-02
  • react怎么实现浮动菜单
    本教程操作环境:Windows10系统、react18.0.0版、Dell G3电脑。react怎么实现浮动菜单?React中hover悬浮菜单的做法对于悬浮菜单,主要是借助html标签的事件机制,或者hover来实现,先看下效果图:当鼠标...
    99+
    2023-05-14
    React
  • 怎么使用Vue递归组件实现树形菜单
    本文小编为大家详细介绍“怎么使用Vue递归组件实现树形菜单”,内容详细,步骤清晰,细节处理妥当,希望这篇“怎么使用Vue递归组件实现树形菜单”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。效果如下图,点击后打开二级...
    99+
    2023-07-04
  • vue怎么实现右键菜单栏
    本篇内容主要讲解“vue怎么实现右键菜单栏”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“vue怎么实现右键菜单栏”吧!vue实现右键菜单栏和原生js大同小异,都是需要明白两个点contextme...
    99+
    2023-06-29
  • Springboot怎么使用docker-compose实现动态配置
    今天小编给大家分享一下Springboot怎么使用docker-compose实现动态配置的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来...
    99+
    2023-07-05
  • CSS怎么实现控制二级菜单动态出现功能
    本篇内容主要讲解“CSS怎么实现控制二级菜单动态出现功能”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“CSS怎么实现控制二级菜单动态出现功能”吧!HTML代码:...
    99+
    2024-04-02
  • LayUI+Shiro实现动态菜单并记住菜单收展的示例
    目录一、Maven 依赖二、菜单相关的类1、主菜单2、子菜单三、Shiro 配置1、ShiroConfig2、自定义shiro密码校验四、控制类1、LoginController2、...
    99+
    2024-04-02
  • vue动态菜单、动态路由加载以及刷新踩坑实战
    目录需求:思路:教训:分享正文:总结需求: 从接口动态获取子菜单数据 动态加载 要求只有展开才加载子菜单数据 支持刷新,页面显示正常 思路: 一开始比较乱,思路很多。想了很多 首先...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作