/*
|
* Copyright (c) 2018-2028, Chill Zhuang All rights reserved.
|
*
|
* Redistribution and use in source and binary forms, with or without
|
* modification, are permitted provided that the following conditions are met:
|
*
|
* Redistributions of source code must retain the above copyright notice,
|
* this list of conditions and the following disclaimer.
|
* Redistributions in binary form must reproduce the above copyright
|
* notice, this list of conditions and the following disclaimer in the
|
* documentation and/or other materials provided with the distribution.
|
* Neither the name of the dreamlu.net developer nor the names of its
|
* contributors may be used to endorse or promote products derived from
|
* this software without specific prior written permission.
|
* Author: Chill 庄骞 (smallchill@163.com)
|
*/
|
package com.vci.ubcs.system.service.impl;
|
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
import com.baomidou.mybatisplus.core.metadata.IPage;
|
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
import com.vci.ubcs.system.cache.SysCache;
|
import com.vci.ubcs.system.dto.MenuDTO;
|
import com.vci.ubcs.system.entity.*;
|
import com.vci.ubcs.system.vo.MenuVO;
|
import com.vci.ubcs.system.mapper.MenuMapper;
|
import com.vci.ubcs.system.service.IMenuService;
|
import com.vci.ubcs.system.service.IRoleMenuService;
|
import com.vci.ubcs.system.service.IRoleScopeService;
|
import com.vci.ubcs.system.service.ITopMenuSettingService;
|
import com.vci.ubcs.system.wrapper.MenuWrapper;
|
import lombok.AllArgsConstructor;
|
import org.springblade.core.log.exception.ServiceException;
|
import org.springblade.core.mp.support.Condition;
|
import org.springblade.core.mp.support.Query;
|
import org.springblade.core.secure.BladeUser;
|
import org.springblade.core.secure.utils.AuthUtil;
|
import org.springblade.core.tool.api.R;
|
import org.springblade.core.tool.constant.BladeConstant;
|
import org.springblade.core.tool.node.ForestNodeMerger;
|
import org.springblade.core.tool.node.TreeNode;
|
import org.springblade.core.tool.support.Kv;
|
import org.springblade.core.tool.utils.Func;
|
import org.springblade.core.tool.utils.StringUtil;
|
import org.springframework.cache.annotation.Cacheable;
|
import org.springframework.stereotype.Service;
|
|
import java.util.*;
|
import java.util.stream.Collectors;
|
|
import static com.vci.ubcs.common.constant.CommonConstant.API_SCOPE_CATEGORY;
|
import static com.vci.ubcs.common.constant.CommonConstant.DATA_SCOPE_CATEGORY;
|
import static org.springblade.core.cache.constant.CacheConstant.MENU_CACHE;
|
|
/**
|
* 服务实现类
|
*
|
* @author Chill
|
*/
|
@Service
|
@AllArgsConstructor
|
public class MenuServiceImpl extends ServiceImpl<MenuMapper, Menu> implements IMenuService {
|
|
private final IRoleMenuService roleMenuService;
|
private final IRoleScopeService roleScopeService;
|
private final ITopMenuSettingService topMenuSettingService;
|
private final static String PARENT_ID = "parentId";
|
private final static Integer MENU_CATEGORY = 1;
|
|
@Override
|
public List<MenuVO> lazyList(Long parentId, Map<String, Object> param) {
|
if (Func.isEmpty(Func.toStr(param.get(PARENT_ID)))) {
|
parentId = null;
|
}
|
return baseMapper.lazyList(parentId, param);
|
}
|
|
@Override
|
public IPage<MenuVO> lazyMenuPage(Long parentId, Map<String, Object> param, Query query) {
|
if (Func.isEmpty(Func.toStr(param.get(PARENT_ID)))) {
|
parentId = null;
|
}
|
return baseMapper.lazyMenuPage(parentId, param, Condition.getPage(query));
|
}
|
|
|
@Override
|
public List<MenuVO> routes(String roleId, Long topMenuId) {
|
if (StringUtil.isBlank(roleId)) {
|
return null;
|
}
|
List<Menu> allMenus = baseMapper.allMenu();
|
List<Menu> roleMenus;
|
// 超级管理员并且不是顶部菜单请求则返回全部菜单
|
if (AuthUtil.isAdministrator() && Func.isEmpty(topMenuId)) {
|
roleMenus = allMenus;
|
}
|
// 非超级管理员并且不是顶部菜单请求则返回对应角色权限菜单
|
else if (!AuthUtil.isAdministrator() && Func.isEmpty(topMenuId)) {
|
roleMenus = tenantPackageMenu(baseMapper.roleMenuByRoleId(Func.toLongList(roleId)));
|
}
|
// 顶部菜单请求返回对应角色权限菜单
|
else {
|
// 角色配置对应菜单
|
List<Menu> roleIdMenus = baseMapper.roleMenuByRoleId(Func.toLongList(roleId));
|
// 反向递归角色菜单所有父级
|
List<Menu> routes = new LinkedList<>(roleIdMenus);
|
roleIdMenus.forEach(roleMenu -> recursion(allMenus, routes, roleMenu));
|
// 顶部配置对应菜单
|
List<Menu> topIdMenus = baseMapper.roleMenuByTopMenuId(topMenuId);
|
// 筛选匹配角色对应的权限菜单
|
roleMenus = topIdMenus.stream().filter(x ->
|
routes.stream().anyMatch(route -> route.getId().longValue() == x.getId().longValue())
|
).collect(Collectors.toList());
|
}
|
return buildRoutes(allMenus, roleMenus);
|
}
|
|
private List<MenuVO> buildRoutes(List<Menu> allMenus, List<Menu> roleMenus) {
|
List<Menu> routes = new LinkedList<>(roleMenus);
|
roleMenus.forEach(roleMenu -> recursion(allMenus, routes, roleMenu));
|
routes.sort(Comparator.comparing(Menu::getSort));
|
MenuWrapper menuWrapper = new MenuWrapper();
|
List<Menu> collect = routes.stream().filter(x -> Func.equals(x.getCategory(), 1)).collect(Collectors.toList());
|
return menuWrapper.listNodeVO(collect);
|
}
|
|
private void recursion(List<Menu> allMenus, List<Menu> routes, Menu roleMenu) {
|
Optional<Menu> menu = allMenus.stream().filter(x -> Func.equals(x.getId(), roleMenu.getParentId())).findFirst();
|
if (menu.isPresent() && !routes.contains(menu.get())) {
|
routes.add(menu.get());
|
recursion(allMenus, routes, menu.get());
|
}
|
}
|
|
@Override
|
public List<MenuVO> buttons(String roleId) {
|
List<Menu> buttons = (AuthUtil.isAdministrator()) ? baseMapper.allButtons() : baseMapper.buttons(Func.toLongList(roleId));
|
MenuWrapper menuWrapper = new MenuWrapper();
|
return menuWrapper.listNodeVO(buttons);
|
}
|
|
@Override
|
public List<TreeNode> tree() {
|
return ForestNodeMerger.merge(baseMapper.tree());
|
}
|
|
@Override
|
public List<TreeNode> grantTree(BladeUser user) {
|
List<TreeNode> menuTree = user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantTree() : baseMapper.grantTreeByRole(Func.toLongList(user.getRoleId()));
|
return ForestNodeMerger.merge(tenantPackageTree(menuTree, user.getTenantId()));
|
}
|
|
@Override
|
public List<TreeNode> grantTopTree(BladeUser user) {
|
List<TreeNode> menuTree = user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantTopTree() : baseMapper.grantTopTreeByRole(Func.toLongList(user.getRoleId()));
|
return ForestNodeMerger.merge(tenantPackageTree(menuTree, user.getTenantId()));
|
}
|
|
/**
|
* 租户菜单权限自定义筛选
|
*/
|
private List<TreeNode> tenantPackageTree(List<TreeNode> menuTree, String tenantId) {
|
TenantPackage tenantPackage = SysCache.getTenantPackage(tenantId);
|
if (!AuthUtil.isAdministrator() && Func.isNotEmpty(tenantPackage) && tenantPackage.getId() > 0L) {
|
List<Long> menuIds = Func.toLongList(tenantPackage.getMenuId());
|
// 筛选出两者菜单交集集合
|
List<TreeNode> collect = menuTree.stream().filter(x -> menuIds.contains(x.getId())).collect(Collectors.toList());
|
// 创建递归基础集合
|
List<TreeNode> packageTree = new LinkedList<>(collect);
|
// 递归筛选出菜单集合所有父级
|
collect.forEach(treeNode -> recursionParent(menuTree, packageTree, treeNode));
|
// 递归筛选出菜单集合所有子级
|
collect.forEach(treeNode -> recursionChild(menuTree, packageTree, treeNode));
|
// 合并在一起返回最终集合
|
return packageTree;
|
}
|
return menuTree;
|
}
|
|
/**
|
* 父节点递归
|
*/
|
public void recursionParent(List<TreeNode> menuTree, List<TreeNode> packageTree, TreeNode treeNode) {
|
Optional<TreeNode> node = menuTree.stream().filter(x -> Func.equals(x.getId(), treeNode.getParentId())).findFirst();
|
if (node.isPresent() && !packageTree.contains(node.get())) {
|
packageTree.add(node.get());
|
recursionParent(menuTree, packageTree, node.get());
|
}
|
}
|
|
/**
|
* 子节点递归
|
*/
|
public void recursionChild(List<TreeNode> menuTree, List<TreeNode> packageTree, TreeNode treeNode) {
|
List<TreeNode> nodes = menuTree.stream().filter(x -> Func.equals(x.getParentId(), treeNode.getId())).collect(Collectors.toList());
|
nodes.forEach(node -> {
|
if (!packageTree.contains(node)) {
|
packageTree.add(node);
|
recursionChild(menuTree, packageTree, node);
|
}
|
});
|
}
|
|
/**
|
* 租户菜单权限自定义筛选
|
*/
|
private List<Menu> tenantPackageMenu(List<Menu> menu) {
|
TenantPackage tenantPackage = SysCache.getTenantPackage(AuthUtil.getTenantId());
|
if (Func.isNotEmpty(tenantPackage) && tenantPackage.getId() > 0L) {
|
List<Long> menuIds = Func.toLongList(tenantPackage.getMenuId());
|
menu = menu.stream().filter(x -> menuIds.contains(x.getId())).collect(Collectors.toList());
|
}
|
return menu;
|
}
|
|
@Override
|
public List<TreeNode> grantDataScopeTree(BladeUser user) {
|
return ForestNodeMerger.merge(user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantDataScopeTree() : baseMapper.grantDataScopeTreeByRole(Func.toLongList(user.getRoleId())));
|
}
|
|
@Override
|
public List<TreeNode> grantApiScopeTree(BladeUser user) {
|
return ForestNodeMerger.merge(user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantApiScopeTree() : baseMapper.grantApiScopeTreeByRole(Func.toLongList(user.getRoleId())));
|
}
|
|
@Override
|
public List<String> roleTreeKeys(String roleIds) {
|
List<RoleMenu> roleMenus = roleMenuService.list(Wrappers.<RoleMenu>query().lambda().in(RoleMenu::getRoleId, Func.toLongList(roleIds)));
|
return roleMenus.stream().map(roleMenu -> Func.toStr(roleMenu.getMenuId())).collect(Collectors.toList());
|
}
|
|
@Override
|
public List<String> topTreeKeys(String topMenuIds) {
|
List<TopMenuSetting> settings = topMenuSettingService.list(Wrappers.<TopMenuSetting>query().lambda().in(TopMenuSetting::getTopMenuId, Func.toLongList(topMenuIds)));
|
return settings.stream().map(setting -> Func.toStr(setting.getMenuId())).collect(Collectors.toList());
|
}
|
|
@Override
|
public List<String> dataScopeTreeKeys(String roleIds) {
|
List<RoleScope> roleScopes = roleScopeService.list(Wrappers.<RoleScope>query().lambda().eq(RoleScope::getScopeCategory, DATA_SCOPE_CATEGORY).in(RoleScope::getRoleId, Func.toLongList(roleIds)));
|
return roleScopes.stream().map(roleScope -> Func.toStr(roleScope.getScopeId())).collect(Collectors.toList());
|
}
|
|
@Override
|
public List<String> apiScopeTreeKeys(String roleIds) {
|
List<RoleScope> roleScopes = roleScopeService.list(Wrappers.<RoleScope>query().lambda().eq(RoleScope::getScopeCategory, API_SCOPE_CATEGORY).in(RoleScope::getRoleId, Func.toLongList(roleIds)));
|
return roleScopes.stream().map(roleScope -> Func.toStr(roleScope.getScopeId())).collect(Collectors.toList());
|
}
|
|
@Override
|
@Cacheable(cacheNames = MENU_CACHE, key = "'auth:routes:' + #user.roleId")
|
public List<Kv> authRoutes(BladeUser user) {
|
List<MenuDTO> routes = baseMapper.authRoutes(Func.toLongList(user.getRoleId()));
|
List<Kv> list = new ArrayList<>();
|
routes.forEach(route -> list.add(Kv.create().set(route.getPath(), Kv.create().set("authority", Func.toStrArray(route.getAlias())))));
|
return list;
|
}
|
|
@Override
|
public boolean removeMenu(String ids) {
|
Long cnt = baseMapper.selectCount(Wrappers.<Menu>query().lambda().in(Menu::getParentId, Func.toLongList(ids)));
|
if (cnt > 0L) {
|
throw new ServiceException("请先删除子节点!");
|
}
|
return removeByIds(Func.toLongList(ids));
|
}
|
|
@Override
|
public boolean submit(Menu menu) {
|
LambdaQueryWrapper<Menu> menuQueryWrapper = Wrappers.lambdaQuery();
|
// 新增
|
if (menu.getId() == null) {
|
//检验是否重复菜单别名,只校验同一父分类菜单下的编号是否重复
|
menuQueryWrapper.eq(Menu::getParentId,menu.getParentId()!=null ? menu.getParentId():BladeConstant.TOP_PARENT_ID)
|
.eq(Menu::getCode, menu.getCode())
|
.or(wrapper -> wrapper.eq(Menu::getName, menu.getName()).eq(Menu::getCategory, MENU_CATEGORY))
|
;
|
} else {
|
// 修改
|
menuQueryWrapper.ne(Menu::getId, menu.getId()).and(
|
wrapper -> wrapper.eq(Menu::getCode, menu.getCode())
|
.eq(Menu::getParentId,menu.getParentId())
|
.or(o -> o.eq(Menu::getName, menu.getName()).eq(Menu::getCategory, MENU_CATEGORY))
|
);
|
}
|
Long cnt = baseMapper.selectCount(menuQueryWrapper);
|
if (cnt > 0L) {
|
throw new ServiceException("菜单名或编号已存在!");
|
}
|
if (menu.getParentId() == null && menu.getId() == null) {
|
menu.setParentId(BladeConstant.TOP_PARENT_ID);
|
}
|
if (menu.getParentId() != null && menu.getId() == null) {
|
Menu parentMenu = baseMapper.selectById(menu.getParentId());
|
if (parentMenu != null && parentMenu.getCategory() != 1) {
|
throw new ServiceException("父节点只可选择菜单类型!");
|
}
|
}
|
menu.setIsDeleted(BladeConstant.DB_NOT_DELETED);
|
return saveOrUpdate(menu);
|
}
|
|
@Override
|
public List<Menu> getMenuButtonByType(String btmType) {
|
return baseMapper.selectMenuChildByBtnType(btmType);
|
}
|
|
/**
|
* 对KeepAlive值转换成布尔类型进行封装
|
*
|
* @param childMenu
|
* @return
|
*/
|
@Override
|
public void handleKeepAlive(List<MenuVO> childMenu) {
|
childMenu.forEach(list->{
|
list.getMeta().put("keepAlive","true".equals(list.getKeepAlive()));
|
if(list.getChildren().size()>0){
|
handleKeepAlive(list.getChildren());
|
}
|
});
|
}
|
|
/**
|
* 克隆其他菜单下按钮
|
* @param menuId 要克隆的菜单按钮主键
|
* @param buttonIds 被克隆的按钮主键
|
* @return
|
*/
|
@Override
|
public R cloneMenuButton(Long menuId, List<String> buttonIds) {
|
if(Func.isEmpty(menuId)){
|
return R.fail("要克隆的菜单主键不能为空!");
|
}
|
if(buttonIds.isEmpty() || buttonIds.size() <= 0){
|
return R.fail("被克隆的按钮主键不能为空!");
|
}
|
// 先根据主键查询出所有按钮的信息
|
List<Menu> buttons = this.listByIds(buttonIds);
|
List<Menu> newButtons = new ArrayList<>();
|
List<String> addButtonCodes = new ArrayList();
|
buttons.parallelStream().forEach(item->{
|
// 判断是否为按钮,非按钮不处理
|
if(item.getCategory().equals(2)){
|
// 改变父节点信息
|
item.setParentId(menuId);
|
// 将主键赋空
|
item.setId(null);
|
addButtonCodes.add(item.getCode());
|
newButtons.add(item);
|
}
|
});
|
//检验是否重复菜单别名,只校验同一父分类菜单下的编号是否重复
|
LambdaQueryWrapper<Menu> menuQueryWrapper = Wrappers.<Menu>lambdaQuery()
|
.eq(Menu::getParentId,menuId)
|
.and(a -> a.in( Menu::getCode, addButtonCodes));
|
Long cnt = baseMapper.selectCount(menuQueryWrapper);
|
if (cnt > 0L) {
|
return R.fail("该菜单下已存在的编号与要克隆的按钮编号存在重复!");
|
}
|
return this.saveBatch(newButtons) ? R.success("按钮克隆成功!"):R.fail("按钮克隆失败!");
|
}
|
|
}
|