【功能修改】工作流:调整 cc 抄送表的字段

This commit is contained in:
YunaiV 2024-10-19 18:23:40 +08:00
parent 1c78cdc26e
commit 8a8544b3bd
9 changed files with 70 additions and 61 deletions

View File

@ -3,6 +3,7 @@ package cn.iocoder.yudao.framework.common.util.collection;
import cn.hutool.core.collection.CollUtil; import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.collection.CollectionUtil; import cn.hutool.core.collection.CollectionUtil;
import cn.hutool.core.util.ArrayUtil; import cn.hutool.core.util.ArrayUtil;
import cn.iocoder.yudao.framework.common.pojo.PageResult;
import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableMap;
import java.util.*; import java.util.*;
@ -73,6 +74,13 @@ public class CollectionUtils {
return from.stream().filter(filter).map(func).filter(Objects::nonNull).collect(Collectors.toList()); return from.stream().filter(filter).map(func).filter(Objects::nonNull).collect(Collectors.toList());
} }
public static <T, U> PageResult<U> convertPage(PageResult<T> from, Function<T, U> func) {
if (ArrayUtil.isEmpty(from)) {
return new PageResult<>(from.getTotal());
}
return new PageResult<>(convertList(from.getList(), func), from.getTotal());
}
public static <T, U> List<U> convertListByFlatMap(Collection<T> from, public static <T, U> List<U> convertListByFlatMap(Collection<T> from,
Function<T, ? extends Stream<? extends U>> func) { Function<T, ? extends Stream<? extends U>> func) {
if (CollUtil.isEmpty(from)) { if (CollUtil.isEmpty(from)) {

View File

@ -6,6 +6,7 @@ import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.framework.common.util.collection.MapUtils; import cn.iocoder.yudao.framework.common.util.collection.MapUtils;
import cn.iocoder.yudao.framework.common.util.date.DateUtils; import cn.iocoder.yudao.framework.common.util.date.DateUtils;
import cn.iocoder.yudao.framework.common.util.object.BeanUtils; import cn.iocoder.yudao.framework.common.util.object.BeanUtils;
import cn.iocoder.yudao.module.bpm.controller.admin.base.user.UserSimpleBaseVO;
import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.cc.BpmProcessInstanceCopyRespVO; import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.cc.BpmProcessInstanceCopyRespVO;
import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.instance.BpmProcessInstanceCopyPageReqVO; import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.instance.BpmProcessInstanceCopyPageReqVO;
import cn.iocoder.yudao.module.bpm.dal.dataobject.task.BpmProcessInstanceCopyDO; import cn.iocoder.yudao.module.bpm.dal.dataobject.task.BpmProcessInstanceCopyDO;
@ -28,8 +29,7 @@ import java.util.Map;
import java.util.stream.Stream; import java.util.stream.Stream;
import static cn.iocoder.yudao.framework.common.pojo.CommonResult.success; import static cn.iocoder.yudao.framework.common.pojo.CommonResult.success;
import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertListByFlatMap; import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.*;
import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertSet;
import static cn.iocoder.yudao.framework.security.core.util.SecurityFrameworkUtils.getLoginUserId; import static cn.iocoder.yudao.framework.security.core.util.SecurityFrameworkUtils.getLoginUserId;
@Tag(name = "管理后台 - 流程实例抄送") @Tag(name = "管理后台 - 流程实例抄送")
@ -62,11 +62,15 @@ public class BpmProcessInstanceCopyController {
convertSet(pageResult.getList(), BpmProcessInstanceCopyDO::getProcessInstanceId)); convertSet(pageResult.getList(), BpmProcessInstanceCopyDO::getProcessInstanceId));
Map<Long, AdminUserRespDTO> userMap = adminUserApi.getUserMap(convertListByFlatMap(pageResult.getList(), Map<Long, AdminUserRespDTO> userMap = adminUserApi.getUserMap(convertListByFlatMap(pageResult.getList(),
copy -> Stream.of(copy.getStartUserId(), Long.parseLong(copy.getCreator())))); copy -> Stream.of(copy.getStartUserId(), Long.parseLong(copy.getCreator()))));
return success(BeanUtils.toBean(pageResult, BpmProcessInstanceCopyRespVO.class, copyVO -> { return success(convertPage(pageResult, copy -> {
MapUtils.findAndThen(userMap, Long.valueOf(copyVO.getCreator()), user -> copyVO.setCreatorName(user.getNickname())); BpmProcessInstanceCopyRespVO copyVO = BeanUtils.toBean(copy, BpmProcessInstanceCopyRespVO.class);
MapUtils.findAndThen(userMap, copyVO.getStartUserId(), user -> copyVO.setStartUserName(user.getNickname())); MapUtils.findAndThen(userMap, Long.valueOf(copy.getCreator()),
user -> copyVO.setStartUser(BeanUtils.toBean(user, UserSimpleBaseVO.class)));
MapUtils.findAndThen(userMap, copy.getStartUserId(),
user -> copyVO.setCreateUser(BeanUtils.toBean(user, UserSimpleBaseVO.class)));
MapUtils.findAndThen(processInstanceMap, copyVO.getProcessInstanceId(), MapUtils.findAndThen(processInstanceMap, copyVO.getProcessInstanceId(),
processInstance -> copyVO.setProcessInstanceStartTime(DateUtils.of(processInstance.getStartTime()))); processInstance -> copyVO.setProcessInstanceStartTime(DateUtils.of(processInstance.getStartTime())));
return copyVO;
})); }));
} }

View File

@ -1,5 +1,6 @@
package cn.iocoder.yudao.module.bpm.controller.admin.task.vo.cc; package cn.iocoder.yudao.module.bpm.controller.admin.task.vo.cc;
import cn.iocoder.yudao.module.bpm.controller.admin.base.user.UserSimpleBaseVO;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data; import lombok.Data;
@ -12,10 +13,8 @@ public class BpmProcessInstanceCopyRespVO {
@Schema(description = "抄送主键", requiredMode = Schema.RequiredMode.REQUIRED, example = "1024") @Schema(description = "抄送主键", requiredMode = Schema.RequiredMode.REQUIRED, example = "1024")
private Long id; private Long id;
@Schema(description = "发起人编号", requiredMode = Schema.RequiredMode.REQUIRED, example = "888") @Schema(description = "发起人", requiredMode = Schema.RequiredMode.REQUIRED)
private Long startUserId; private UserSimpleBaseVO startUser;
@Schema(description = "发起人昵称", requiredMode = Schema.RequiredMode.REQUIRED, example = "芋道")
private String startUserName;
@Schema(description = "流程实例编号", requiredMode = Schema.RequiredMode.REQUIRED, example = "A233") @Schema(description = "流程实例编号", requiredMode = Schema.RequiredMode.REQUIRED, example = "A233")
private String processInstanceId; private String processInstanceId;
@ -24,20 +23,20 @@ public class BpmProcessInstanceCopyRespVO {
@Schema(description = "流程实例的发起时间", requiredMode = Schema.RequiredMode.REQUIRED) @Schema(description = "流程实例的发起时间", requiredMode = Schema.RequiredMode.REQUIRED)
private LocalDateTime processInstanceStartTime; private LocalDateTime processInstanceStartTime;
@Schema(description = "抄送的节点的活动编号") @Schema(description = "流程活动的编号", requiredMode = Schema.RequiredMode.REQUIRED)
private String activityId; private String activityId;
@Schema(description = "发起抄送的任务编号") @Schema(description = "流程活动的名字", requiredMode = Schema.RequiredMode.REQUIRED)
private String taskId; private String activityName;
@Schema(description = "发起抄送的任务名称")
private String taskName; @Schema(description = "流程活动的编号")
private String taskId;
@Schema(description = "抄送人", requiredMode = Schema.RequiredMode.REQUIRED, example = "1")
private String creator;
@Schema(description = "抄送人昵称")
private String creatorName;
@Schema(description = "抄送人意见") @Schema(description = "抄送人意见")
private String reason; private String reason;
@Schema(description = "创建人", requiredMode = Schema.RequiredMode.REQUIRED)
private UserSimpleBaseVO createUser;
@Schema(description = "抄送时间", requiredMode = Schema.RequiredMode.REQUIRED) @Schema(description = "抄送时间", requiredMode = Schema.RequiredMode.REQUIRED)
private LocalDateTime createTime; private LocalDateTime createTime;

View File

@ -4,7 +4,6 @@ import io.swagger.v3.oas.annotations.media.Schema;
import jakarta.validation.constraints.NotEmpty; import jakarta.validation.constraints.NotEmpty;
import lombok.Data; import lombok.Data;
import java.util.Collection;
import java.util.Map; import java.util.Map;
@Schema(description = "管理后台 - 通过流程任务的 Request VO") @Schema(description = "管理后台 - 通过流程任务的 Request VO")
@ -19,9 +18,6 @@ public class BpmTaskApproveReqVO {
@NotEmpty(message = "审批意见不能为空") @NotEmpty(message = "审批意见不能为空")
private String reason; private String reason;
@Schema(description = "抄送的用户编号数组", example = "1,2")
private Collection<Long> copyUserIds;
@Schema(description = "变量实例(动态表单)", requiredMode = Schema.RequiredMode.REQUIRED) @Schema(description = "变量实例(动态表单)", requiredMode = Schema.RequiredMode.REQUIRED)
private Map<String, Object> variables; private Map<String, Object> variables;

View File

@ -4,6 +4,8 @@ import cn.iocoder.yudao.framework.mybatis.core.dataobject.BaseDO;
import com.baomidou.mybatisplus.annotation.TableId; import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName; import com.baomidou.mybatisplus.annotation.TableName;
import lombok.*; import lombok.*;
import org.flowable.bpmn.model.FlowNode;
import org.flowable.task.api.history.HistoricTaskInstance;
/** /**
* 流程抄送 DO * 流程抄送 DO
@ -49,23 +51,25 @@ public class BpmProcessInstanceCopyDO extends BaseDO {
*/ */
private String category; private String category;
/** /**
* 流程活动编号 * 流程活动编号
* <p/> * <p/>
* 对应 BPMN XML 节点编号用于查询抄送节点的表单字段权限 *
* 这里冗余的原因如果是钉钉易搭的抄送节点 (ServiceTask)使用 taskId 可能查不到对应的 activityId * 冗余 {@link FlowNode#getId()}对应 BPMN XML 节点编号
* 原因用于查询抄送节点的表单字段权限因为仿钉钉/飞书的抄送节点 (ServiceTask)没有 taskId只有 activityId
*/ */
private String activityId; private String activityId;
/** /**
* 任务主键 * 流程活动的名字
* 关联 Task id 属性 *
* 冗余 {@link FlowNode#getName()}
*/
private String activityName;
/**
* 流程活动的编号
*
* 关联 {@link HistoricTaskInstance#getId()}
*/ */
private String taskId; private String taskId;
/**
* 任务名称
*
* 冗余 Task name 属性
*/
private String taskName;
/** /**
* 用户编号被抄送的用户编号 * 用户编号被抄送的用户编号

View File

@ -16,7 +16,7 @@ import static cn.iocoder.yudao.module.bpm.framework.flowable.core.listener.BpmCo
/** /**
* 处理抄送用户的 {@link JavaDelegate} 的实现类 * 处理抄送用户的 {@link JavaDelegate} 的实现类
* <p> * <p>
* 目前只有快搭模式的抄送节点使用 * 目前只有仿钉钉/飞书模式的抄送节点使用
* *
* @author jason * @author jason
*/ */
@ -41,7 +41,7 @@ public class BpmCopyTaskDelegate implements JavaDelegate {
// 2. 执行抄送 // 2. 执行抄送
FlowElement currentFlowElement = execution.getCurrentFlowElement(); FlowElement currentFlowElement = execution.getCurrentFlowElement();
processInstanceCopyService.createProcessInstanceCopy(userIds, null, execution.getProcessInstanceId(), processInstanceCopyService.createProcessInstanceCopy(userIds, null, execution.getProcessInstanceId(),
currentFlowElement.getId(), null, currentFlowElement.getName()); currentFlowElement.getId(), currentFlowElement.getName(), null);
} }
} }

View File

@ -3,9 +3,10 @@ package cn.iocoder.yudao.module.bpm.service.task;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.instance.BpmProcessInstanceCopyPageReqVO; import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.instance.BpmProcessInstanceCopyPageReqVO;
import cn.iocoder.yudao.module.bpm.dal.dataobject.task.BpmProcessInstanceCopyDO; import cn.iocoder.yudao.module.bpm.dal.dataobject.task.BpmProcessInstanceCopyDO;
import jakarta.validation.constraints.NotEmpty;
import org.flowable.bpmn.model.FlowNode;
import java.util.Collection; import java.util.Collection;
import java.util.Set;
/** /**
* 流程抄送 Service 接口 * 流程抄送 Service 接口
@ -15,7 +16,7 @@ import java.util.Set;
public interface BpmProcessInstanceCopyService { public interface BpmProcessInstanceCopyService {
/** /**
* 流程实例的抄送 * 管理员流程实例的抄送
* *
* @param userIds 抄送的用户编号 * @param userIds 抄送的用户编号
* @param reason 抄送意见 * @param reason 抄送意见
@ -24,17 +25,20 @@ public interface BpmProcessInstanceCopyService {
void createProcessInstanceCopy(Collection<Long> userIds, String reason, String taskId); void createProcessInstanceCopy(Collection<Long> userIds, String reason, String taskId);
/** /**
* 流程实例的抄送 * 自动抄送流程实例的抄送
* *
* @param userIds 抄送的用户编号 * @param userIds 抄送的用户编号
* @param reason 抄送意见 * @param reason 抄送意见
* @param processInstanceId 流程编号 * @param processInstanceId 流程编号
* @param activityId 流程活动编号 id (对应 BPMN XML 节点 Id) * @param activityId 流程活动编号对应 {@link FlowNode#getId()}
* @param taskId 任务编号 * @param activityName 任务编号对应 {@link FlowNode#getName()}
* @param taskName 任务名称 * @param taskId 任务编号允许空
*/ */
void createProcessInstanceCopy(Collection<Long> userIds, String reason, String processInstanceId, String activityId, void createProcessInstanceCopy(Collection<Long> userIds, String reason,
String taskId, String taskName); @NotEmpty(message = "流程实例编号不能为空") String processInstanceId,
@NotEmpty(message = "流程活动编号不能为空") String activityId,
@NotEmpty(message = "流程活动名字不能为空") String activityName,
String taskId);
/** /**
* 获得抄送的流程的分页 * 获得抄送的流程的分页

View File

@ -2,7 +2,6 @@ package cn.iocoder.yudao.module.bpm.service.task;
import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.ObjectUtil;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.framework.common.util.collection.CollectionUtils;
import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.instance.BpmProcessInstanceCopyPageReqVO; import cn.iocoder.yudao.module.bpm.controller.admin.task.vo.instance.BpmProcessInstanceCopyPageReqVO;
import cn.iocoder.yudao.module.bpm.dal.dataobject.task.BpmProcessInstanceCopyDO; import cn.iocoder.yudao.module.bpm.dal.dataobject.task.BpmProcessInstanceCopyDO;
import cn.iocoder.yudao.module.bpm.dal.mysql.task.BpmProcessInstanceCopyMapper; import cn.iocoder.yudao.module.bpm.dal.mysql.task.BpmProcessInstanceCopyMapper;
@ -19,7 +18,6 @@ import org.springframework.validation.annotation.Validated;
import java.util.Collection; import java.util.Collection;
import java.util.List; import java.util.List;
import java.util.Set;
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception; import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertList; import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertList;
@ -54,13 +52,14 @@ public class BpmProcessInstanceCopyServiceImpl implements BpmProcessInstanceCopy
if (ObjectUtil.isNull(task)) { if (ObjectUtil.isNull(task)) {
throw exception(ErrorCodeConstants.TASK_NOT_EXISTS); throw exception(ErrorCodeConstants.TASK_NOT_EXISTS);
} }
String processInstanceId = task.getProcessInstanceId(); // 执行抄送
createProcessInstanceCopy(userIds, reason, processInstanceId, task.getTaskDefinitionKey(), task.getId(), task.getName()); createProcessInstanceCopy(userIds, reason,
task.getProcessInstanceId(), task.getTaskDefinitionKey(), task.getId(), task.getName());
} }
@Override @Override
public void createProcessInstanceCopy(Collection<Long> userIds, String reason, String processInstanceId, String activityId, public void createProcessInstanceCopy(Collection<Long> userIds, String reason, String processInstanceId,
String taskId, String taskName) { String activityId, String activityName, String taskId) {
// 1.1 校验流程实例存在 // 1.1 校验流程实例存在
ProcessInstance processInstance = processInstanceService.getProcessInstance(processInstanceId); ProcessInstance processInstance = processInstanceService.getProcessInstance(processInstanceId);
if (processInstance == null) { if (processInstance == null) {
@ -77,8 +76,8 @@ public class BpmProcessInstanceCopyServiceImpl implements BpmProcessInstanceCopy
List<BpmProcessInstanceCopyDO> copyList = convertList(userIds, userId -> new BpmProcessInstanceCopyDO() List<BpmProcessInstanceCopyDO> copyList = convertList(userIds, userId -> new BpmProcessInstanceCopyDO()
.setUserId(userId).setReason(reason).setStartUserId(Long.valueOf(processInstance.getStartUserId())) .setUserId(userId).setReason(reason).setStartUserId(Long.valueOf(processInstance.getStartUserId()))
.setProcessInstanceId(processInstanceId).setProcessInstanceName(processInstance.getName()) .setProcessInstanceId(processInstanceId).setProcessInstanceName(processInstance.getName())
.setCategory(processDefinition.getCategory()).setActivityId(activityId) .setCategory(processDefinition.getCategory()).setTaskId(taskId)
.setTaskId(taskId).setTaskName(taskName)); .setActivityId(activityId).setActivityName(activityName));
processInstanceCopyMapper.insertBatch(copyList); processInstanceCopyMapper.insertBatch(copyList);
} }

View File

@ -426,11 +426,6 @@ public class BpmTaskServiceImpl implements BpmTaskService {
throw exception(PROCESS_INSTANCE_NOT_EXISTS); throw exception(PROCESS_INSTANCE_NOT_EXISTS);
} }
// 2. 抄送用户
if (CollUtil.isNotEmpty(reqVO.getCopyUserIds())) {
processInstanceCopyService.createProcessInstanceCopy(reqVO.getCopyUserIds(), null, reqVO.getId());
}
// 情况一被委派的任务不调用 complete 去完成任务 // 情况一被委派的任务不调用 complete 去完成任务
if (DelegationState.PENDING.equals(task.getDelegationState())) { if (DelegationState.PENDING.equals(task.getDelegationState())) {
approveDelegateTask(reqVO, task); approveDelegateTask(reqVO, task);
@ -444,12 +439,12 @@ public class BpmTaskServiceImpl implements BpmTaskService {
} }
// 情况三审批普通的任务大多数情况下都是这样 // 情况三审批普通的任务大多数情况下都是这样
// 3.1 更新 task 状态原因 // 2.1 更新 task 状态原因
updateTaskStatusAndReason(task.getId(), BpmTaskStatusEnum.APPROVE.getStatus(), reqVO.getReason()); updateTaskStatusAndReason(task.getId(), BpmTaskStatusEnum.APPROVE.getStatus(), reqVO.getReason());
// 3.2 添加评论 // 2.2 添加评论
taskService.addComment(task.getId(), task.getProcessInstanceId(), BpmCommentTypeEnum.APPROVE.getType(), taskService.addComment(task.getId(), task.getProcessInstanceId(), BpmCommentTypeEnum.APPROVE.getType(),
BpmCommentTypeEnum.APPROVE.formatComment(reqVO.getReason())); BpmCommentTypeEnum.APPROVE.formatComment(reqVO.getReason()));
// 3.3 调用 BPM complete 去完成任务 // 2.3 调用 BPM complete 去完成任务
// 其中variables 是存储动态表单到 local 任务级别过滤一下避免 ProcessInstance 系统级的变量被占用 // 其中variables 是存储动态表单到 local 任务级别过滤一下避免 ProcessInstance 系统级的变量被占用
if (CollUtil.isNotEmpty(reqVO.getVariables())) { if (CollUtil.isNotEmpty(reqVO.getVariables())) {
Map<String, Object> variables = FlowableUtils.filterTaskFormVariable(reqVO.getVariables()); Map<String, Object> variables = FlowableUtils.filterTaskFormVariable(reqVO.getVariables());