crm-客户管理:完善客户、公海客户分页查询

This commit is contained in:
puhui999 2023-11-23 10:12:50 +08:00
parent b41d469d46
commit 0d821d705d
6 changed files with 121 additions and 78 deletions

View File

@ -8,18 +8,18 @@ import lombok.Getter;
import java.util.Arrays; import java.util.Arrays;
/** /**
* CRM 客户等级 * CRM 列表检索场景
* *
* @author Wanwan * @author HUIHUI
*/ */
@Getter @Getter
@AllArgsConstructor @AllArgsConstructor
public enum CrmCustomerSceneEnum implements IntArrayValuable { public enum CrmSceneEnum implements IntArrayValuable {
OWNER(1, "我负责的客户"), OWNER(1, "我负责的"),
FOLLOW(2, "我关注的客户"); FOLLOW(2, "我关注的");
public static final int[] ARRAYS = Arrays.stream(values()).mapToInt(CrmCustomerSceneEnum::getType).toArray(); public static final int[] ARRAYS = Arrays.stream(values()).mapToInt(CrmSceneEnum::getType).toArray();
/** /**
* 场景类型 * 场景类型

View File

@ -1,5 +1,6 @@
package cn.iocoder.yudao.module.crm.controller.admin.customer; package cn.iocoder.yudao.module.crm.controller.admin.customer;
import cn.hutool.core.collection.CollUtil;
import cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants; import cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants;
import cn.iocoder.yudao.framework.common.pojo.CommonResult; import cn.iocoder.yudao.framework.common.pojo.CommonResult;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
@ -9,7 +10,6 @@ import cn.iocoder.yudao.module.crm.controller.admin.customer.vo.*;
import cn.iocoder.yudao.module.crm.convert.customer.CrmCustomerConvert; import cn.iocoder.yudao.module.crm.convert.customer.CrmCustomerConvert;
import cn.iocoder.yudao.module.crm.dal.dataobject.customer.CrmCustomerDO; import cn.iocoder.yudao.module.crm.dal.dataobject.customer.CrmCustomerDO;
import cn.iocoder.yudao.module.crm.service.customer.CrmCustomerService; import cn.iocoder.yudao.module.crm.service.customer.CrmCustomerService;
import cn.iocoder.yudao.module.crm.service.permission.CrmPermissionService;
import cn.iocoder.yudao.module.system.api.dept.DeptApi; import cn.iocoder.yudao.module.system.api.dept.DeptApi;
import cn.iocoder.yudao.module.system.api.dept.dto.DeptRespDTO; import cn.iocoder.yudao.module.system.api.dept.dto.DeptRespDTO;
import cn.iocoder.yudao.module.system.api.user.AdminUserApi; import cn.iocoder.yudao.module.system.api.user.AdminUserApi;
@ -51,8 +51,6 @@ public class CrmCustomerController {
private DeptApi deptApi; private DeptApi deptApi;
@Resource @Resource
private AdminUserApi adminUserApi; private AdminUserApi adminUserApi;
@Resource
private CrmPermissionService permissionService;
@PostMapping("/create") @PostMapping("/create")
@Operation(summary = "创建客户") @Operation(summary = "创建客户")
@ -119,20 +117,14 @@ public class CrmCustomerController {
return success(true); return success(true);
} }
// TODO @puhui999可以在 CrmCustomerPageReqVO 里面加个 pool 参数 true 代表来自公海客户的分页
@GetMapping("/page") @GetMapping("/page")
@Operation(summary = "获得客户分页") @Operation(summary = "获得客户分页")
@PreAuthorize("@ss.hasPermission('crm:customer:query')") @PreAuthorize("@ss.hasPermission('crm:customer:query')")
public CommonResult<PageResult<CrmCustomerRespVO>> getCustomerPage(@Valid CrmCustomerPageReqVO pageVO) { public CommonResult<PageResult<CrmCustomerRespVO>> getCustomerPage(@Valid CrmCustomerPageReqVO pageVO) {
//PageResult<CrmCustomerDO> pageResult = customerService.getCustomerPage(pageVO, getLoginUserId()); PageResult<CrmCustomerDO> pageResult = customerService.getCustomerPage(pageVO, getLoginUserId());
//if (CollUtil.isEmpty(pageResult.getList())) { if (CollUtil.isEmpty(pageResult.getList())) {
// return success(PageResult.empty(pageResult.getTotal())); return success(PageResult.empty(pageResult.getTotal()));
//}
// 拼接数据
return convertPage(customerService.getCustomerPage(pageVO, getLoginUserId()));
} }
private CommonResult<PageResult<CrmCustomerRespVO>> convertPage(PageResult<CrmCustomerDO> pageResult) {
// 1.1 获取负责人详情 // 1.1 获取负责人详情
Set<Long> userIds = convertSet(pageResult.getList(), CrmCustomerDO::getOwnerUserId); Set<Long> userIds = convertSet(pageResult.getList(), CrmCustomerDO::getOwnerUserId);
userIds.addAll(convertSet(pageResult.getList(), item -> Long.parseLong(item.getCreator()))); // 加入创建者 userIds.addAll(convertSet(pageResult.getList(), item -> Long.parseLong(item.getCreator()))); // 加入创建者
@ -172,19 +164,19 @@ public class CrmCustomerController {
return success(true); return success(true);
} }
@PutMapping("/receive") //@PutMapping("/receive")
@Operation(summary = "领取公海客户") //@Operation(summary = "领取公海客户")
// TODO @xiaqing1receiveCustomer 方法名字2cIds 改成 ids要加下 @RequestParam还有 swagger 注解3参数非空使用 validator 校验4返回 true 即可 //// TODO @xiaqing1receiveCustomer 方法名字2cIds 改成 ids要加下 @RequestParam还有 swagger 注解3参数非空使用 validator 校验4返回 true 即可
@PreAuthorize("@ss.hasPermission('crm:customer:receive')") //@PreAuthorize("@ss.hasPermission('crm:customer:receive')")
public CommonResult<String> receiveByIds(List<Long> cIds) { //public CommonResult<String> receiveByIds(List<Long> cIds) {
// 判断是否为空 // // 判断是否为空
if (CollectionUtils.isEmpty(cIds)) // if (CollectionUtils.isEmpty(cIds))
return error(GlobalErrorCodeConstants.BAD_REQUEST.getCode(), GlobalErrorCodeConstants.BAD_REQUEST.getMsg()); // return error(GlobalErrorCodeConstants.BAD_REQUEST.getCode(), GlobalErrorCodeConstants.BAD_REQUEST.getMsg());
// 领取公海任务 // // 领取公海任务
// TODO @xiaqinguserid通过 controller 传递给 service不要在 service 里面获取无状态 // // TODO @xiaqinguserid通过 controller 传递给 service不要在 service 里面获取无状态
customerService.receive(cIds); // customerService.receive(cIds);
return success("领取成功"); // return success("领取成功");
} //}
// TODO @xiaqing1distributeCustomer 方法名2cIds 同上3参数校验同上4ownerId 改成 ownerUserId和别的模块统一5返回 true 即可 // TODO @xiaqing1distributeCustomer 方法名2cIds 同上3参数校验同上4ownerId 改成 ownerUserId和别的模块统一5返回 true 即可
@PutMapping("/distributeByIds") @PutMapping("/distributeByIds")

View File

@ -1,12 +1,14 @@
package cn.iocoder.yudao.module.crm.controller.admin.customer.vo; package cn.iocoder.yudao.module.crm.controller.admin.customer.vo;
import cn.iocoder.yudao.framework.common.pojo.PageParam; import cn.iocoder.yudao.framework.common.pojo.PageParam;
import cn.iocoder.yudao.module.crm.enums.common.CrmCustomerSceneEnum; import cn.iocoder.yudao.module.crm.enums.common.CrmSceneEnum;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data; import lombok.Data;
import lombok.EqualsAndHashCode; import lombok.EqualsAndHashCode;
import lombok.ToString; import lombok.ToString;
import javax.validation.constraints.NotNull;
@Schema(description = "管理后台 - CRM 客户分页 Request VO") @Schema(description = "管理后台 - CRM 客户分页 Request VO")
@Data @Data
@EqualsAndHashCode(callSuper = true) @EqualsAndHashCode(callSuper = true)
@ -31,9 +33,13 @@ public class CrmCustomerPageReqVO extends PageParam {
/** /**
* 场景类型 * 场景类型
* *
* 关联 {@link CrmCustomerSceneEnum} * 关联 {@link CrmSceneEnum}
*/ */
@Schema(description = "场景类型", example = "1") @Schema(description = "场景类型", example = "1")
private Integer sceneType; private Integer sceneType;
@Schema(description = "是否为公海数据", requiredMode = Schema.RequiredMode.REQUIRED, example = "false")
@NotNull(message = "是否为公海数据不能为空")
private Boolean pool;
} }

