package org.dromara.system.service.impl;
|
|
import cn.hutool.core.bean.BeanUtil;
|
import cn.hutool.core.collection.CollUtil;
|
import cn.hutool.core.convert.Convert;
|
import cn.hutool.core.lang.tree.Tree;
|
import cn.hutool.core.util.ObjectUtil;
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
|
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
|
import lombok.RequiredArgsConstructor;
|
import org.dromara.common.core.constant.CacheNames;
|
import org.dromara.common.core.constant.SystemConstants;
|
import org.dromara.common.core.domain.dto.DeptDTO;
|
import org.dromara.common.core.exception.ServiceException;
|
import org.dromara.common.core.service.DeptService;
|
import org.dromara.common.core.utils.*;
|
import org.dromara.common.mybatis.helper.DataBaseHelper;
|
import org.dromara.common.redis.utils.CacheUtils;
|
import org.dromara.common.satoken.utils.LoginHelper;
|
import org.dromara.system.domain.SysDept;
|
import org.dromara.system.domain.SysRole;
|
import org.dromara.system.domain.SysUser;
|
import org.dromara.system.domain.bo.SysDeptBo;
|
import org.dromara.system.domain.vo.SysDeptVo;
|
import org.dromara.system.mapper.SysDeptMapper;
|
import org.dromara.system.mapper.SysRoleMapper;
|
import org.dromara.system.mapper.SysUserMapper;
|
import org.dromara.system.service.ISysDeptService;
|
import org.springframework.cache.annotation.CacheEvict;
|
import org.springframework.cache.annotation.Cacheable;
|
import org.springframework.cache.annotation.Caching;
|
import org.springframework.stereotype.Service;
|
import org.springframework.transaction.annotation.Transactional;
|
|
import java.util.ArrayList;
|
import java.util.Arrays;
|
import java.util.List;
|
|
/**
|
* 部门管理 服务实现
|
*
|
* @author Lion Li
|
*/
|
@RequiredArgsConstructor
|
@Service
|
public class SysDeptServiceImpl implements ISysDeptService, DeptService {
|
|
private final SysDeptMapper baseMapper;
|
private final SysRoleMapper roleMapper;
|
private final SysUserMapper userMapper;
|
|
/**
|
* 查询部门管理数据
|
*
|
* @param dept 部门信息
|
* @return 部门信息集合
|
*/
|
@Override
|
public List<SysDeptVo> selectDeptList(SysDeptBo dept) {
|
LambdaQueryWrapper<SysDept> lqw = buildQueryWrapper(dept);
|
return baseMapper.selectDeptList(lqw);
|
}
|
|
/**
|
* 查询部门树结构信息
|
*
|
* @param bo 部门信息
|
* @return 部门树信息集合
|
*/
|
@Override
|
public List<Tree<Long>> selectDeptTreeList(SysDeptBo bo) {
|
LambdaQueryWrapper<SysDept> lqw = buildQueryWrapper(bo);
|
List<SysDeptVo> depts = baseMapper.selectDeptList(lqw);
|
return buildDeptTreeSelect(depts);
|
}
|
|
private LambdaQueryWrapper<SysDept> buildQueryWrapper(SysDeptBo bo) {
|
LambdaQueryWrapper<SysDept> lqw = Wrappers.lambdaQuery();
|
lqw.eq(SysDept::getDelFlag, SystemConstants.NORMAL);
|
lqw.eq(ObjectUtil.isNotNull(bo.getDeptId()), SysDept::getDeptId, bo.getDeptId());
|
lqw.eq(ObjectUtil.isNotNull(bo.getParentId()), SysDept::getParentId, bo.getParentId());
|
lqw.like(StringUtils.isNotBlank(bo.getDeptName()), SysDept::getDeptName, bo.getDeptName());
|
lqw.like(StringUtils.isNotBlank(bo.getDeptCategory()), SysDept::getDeptCategory, bo.getDeptCategory());
|
lqw.eq(StringUtils.isNotBlank(bo.getStatus()), SysDept::getStatus, bo.getStatus());
|
lqw.orderByAsc(SysDept::getAncestors);
|
lqw.orderByAsc(SysDept::getParentId);
|
lqw.orderByAsc(SysDept::getOrderNum);
|
lqw.orderByAsc(SysDept::getDeptId);
|
return lqw;
|
}
|
|
/**
|
* 构建前端所需要下拉树结构
|
*
|
* @param depts 部门列表
|
* @return 下拉树结构列表
|
*/
|
@Override
|
public List<Tree<Long>> buildDeptTreeSelect(List<SysDeptVo> depts) {
|
if (CollUtil.isEmpty(depts)) {
|
return CollUtil.newArrayList();
|
}
|
// 获取当前列表中每一个节点的parentId,然后在列表中查找是否有id与其parentId对应,若无对应,则表明此时节点列表中,该节点在当前列表中属于顶级节点
|
List<Tree<Long>> treeList = CollUtil.newArrayList();
|
for (SysDeptVo d : depts) {
|
Long parentId = d.getParentId();
|
SysDeptVo sysDeptVo = StreamUtils.findFirst(depts, it -> it.getDeptId().longValue() == parentId);
|
if (ObjectUtil.isNull(sysDeptVo)) {
|
List<Tree<Long>> trees = TreeBuildUtils.build(depts, parentId, (dept, tree) ->
|
tree.setId(dept.getDeptId())
|
.setParentId(dept.getParentId())
|
.setName(dept.getDeptName())
|
.setWeight(dept.getOrderNum())
|
.putExtra("disabled", SystemConstants.DISABLE.equals(dept.getStatus())));
|
Tree<Long> tree = StreamUtils.findFirst(trees, it -> it.getId().longValue() == d.getDeptId());
|
treeList.add(tree);
|
}
|
}
|
return treeList;
|
}
|
|
/**
|
* 根据角色ID查询部门树信息
|
*
|
* @param roleId 角色ID
|
* @return 选中部门列表
|
*/
|
@Override
|
public List<Long> selectDeptListByRoleId(Long roleId) {
|
SysRole role = roleMapper.selectById(roleId);
|
return baseMapper.selectDeptListByRoleId(roleId, role.getDeptCheckStrictly());
|
}
|
|
/**
|
* 根据部门ID查询信息
|
*
|
* @param deptId 部门ID
|
* @return 部门信息
|
*/
|
@Cacheable(cacheNames = CacheNames.SYS_DEPT, key = "#deptId")
|
@Override
|
public SysDeptVo selectDeptById(Long deptId) {
|
SysDeptVo dept = baseMapper.selectVoById(deptId);
|
if (ObjectUtil.isNull(dept)) {
|
return null;
|
}
|
SysDeptVo parentDept = baseMapper.selectVoOne(new LambdaQueryWrapper<SysDept>()
|
.select(SysDept::getDeptName).eq(SysDept::getDeptId, dept.getParentId()));
|
dept.setParentName(ObjectUtils.notNullGetter(parentDept, SysDeptVo::getDeptName));
|
return dept;
|
}
|
|
@Override
|
public List<SysDeptVo> selectDeptByIds(List<Long> deptIds) {
|
return baseMapper.selectDeptList(new LambdaQueryWrapper<SysDept>()
|
.select(SysDept::getDeptId, SysDept::getDeptName, SysDept::getLeader)
|
.eq(SysDept::getStatus, SystemConstants.NORMAL)
|
.in(CollUtil.isNotEmpty(deptIds), SysDept::getDeptId, deptIds));
|
}
|
|
/**
|
* 通过部门ID查询部门名称
|
*
|
* @param deptIds 部门ID串逗号分隔
|
* @return 部门名称串逗号分隔
|
*/
|
@Override
|
public String selectDeptNameByIds(String deptIds) {
|
List<String> list = new ArrayList<>();
|
for (Long id : StringUtils.splitTo(deptIds, Convert::toLong)) {
|
SysDeptVo vo = SpringUtils.getAopProxy(this).selectDeptById(id);
|
if (ObjectUtil.isNotNull(vo)) {
|
list.add(vo.getDeptName());
|
}
|
}
|
return String.join(StringUtils.SEPARATOR, list);
|
}
|
|
/**
|
* 根据部门ID查询部门负责人
|
*
|
* @param deptId 部门ID,用于指定需要查询的部门
|
* @return 返回该部门的负责人ID
|
*/
|
@Override
|
public Long selectDeptLeaderById(Long deptId) {
|
SysDeptVo vo = SpringUtils.getAopProxy(this).selectDeptById(deptId);
|
return vo.getLeader();
|
}
|
|
/**
|
* 查询部门
|
*
|
* @return 部门列表
|
*/
|
@Override
|
public List<DeptDTO> selectDeptsByList() {
|
List<SysDeptVo> list = baseMapper.selectDeptList(new LambdaQueryWrapper<SysDept>()
|
.select(SysDept::getDeptId, SysDept::getDeptName, SysDept::getParentId)
|
.eq(SysDept::getStatus, SystemConstants.NORMAL));
|
return BeanUtil.copyToList(list, DeptDTO.class);
|
}
|
|
/**
|
* 根据ID查询所有子部门数(正常状态)
|
*
|
* @param deptId 部门ID
|
* @return 子部门数
|
*/
|
@Override
|
public long selectNormalChildrenDeptById(Long deptId) {
|
return baseMapper.selectCount(new LambdaQueryWrapper<SysDept>()
|
.eq(SysDept::getStatus, SystemConstants.NORMAL)
|
.apply(DataBaseHelper.findInSet(deptId, "ancestors")));
|
}
|
|
/**
|
* 是否存在子节点
|
*
|
* @param deptId 部门ID
|
* @return 结果
|
*/
|
@Override
|
public boolean hasChildByDeptId(Long deptId) {
|
return baseMapper.exists(new LambdaQueryWrapper<SysDept>()
|
.eq(SysDept::getParentId, deptId));
|
}
|
|
/**
|
* 查询部门是否存在用户
|
*
|
* @param deptId 部门ID
|
* @return 结果 true 存在 false 不存在
|
*/
|
@Override
|
public boolean checkDeptExistUser(Long deptId) {
|
return userMapper.exists(new LambdaQueryWrapper<SysUser>()
|
.eq(SysUser::getDeptId, deptId));
|
}
|
|
/**
|
* 校验部门名称是否唯一
|
*
|
* @param dept 部门信息
|
* @return 结果
|
*/
|
@Override
|
public boolean checkDeptNameUnique(SysDeptBo dept) {
|
boolean exist = baseMapper.exists(new LambdaQueryWrapper<SysDept>()
|
.eq(SysDept::getDeptName, dept.getDeptName())
|
.eq(SysDept::getParentId, dept.getParentId())
|
.ne(ObjectUtil.isNotNull(dept.getDeptId()), SysDept::getDeptId, dept.getDeptId()));
|
return !exist;
|
}
|
|
/**
|
* 校验部门是否有数据权限
|
*
|
* @param deptId 部门id
|
*/
|
@Override
|
public void checkDeptDataScope(Long deptId) {
|
if (ObjectUtil.isNull(deptId)) {
|
return;
|
}
|
if (LoginHelper.isSuperAdmin()) {
|
return;
|
}
|
if (baseMapper.countDeptById(deptId) == 0) {
|
throw new ServiceException("没有权限访问部门数据!");
|
}
|
}
|
|
/**
|
* 新增保存部门信息
|
*
|
* @param bo 部门信息
|
* @return 结果
|
*/
|
@CacheEvict(cacheNames = CacheNames.SYS_DEPT_AND_CHILD, allEntries = true)
|
@Override
|
public int insertDept(SysDeptBo bo) {
|
SysDept info = baseMapper.selectById(bo.getParentId());
|
// 如果父节点不为正常状态,则不允许新增子节点
|
if (!SystemConstants.NORMAL.equals(info.getStatus())) {
|
throw new ServiceException("部门停用,不允许新增");
|
}
|
SysDept dept = MapstructUtils.convert(bo, SysDept.class);
|
dept.setAncestors(info.getAncestors() + StringUtils.SEPARATOR + dept.getParentId());
|
return baseMapper.insert(dept);
|
}
|
|
/**
|
* 修改保存部门信息
|
*
|
* @param bo 部门信息
|
* @return 结果
|
*/
|
@Caching(evict = {
|
@CacheEvict(cacheNames = CacheNames.SYS_DEPT, key = "#bo.deptId"),
|
@CacheEvict(cacheNames = CacheNames.SYS_DEPT_AND_CHILD, allEntries = true)
|
})
|
@Override
|
@Transactional(rollbackFor = Exception.class)
|
public int updateDept(SysDeptBo bo) {
|
SysDept dept = MapstructUtils.convert(bo, SysDept.class);
|
SysDept oldDept = baseMapper.selectById(dept.getDeptId());
|
if (ObjectUtil.isNull(oldDept)) {
|
throw new ServiceException("部门不存在,无法修改");
|
}
|
if (!oldDept.getParentId().equals(dept.getParentId())) {
|
// 如果是新父部门 则校验是否具有新父部门权限 避免越权
|
this.checkDeptDataScope(dept.getParentId());
|
SysDept newParentDept = baseMapper.selectById(dept.getParentId());
|
if (ObjectUtil.isNotNull(newParentDept)) {
|
String newAncestors = newParentDept.getAncestors() + StringUtils.SEPARATOR + newParentDept.getDeptId();
|
String oldAncestors = oldDept.getAncestors();
|
dept.setAncestors(newAncestors);
|
updateDeptChildren(dept.getDeptId(), newAncestors, oldAncestors);
|
}
|
} else {
|
dept.setAncestors(oldDept.getAncestors());
|
}
|
int result = baseMapper.updateById(dept);
|
if (SystemConstants.NORMAL.equals(dept.getStatus()) && StringUtils.isNotEmpty(dept.getAncestors())
|
&& !StringUtils.equals(SystemConstants.NORMAL, dept.getAncestors())) {
|
// 如果该部门是启用状态,则启用该部门的所有上级部门
|
updateParentDeptStatusNormal(dept);
|
}
|
return result;
|
}
|
|
/**
|
* 修改该部门的父级部门状态
|
*
|
* @param dept 当前部门
|
*/
|
private void updateParentDeptStatusNormal(SysDept dept) {
|
String ancestors = dept.getAncestors();
|
Long[] deptIds = Convert.toLongArray(ancestors);
|
baseMapper.update(null, new LambdaUpdateWrapper<SysDept>()
|
.set(SysDept::getStatus, SystemConstants.NORMAL)
|
.in(SysDept::getDeptId, Arrays.asList(deptIds)));
|
}
|
|
/**
|
* 修改子元素关系
|
*
|
* @param deptId 被修改的部门ID
|
* @param newAncestors 新的父ID集合
|
* @param oldAncestors 旧的父ID集合
|
*/
|
private void updateDeptChildren(Long deptId, String newAncestors, String oldAncestors) {
|
List<SysDept> children = baseMapper.selectList(new LambdaQueryWrapper<SysDept>()
|
.apply(DataBaseHelper.findInSet(deptId, "ancestors")));
|
List<SysDept> list = new ArrayList<>();
|
for (SysDept child : children) {
|
SysDept dept = new SysDept();
|
dept.setDeptId(child.getDeptId());
|
dept.setAncestors(child.getAncestors().replaceFirst(oldAncestors, newAncestors));
|
list.add(dept);
|
}
|
if (CollUtil.isNotEmpty(list)) {
|
if (baseMapper.updateBatchById(list)) {
|
list.forEach(dept -> CacheUtils.evict(CacheNames.SYS_DEPT, dept.getDeptId()));
|
}
|
}
|
}
|
|
/**
|
* 删除部门管理信息
|
*
|
* @param deptId 部门ID
|
* @return 结果
|
*/
|
@Caching(evict = {
|
@CacheEvict(cacheNames = CacheNames.SYS_DEPT, key = "#deptId"),
|
@CacheEvict(cacheNames = CacheNames.SYS_DEPT_AND_CHILD, key = "#deptId")
|
})
|
@Override
|
public int deleteDeptById(Long deptId) {
|
return baseMapper.deleteById(deptId);
|
}
|
|
}
|