springboot + vue 实现递归生成多级菜单(实例代码)

开发过程中,涉及到多级菜单的应用,找了一些资料案例实现如下(使用springboot+layui+oracle):

创建菜单表 :

-- 创建菜单表
create table wxmini_menus(
 menu_id         number unique,   --菜单ID
 menu_name        varchar2(20),
 menu_url        varchar2(200),
 menu_icon        varchar2(100),
 parent_id        number,      --父菜单ID
 status         varchar2(10),
 menu_sort        number,
 last_update_date    DATE not null,
 last_updated_by     NUMBER not null,
 creation_date      DATE not null,
 created_by       NUMBER not null,
 last_update_login    NUMBER
);

-- 插入菜单记录
insert into wxmini_menus values((select 1 from dual),'设置',null,null,null,1,100,sysdate,-1,sysdate,-1,-1);
insert into wxmini_menus values((select max(menu_id)+1 from yl_wxmini_menus),'用户',null,null,1,1,null,sysdate,-1,sysdate,-1,-1);
insert into wxmini_menus values((select max(menu_id)+1 from yl_wxmini_menus),'网站用户',null,null,1,1,null,sysdate,-1,sysdate,-1,-1);
insert into wxmini_menus values((select max(menu_id)+1 from yl_wxmini_menus),'角色管理',null,null,1,1,null,sysdate,-1,sysdate,-1,-1);
--主菜单二
insert into wxmini_menus values((select max(menu_id)+1 from yl_wxmini_menus),'主页',null,null,null,1,1,sysdate,-1,sysdate,-1,-1);
insert into wxmini_menus values((select max(menu_id)+1 from yl_wxmini_menus),'控制台',null,null,5,1,null,sysdate,-1,sysdate,-1,-1);
insert into wxmini_menus values((select max(menu_id)+1 from yl_wxmini_menus),'主页一',null,null,5,1,null,sysdate,-1,sysdate,-1,-1);
insert into wxmini_menus values((select max(menu_id)+1 from yl_wxmini_menus),'主页二',null,null,5,1,null,sysdate,-1,sysdate,-1,-1);

实体类 WxMenu.java:

public class WxMenu {

  private String menu_id;
  private String menu_name;
  private String menu_url;
  private String menu_icon;
  private String parent_id;
  private String parent_menu_name;
  private String status;
  private String menu_sort;
  private String last_update_date;
  private long last_updated_by;
  private String creation_date;
  private long created_by;
  private long last_update_login;

  private List<WxMenu> childMenus;

  ......
}
Controller控制层 :
@Controller
@RequestMapping("/index")
public class IndexController {

  @Autowired
  private WxUserService wxUserService;

  /**
   * 获取所有菜单数据
   * @return
   */
  @ResponseBody
  @RequestMapping(value = "/loadAuthMenus",method = RequestMethod.GET)
  public LayuiJsonFormat<HashMap> login(String username, String password, HttpServletRequest request) {

    // 获取所有菜单数据
    List<WxMenu> wxMenus = wxUserService.getAllMenus();
    //定义 存储一级菜单
    List<WxMenu> menuList = new ArrayList<WxMenu>();
    // 先找到所有的一级菜单
    for (int i = 0; i < wxMenus.size(); i++) {
      // 一级菜单没有父菜单,为null
      if (StringUtils.isBlank(wxMenus.get(i).getParent_id())) {
        menuList.add(wxMenus.get(i));
      }
    }
    // for调用递归,循环获取所有子菜单
    for (WxMenu menu : menuList) {
      menu.setChildMenus(getChildmenus(menu.getMenu_id(), wxMenus));
    }
    //存储所有菜单,将数据回传至 layui 前端
    map.put("menus", menuList);

    return LayuiJsonFormat.createBySuccess(map.size(),map);
  }

  /**
   * 递归子菜单
   * @param id
   * @param wxMenus
   * @return
   */
  private List<WxMenu> getChildmenus(String id, List<WxMenu> wxMenus) {
    // 子菜单
    List<WxMenu> childMenus = new ArrayList<>();
    for (WxMenu menu : wxMenus) {
      if (StringUtils.isNotBlank(menu.getParent_id())) {
        if (menu.getParent_id().equals(id)) {
          childMenus.add(menu);
        }
      }
    }
    for (WxMenu menu : childMenus) {
      //数据库动态存储菜单的url,此时判断url为空的,则是节点菜单(存在子菜单)
      if (StringUtils.isBlank(menu.getMenu_url())) {
        // 递归调用
        menu.setChildMenus(getChildmenus(menu.getMenu_id(), wxMenus));
      }
    }
    if (childMenus.size() == 0) {
      return null;
    }
    return childMenus;
  }
}