View File

@ -7,13 +7,12 @@ import cn.iocoder.yudao.framework.mybatis.core.query.MPJLambdaWrapperX;
import cn.iocoder.yudao.framework.mybatis.core.util.MyBatisUtils; import cn.iocoder.yudao.framework.mybatis.core.util.MyBatisUtils;
import cn.iocoder.yudao.module.crm.controller.admin.customer.vo.CrmCustomerPageReqVO; import cn.iocoder.yudao.module.crm.controller.admin.customer.vo.CrmCustomerPageReqVO;
import cn.iocoder.yudao.module.crm.dal.dataobject.customer.CrmCustomerDO; import cn.iocoder.yudao.module.crm.dal.dataobject.customer.CrmCustomerDO;
import cn.iocoder.yudao.module.crm.dal.dataobject.permission.CrmPermissionDO;
import cn.iocoder.yudao.module.crm.enums.common.CrmSceneEnum;
import cn.iocoder.yudao.module.crm.framework.enums.CrmBizTypeEnum; import cn.iocoder.yudao.module.crm.framework.enums.CrmBizTypeEnum;
import cn.iocoder.yudao.module.crm.util.CrmPermissionUtils;
import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.metadata.IPage;
import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Mapper;
import java.util.Collection;
/** /**
* 客户 Mapper * 客户 Mapper
* *
@ -22,9 +21,30 @@ import java.util.Collection;
@Mapper @Mapper
public interface CrmCustomerMapper extends BaseMapperX<CrmCustomerDO> { public interface CrmCustomerMapper extends BaseMapperX<CrmCustomerDO> {
default PageResult<CrmCustomerDO> selectPage(CrmCustomerPageReqVO pageReqVO, Collection<Long> ids) { default PageResult<CrmCustomerDO> selectPage(CrmCustomerPageReqVO pageReqVO) {
return selectPage(pageReqVO, new LambdaQueryWrapperX<CrmCustomerDO>() LambdaQueryWrapperX<CrmCustomerDO> queryWrapperX = new LambdaQueryWrapperX<>();
.inIfPresent(CrmCustomerDO::getId, ids) if (pageReqVO.getPool()) { // 情况一公海
queryWrapperX.isNull(CrmCustomerDO::getOwnerUserId);
} else {// 情况一不是公海
queryWrapperX.isNotNull(CrmCustomerDO::getOwnerUserId);
}
return selectPage(pageReqVO, queryWrapperX
.likeIfPresent(CrmCustomerDO::getName, pageReqVO.getName())
.eqIfPresent(CrmCustomerDO::getMobile, pageReqVO.getMobile())
.eqIfPresent(CrmCustomerDO::getIndustryId, pageReqVO.getIndustryId())
.eqIfPresent(CrmCustomerDO::getLevel, pageReqVO.getLevel())
.eqIfPresent(CrmCustomerDO::getSource, pageReqVO.getSource()));
}
default PageResult<CrmCustomerDO> selectPage1(CrmCustomerPageReqVO pageReqVO, Long userId) {
LambdaQueryWrapperX<CrmCustomerDO> queryWrapperX = new LambdaQueryWrapperX<>();
//queryWrapperX.sql
if (pageReqVO.getPool()) { // 情况一公海
queryWrapperX.isNull(CrmCustomerDO::getOwnerUserId);
} else {// 情况一不是公海
queryWrapperX.isNotNull(CrmCustomerDO::getOwnerUserId);
}
return selectPage(pageReqVO, queryWrapperX
.likeIfPresent(CrmCustomerDO::getName, pageReqVO.getName()) .likeIfPresent(CrmCustomerDO::getName, pageReqVO.getName())
.eqIfPresent(CrmCustomerDO::getMobile, pageReqVO.getMobile()) .eqIfPresent(CrmCustomerDO::getMobile, pageReqVO.getMobile())
.eqIfPresent(CrmCustomerDO::getIndustryId, pageReqVO.getIndustryId()) .eqIfPresent(CrmCustomerDO::getIndustryId, pageReqVO.getIndustryId())
@ -33,11 +53,50 @@ public interface CrmCustomerMapper extends BaseMapperX<CrmCustomerDO> {
} }
default PageResult<CrmCustomerDO> selectPage(CrmCustomerPageReqVO pageReqVO, Long userId) { default PageResult<CrmCustomerDO> selectPage(CrmCustomerPageReqVO pageReqVO, Long userId) {
// MyBatis Plus 查询
IPage<CrmCustomerDO> mpPage = MyBatisUtils.buildPage(pageReqVO); IPage<CrmCustomerDO> mpPage = MyBatisUtils.buildPage(pageReqVO);
MPJLambdaWrapperX<CrmCustomerDO> mpjLambdaWrapperX = new MPJLambdaWrapperX<>(); MPJLambdaWrapperX<CrmCustomerDO> mpjLambdaWrapperX = new MPJLambdaWrapperX<>();
// 构建数据权限连表条件 // 构建数据权限连表条件
CrmPermissionUtils.builderLeftJoinQuery(mpjLambdaWrapperX, CrmBizTypeEnum.CRM_CUSTOMER.getType(), userId); //CrmPermissionUtils.builderRightJoinQuery(mpjLambdaWrapperX, CrmBizTypeEnum.CRM_CUSTOMER.getType(), userId);
mpjLambdaWrapperX
//.rightJoin("(SELECT t1.biz_id FROM crm_permission t1 WHERE (t1.biz_type = 1 AND t1.user_id = 1)) t2 on t.id = t2.biz_id");
.rightJoin(CrmPermissionDO.class, CrmPermissionDO::getBizId, CrmCustomerDO::getId)
.eq(CrmPermissionDO::getBizType, CrmBizTypeEnum.CRM_CUSTOMER.getType())
.eq(CrmPermissionDO::getUserId, userId);
/** TODO @芋艿
-- 常规连表-查询正常
| ==> Preparing:
SELECT t.id, t.name, t.follow_up_status, t.lock_status, t.deal_status,
t.industry_id, t.level, t.source, t.mobile, t.telephone, t.website,
t.qq, t.wechat, t.email, t.description, t.remark, t.owner_user_id,
t.area_id, t.detail_address, t.contact_last_time, t.contact_next_time,
t.create_time, t.update_time, t.creator, t.updater, t.deleted
FROM crm_customer t RIGHT JOIN crm_permission t1 ON (t1.biz_id = t.id) AND t.tenant_id = 1
WHERE t.deleted = 0 AND t1.deleted = 0
AND (t1.biz_type = ? AND t1.user_id = ?
AND t.owner_user_id IS NOT NULL AND t.level = ?) AND t1.tenant_id = 1 LIMIT ?
| ==> Parameters: 2(Integer), 1(Long), 3(Integer), 10(Long)
-- 连接子查询-报错但是复制到 navicat 是可以正常执行的
-- 区别点常规连表会自动拼接租户 AND t.tenant_id = 1
SELECT
t.id,t.name,t.follow_up_status,t.lock_status,t.deal_status,t.industry_id,t.level,
t.source,t.mobile,t.telephone,t.website,t.qq,t.wechat,t.email,t.description,t.remark,
t.owner_user_id,t.area_id,t.detail_address,t.contact_last_time,t.contact_next_time,
t.create_time,t.update_time,t.creator,t.updater,t.deleted
FROM crm_customer t
RIGHT JOIN (SELECT t1.biz_id FROM crm_permission t1 WHERE (t1.biz_type = 2 AND t1.user_id = 1)) t2 on t.id = t2.biz_id
WHERE t.deleted=0
AND (t.owner_user_id IS NOT NULL)
*/
if (pageReqVO.getPool()) { // 情况一公海
mpjLambdaWrapperX.isNull(CrmCustomerDO::getOwnerUserId);
} else {// 情况一不是公海
mpjLambdaWrapperX.isNotNull(CrmCustomerDO::getOwnerUserId);
}
// TODO 场景数据过滤
if (CrmSceneEnum.isOwner(pageReqVO.getSceneType())) { // 场景一我负责的数据
mpjLambdaWrapperX.eq(CrmCustomerDO::getOwnerUserId, userId);
}
mpPage = selectJoinPage(mpPage, CrmCustomerDO.class, mpjLambdaWrapperX mpPage = selectJoinPage(mpPage, CrmCustomerDO.class, mpjLambdaWrapperX
.selectAll(CrmCustomerDO.class) .selectAll(CrmCustomerDO.class)
.likeIfPresent(CrmCustomerDO::getName, pageReqVO.getName()) .likeIfPresent(CrmCustomerDO::getName, pageReqVO.getName())

View File

@ -39,6 +39,7 @@ public class CrmCustomerServiceImpl implements CrmCustomerService {
private CrmPermissionService crmPermissionService; private CrmPermissionService crmPermissionService;
@Override @Override
@Transactional(rollbackFor = Exception.class)
public Long createCustomer(CrmCustomerCreateReqVO createReqVO, Long userId) { public Long createCustomer(CrmCustomerCreateReqVO createReqVO, Long userId) {
// 插入 // 插入
CrmCustomerDO customer = CrmCustomerConvert.INSTANCE.convert(createReqVO); CrmCustomerDO customer = CrmCustomerConvert.INSTANCE.convert(createReqVO);
@ -89,32 +90,19 @@ public class CrmCustomerServiceImpl implements CrmCustomerService {
@Override @Override
public PageResult<CrmCustomerDO> getCustomerPage(CrmCustomerPageReqVO pageReqVO, Long userId) { public PageResult<CrmCustomerDO> getCustomerPage(CrmCustomerPageReqVO pageReqVO, Long userId) {
//// 1.1 TODO 如果是超级管理员 // 1.1. TODO 如果是超级管理员
//boolean admin = false; boolean admin = false;
//if (admin && ObjUtil.notEqual(userId, CrmPermissionDO.POOL_USER_ID)) { if (admin) {
// return customerMapper.selectPage(pageReqVO, Collections.emptyList()); return customerMapper.selectPage(pageReqVO);
//} }
//// 1.2 获取当前用户能看的分页数据 // 1.2. 获取当前用户能看的分页数据
//// TODO @puhui999如果业务的数据量比较大in 太多可能有性能问题噢看看是不是搞成 join 连表了可以微信讨论下
//List<CrmPermissionDO> permissions = crmPermissionService.getPermissionListByBizTypeAndUserId(
// CrmBizTypeEnum.CRM_CUSTOMER.getType(), userId);
//// 1.3 TODO 场景数据过滤
//if (CrmCustomerSceneEnum.isOwner(pageReqVO.getSceneType())) { // 场景一我负责的数据
// permissions = CollectionUtils.filterList(permissions, item -> CrmPermissionLevelEnum.isOwner(item.getLevel()));
//}
//Set<Long> ids = convertSet(permissions, CrmPermissionDO::getBizId);
//if (CollUtil.isEmpty(ids)) { // 没得说明没有什么给他看的
// return PageResult.empty();
//}
//
//// 2. 获取客户分页数据
//return customerMapper.selectPage(pageReqVO, ids);
return customerMapper.selectPage(pageReqVO, userId); return customerMapper.selectPage(pageReqVO, userId);
} }
@Override @Override
public List<CrmCustomerDO> getCustomerList(CrmCustomerExportReqVO exportReqVO) { public List<CrmCustomerDO> getCustomerList(CrmCustomerExportReqVO exportReqVO) {
//return customerMapper.selectList(exportReqVO); //return customerMapper.selectList(exportReqVO);
// TODO puhui999: 等数据权限完善后再实现
return Collections.emptyList(); return Collections.emptyList();
} }
@ -191,6 +179,7 @@ public class CrmCustomerServiceImpl implements CrmCustomerService {
} }
@Override @Override
@Transactional(rollbackFor = Exception.class)
@CrmPermission(bizType = CrmBizTypeEnum.CRM_CUSTOMER, bizId = "#id", level = CrmPermissionLevelEnum.OWNER) @CrmPermission(bizType = CrmBizTypeEnum.CRM_CUSTOMER, bizId = "#id", level = CrmPermissionLevelEnum.OWNER)
public void putPool(Long id) { public void putPool(Long id) {
// 1. 校验存在 // 1. 校验存在

View File

@ -17,13 +17,10 @@ public class CrmPermissionUtils {
* @param bizTyp 模块类型 * @param bizTyp 模块类型
* @param userId 用户 * @param userId 用户
*/ */
public static void builderLeftJoinQuery(MPJLambdaWrapper<?> mpjLambdaWrapper, Integer bizTyp, Long userId) { public static void builderRightJoinQuery(MPJLambdaWrapper<?> mpjLambdaWrapper, Integer bizTyp, Long userId) {
String querySql = "(SELECT t1.biz_id FROM crm_permission t1 WHERE (t1.biz_type = {} AND t1.user_id = {})) t2 on t.id = t2.biz_id";
// 默认主表别名是 t // 默认主表别名是 t
mpjLambdaWrapper.leftJoin(StrUtil.format("(" + mpjLambdaWrapper.rightJoin(StrUtil.format(querySql, bizTyp, userId));
"select p.biz_id from crm_permission p" +
" where p.biz_type = {} and p.user_id = {}" +
") t2" +
" on t.id = t2.biz_id", bizTyp, userId));
} }
} }