service类 WxUserService.java

public interface WxUserService {

  //获取所有菜单
  List<WxMenu> getAllMenus();
}
service实现类:
@Service("wxUserService")
public class WxUserServiceImpl implements WxUserService {

  @Autowired
  private WxUserMapper wxUserMapper;

  @Override
  public List<WxMenu> getAllMenus() {
    return wxUserMapper.getAllMenus();
  }
}
mapper:
public interface WxUserMapper {
  List<WxMenu> getAllMenus();
}

mybatis sql :

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.yinlong.wxmini.mapper.WxUserMapper">

  <!-- 获取所有菜单 -->
  <select id="getAllMenus" resultType="com.yinlong.wxmini.entity.WxMenu">
    select * from wxmini_menus wm order by wm.menu_sort asc
  </select>
</mapper>
最后,前端样式使用 layui ,html + js 如下:
<ul id="app-4" class="layui-nav layui-nav-tree" lay-shrink="all" lay-filter="layadmin-system-side-menu">
 <li data-name="home" class="layui-nav-item layui-nav-itemed">
  <div v-for="menu in menus" >
   <a href="javascript:;" lay-tips="主页" lay-direction="2">
    <i class="layui-icon layui-icon-home"></i>
    <cite>{{ menu.menu_name }}</cite>
   </a>
   <dl v-for="child in menu.childMenus" style="padding: 0px !important;" class="layui-nav-child">
    <dd data-name="console" >
     <a class="v-href-menu" lay-href="javascript:void(0)" >{{ child.menu_name }}</a>
    </dd>
   </dl>
  </div>
 </li>
</ul>
<script type="text/javascript" th:src="@{/layuiadmin/layui/layui.js}"></script>
<script type="text/javascript" th:src="@{/layuiadmin/vue.js}" ></script>
<script type="text/javascript" th:src="@{/layuiadmin/layui/jquery-3.2.1.min.js}" ></script>

<script>
 $(document).ready(function(){
  $.get("/index/loadAuthMenus", function(data, status){
   var vm = new Vue({
    el: '#app-4',
    data: {
     menus: data.data.menus
    }
   })
  });
 });

</script>

最后效果图如下:

总结

以上所述是小编给大家介绍的springboot + vue 实现递归生成多级菜单,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

(0)

相关推荐

  • 详解VUE Element-UI多级菜单动态渲染的组件

    以下是组件代码: <template> <div class="navMenu"> <label v-for="navMenu in navMenus"> <el-menu-item v-if="navMenu.childs==null&&navMenu.entity&&navMenu.entity.state==='ENABLE'" :key="navMenu.

  • vue+springboot实现项目的CORS跨域请求

    跨域资源共享CORS(Cross-origin Resource Sharing),是W3C的一个标准,允许浏览器向跨源的服务器发起XMLHttpRequest请求,克服ajax请求只能同源使用的限制.关于CORS的详细解读,可参考阮一峰大神的博客:跨域资源共享CORS详解.本文为通过一个小demo对该博客中分析内容的一些验证. 1.springboot+vue项目的构建和启动 细节不在此赘述,任何简单的springboot项目就可以,而前端vue项目只需用axios发ajax请求即可. 我的d

  • 解决前后端分离 vue+springboot 跨域 session+cookie失效问题

    环境: 前端 vue ip地址:192.168.1.205 后端 springboot2.0 ip地址:192.168.1.217 主要开发后端. 问题: 首先登陆成功时将用户存在session中,后续请求在将用户从session中取出检查.后续请求取出的用户都为null. 解决过程: 首先发现sessionID不一致,导致每一次都是新的会话,当然不可能存在用户了.然后发现cookie浏览器不能自动保存,服务器响应set-cookie了 搜索问题,发现跨域,服务器响应的setCookie浏览器无

  • spring boot+vue 的前后端分离与合并方案实例详解

    springboot和vue结合的方案网络上的主要有以下两种: 1. [不推荐]在html中直接使用script标签引入vue和一些常用的组件,这种方式和以前传统的开发是一样的,只是可以很爽的使用vue的双向数据绑定,这种方式只适合于普通的全栈开发. 2.[推荐]使用vue官方的脚手架创建单独的前端工程项目,做到和后端完全独立开发和部署,后端单独部署一个纯restful的服务,而前端直接采用nginx来部署,这种称为完全的前后端分离架构开发模式,但是在分离中有很多api权限的问题需要解决,包括部

  • vue+springboot前后端分离实现单点登录跨域问题解决方法

    最近在做一个后台管理系统,前端是用时下火热的vue.js,后台是基于springboot的.因为后台系统没有登录功能,但是公司要求统一登录,登录认证统一使用.net项目组的认证系统.那就意味着做单点登录咯,至于不知道什么是单点登录的同学,建议去找一下万能的度娘. 刚接到这个需求的时候,老夫心里便不屑的认为:区区登录何足挂齿,但是,开发的过程狠狠的打了我一巴掌(火辣辣的一巴掌)...,所以这次必须得好好记录一下这次教训,以免以后再踩这样的坑. 我面临的第一个问题是跨域,浏览器控制台直接报CORS,

  • springboot + vue 实现递归生成多级菜单(实例代码)

    开发过程中,涉及到多级菜单的应用,找了一些资料案例实现如下(使用springboot+layui+oracle): 创建菜单表 : -- 创建菜单表 create table wxmini_menus( menu_id number unique, --菜单ID menu_name varchar2(20), menu_url varchar2(200), menu_icon varchar2(100), parent_id number, --父菜单ID status varchar2(10)

  • vue配置文件自动生成路由和菜单实例代码

    目录 写在前面 router.json 路由生成 菜单生成 效果 总结 写在前面 每次重复写路由的时候是不是会觉得很烦,特别是项目大的时候,路由会有特别多,看都看不过来,所以这里我是有了一个router.json的配置文件来对路由做一些简单的配置,然后让路由和左侧菜单栏可以同时生成. router.json 主要配置项如下: { "name": "routerConfig", "menu": [{ "id": "1&

  • vue+element-ui动态生成多级表头的方法

    vue+element html配置: <div id="table">{{tableData}} <el-table :data="tabledata01" :span-method="tableSpanMethod" max-height="420"> <el-table-column v-for='item in tableConfig' :label="item.label&qu

  • vue侧边栏动态生成下级菜单的方法

    循环传入的数据去生成下级菜单 <template> <div class="headBar"> <div class="title"> 微商城管理后台 </div> <el-menu class="el-menu-headBar" mode="horizontal" @select="handleSelect" background-color=&quo

  • JAVA递归生成树形菜单的实现过程

    递归生成一个如图的菜单,编写两个类数据模型Menu.和创建树形的MenuTree.通过以下过程实现: 1.首先从菜单数据中获取所有根节点. 2.为根节点建立次级子树并拼接上. 3.递归为子节点建立次级子树并接上,直至为末端节点拼接上空的“树”. 首先,编写数据模型Menu.每条菜单有自己的id.父节点parentId.菜单名称text.菜单还拥有次级菜单children. import java.util.List; public class Menu { private String id;

  • vue 实现的树形菜的实例代码

    下面一段代码给大家介绍vue 实现的树形菜单功能,具体代码如下所示: <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>vue</title> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <l

  • 原生JavaScript实现动态省市县三级联动下拉框菜单实例代码

    像平时购物选择地址时一样,通过选择的省动态加载城市列表,通过选择的城市动态加载县区列表,从而可以实现省市县的三级联动,下面使用原生的JavaScript来实现这个功能: 先给大家展示下测试结果: 未做任何选择时: 选择时: 代码如下所示: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title>三级联动测试</titl

  • vue的全局提示框组件实例代码

    这篇文章给大家介绍一个vue全局提示框组件,具体代码如下所示: <template> <!-- 全局提示框 --> <div v-show="visible" class="dialog-tips dialog-center"> <div>{{message}}</div> </div> </template> <script> export default { data

  • Vue实现侧边菜单栏手风琴效果实例代码

    效果图如下所示: <template> <div class="asideBox"> <aside> <ul class="asideMenu"> <li v-for="(item,index) in menuList"> <div class="oneMenu" @click="showToggle(item,index)"> <

  • vue实现随机验证码功能的实例代码

    1.html代码 <div class="form-group" style="display: flex;"> <div> <span>验证码:</span> <input type="text" id="code" v-model="code" class="code" placeholder="请输入您的验证码&quo

随机推荐