|
|
|
|
@ -0,0 +1,295 @@
|
|
|
|
|
package org.dromara.oa.erp.service.impl;
|
|
|
|
|
|
|
|
|
|
import cn.hutool.core.collection.CollUtil;
|
|
|
|
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
|
|
|
|
import lombok.RequiredArgsConstructor;
|
|
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
|
import org.apache.seata.spring.annotation.GlobalTransactional;
|
|
|
|
|
import org.dromara.common.core.enums.BusinessStatusEnum;
|
|
|
|
|
import org.dromara.common.core.enums.OAStatusEnum;
|
|
|
|
|
import org.dromara.common.core.exception.ServiceException;
|
|
|
|
|
import org.dromara.common.core.utils.MapstructUtils;
|
|
|
|
|
import org.dromara.common.core.utils.StringUtils;
|
|
|
|
|
import org.dromara.common.tenant.helper.TenantHelper;
|
|
|
|
|
import org.dromara.oa.erp.domain.ErpProjectInfo;
|
|
|
|
|
import org.dromara.oa.erp.domain.ErpProjectPlan;
|
|
|
|
|
import org.dromara.oa.erp.domain.ErpProjectPlanStage;
|
|
|
|
|
import org.dromara.oa.erp.domain.bo.ErpProjectInfoBo;
|
|
|
|
|
import org.dromara.oa.erp.domain.bo.ErpProjectPlanStageBo;
|
|
|
|
|
import org.dromara.oa.erp.domain.vo.ErpProjectInfoVo;
|
|
|
|
|
import org.dromara.oa.erp.mapper.ErpProjectInfoMapper;
|
|
|
|
|
import org.dromara.oa.erp.mapper.ErpProjectPlanMapper;
|
|
|
|
|
import org.dromara.oa.erp.mapper.ErpProjectPlanStageMapper;
|
|
|
|
|
import org.dromara.oa.erp.service.IErpContractOrderService;
|
|
|
|
|
import org.dromara.system.api.RemoteCodeRuleService;
|
|
|
|
|
import org.dromara.workflow.api.RemoteWorkflowService;
|
|
|
|
|
import org.dromara.workflow.api.domain.RemoteStartProcess;
|
|
|
|
|
import org.dromara.workflow.api.event.ProcessEvent;
|
|
|
|
|
import org.springframework.context.event.EventListener;
|
|
|
|
|
import org.springframework.stereotype.Service;
|
|
|
|
|
import org.apache.dubbo.config.annotation.DubboReference;
|
|
|
|
|
|
|
|
|
|
import java.util.List;
|
|
|
|
|
import java.util.Map;
|
|
|
|
|
import java.util.Objects;
|
|
|
|
|
import java.util.stream.Collectors;
|
|
|
|
|
import cn.hutool.core.map.MapUtil;
|
|
|
|
|
import cn.hutool.core.convert.Convert;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 合同订单Service业务层处理
|
|
|
|
|
*
|
|
|
|
|
* @author Yinq
|
|
|
|
|
* @date 2025-12-05
|
|
|
|
|
*/
|
|
|
|
|
@RequiredArgsConstructor
|
|
|
|
|
@Service
|
|
|
|
|
@Slf4j
|
|
|
|
|
public class ErpContractOrderServiceImpl implements IErpContractOrderService {
|
|
|
|
|
|
|
|
|
|
private final ErpProjectInfoMapper projectInfoMapper;
|
|
|
|
|
private final ErpProjectPlanMapper projectPlanMapper;
|
|
|
|
|
private final ErpProjectPlanStageMapper planStageMapper;
|
|
|
|
|
|
|
|
|
|
@DubboReference(timeout = 30000)
|
|
|
|
|
private RemoteWorkflowService remoteWorkflowService;
|
|
|
|
|
|
|
|
|
|
@DubboReference(timeout = 30000)
|
|
|
|
|
private RemoteCodeRuleService remoteCodeRuleService;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 查询合同订单(项目信息)
|
|
|
|
|
*
|
|
|
|
|
* @param projectId 主键
|
|
|
|
|
* @return 合同订单(项目信息)
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
public ErpProjectInfoVo queryById(Long projectId) {
|
|
|
|
|
ErpProjectInfo projectInfo = projectInfoMapper.selectById(projectId);
|
|
|
|
|
if (projectInfo == null) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
return MapstructUtils.convert(projectInfo, ErpProjectInfoVo.class);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 暂存合同订单(项目信息)
|
|
|
|
|
*
|
|
|
|
|
* @param bo 合同订单(项目信息)
|
|
|
|
|
* @return 合同订单(项目信息)
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
@GlobalTransactional(rollbackFor = Exception.class)
|
|
|
|
|
public ErpProjectInfoVo saveContractOrder(ErpProjectInfoBo bo) {
|
|
|
|
|
// 保存项目信息
|
|
|
|
|
ErpProjectInfo projectInfo = MapstructUtils.convert(bo, ErpProjectInfo.class);
|
|
|
|
|
validEntityBeforeSave(projectInfo);
|
|
|
|
|
|
|
|
|
|
if (StringUtils.isNull(bo.getProjectId())) {
|
|
|
|
|
// 新增
|
|
|
|
|
projectInfoMapper.insert(projectInfo);
|
|
|
|
|
bo.setProjectId(projectInfo.getProjectId());
|
|
|
|
|
} else {
|
|
|
|
|
// 更新
|
|
|
|
|
projectInfoMapper.updateById(projectInfo);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 保存项目阶段计划列表
|
|
|
|
|
savePlanStageList(projectInfo, bo.getPlanStageList());
|
|
|
|
|
|
|
|
|
|
return MapstructUtils.convert(projectInfo, ErpProjectInfoVo.class);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 修改合同订单(项目信息)
|
|
|
|
|
*
|
|
|
|
|
* @param bo 合同订单(项目信息)
|
|
|
|
|
* @return 合同订单(项目信息)
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
@GlobalTransactional(rollbackFor = Exception.class)
|
|
|
|
|
public ErpProjectInfoVo updateContractOrder(ErpProjectInfoBo bo) {
|
|
|
|
|
return saveContractOrder(bo);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 提交合同订单并提交流程
|
|
|
|
|
*
|
|
|
|
|
* @param bo 合同订单(项目信息)
|
|
|
|
|
* @return 合同订单(项目信息)
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
@GlobalTransactional(rollbackFor = Exception.class)
|
|
|
|
|
public ErpProjectInfoVo submitContractOrderAndFlowStart(ErpProjectInfoBo bo) {
|
|
|
|
|
// 先保存项目信息和项目阶段计划
|
|
|
|
|
ErpProjectInfoVo vo = saveContractOrder(bo);
|
|
|
|
|
|
|
|
|
|
if (1==1) {
|
|
|
|
|
return vo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 后端发起需要忽略权限
|
|
|
|
|
bo.getVariables().put("ignore", true);
|
|
|
|
|
RemoteStartProcess startProcess = new RemoteStartProcess();
|
|
|
|
|
startProcess.setBusinessId(bo.getProjectId().toString());
|
|
|
|
|
startProcess.setFlowCode(bo.getFlowCode());
|
|
|
|
|
startProcess.setVariables(bo.getVariables());
|
|
|
|
|
startProcess.setBizExt(bo.getBizExt());
|
|
|
|
|
bo.getBizExt().setBusinessId(startProcess.getBusinessId());
|
|
|
|
|
boolean flagOne = remoteWorkflowService.startCompleteTask(startProcess);
|
|
|
|
|
if (!flagOne) {
|
|
|
|
|
throw new ServiceException("流程发起异常");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return vo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 保存项目阶段计划列表
|
|
|
|
|
*
|
|
|
|
|
* @param projectInfo 项目信息
|
|
|
|
|
* @param planStageList 项目阶段计划列表
|
|
|
|
|
*/
|
|
|
|
|
private void savePlanStageList(ErpProjectInfo projectInfo, List<ErpProjectPlanStageBo> planStageList) {
|
|
|
|
|
if (projectInfo == null || projectInfo.getProjectId() == null) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 如果planStageList为null,表示前端没有传递该字段,不处理
|
|
|
|
|
if (planStageList == null) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Long projectId = projectInfo.getProjectId();
|
|
|
|
|
// 先获取或创建项目计划
|
|
|
|
|
Long projectPlanId = getOrCreateProjectPlan(projectInfo);
|
|
|
|
|
|
|
|
|
|
// 查询原有的项目阶段计划ID列表(根据projectPlanId查询)
|
|
|
|
|
LambdaQueryWrapper<ErpProjectPlanStage> queryWrapper = new LambdaQueryWrapper<>();
|
|
|
|
|
queryWrapper.eq(ErpProjectPlanStage::getProjectPlanId, projectPlanId);
|
|
|
|
|
List<ErpProjectPlanStage> existingStages = planStageMapper.selectList(queryWrapper);
|
|
|
|
|
List<Long> existingIds = existingStages.stream()
|
|
|
|
|
.map(ErpProjectPlanStage::getPlanStageId)
|
|
|
|
|
.filter(Objects::nonNull)
|
|
|
|
|
.collect(Collectors.toList());
|
|
|
|
|
|
|
|
|
|
// 收集前端传来的ID列表(已存在的记录)
|
|
|
|
|
List<Long> newIds = planStageList.stream()
|
|
|
|
|
.map(ErpProjectPlanStageBo::getPlanStageId)
|
|
|
|
|
.filter(Objects::nonNull)
|
|
|
|
|
.collect(Collectors.toList());
|
|
|
|
|
|
|
|
|
|
// 删除不在新列表中的原有记录
|
|
|
|
|
List<Long> idsToDelete = existingIds.stream()
|
|
|
|
|
.filter(id -> !newIds.contains(id))
|
|
|
|
|
.collect(Collectors.toList());
|
|
|
|
|
if (CollUtil.isNotEmpty(idsToDelete)) {
|
|
|
|
|
planStageMapper.deleteBatchIds(idsToDelete);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 保存或更新项目阶段计划
|
|
|
|
|
for (ErpProjectPlanStageBo stageBo : planStageList) {
|
|
|
|
|
ErpProjectPlanStage stage = MapstructUtils.convert(stageBo, ErpProjectPlanStage.class);
|
|
|
|
|
stage.setProjectId(projectId);
|
|
|
|
|
stage.setProjectPlanId(projectPlanId);
|
|
|
|
|
if (StringUtils.isNull(stage.getPlanStageId())) {
|
|
|
|
|
// 新增
|
|
|
|
|
planStageMapper.insert(stage);
|
|
|
|
|
} else {
|
|
|
|
|
// 更新
|
|
|
|
|
planStageMapper.updateById(stage);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 获取或创建项目计划
|
|
|
|
|
*
|
|
|
|
|
* @param projectInfo 项目信息
|
|
|
|
|
* @return 项目计划ID
|
|
|
|
|
*/
|
|
|
|
|
private Long getOrCreateProjectPlan(ErpProjectInfo projectInfo) {
|
|
|
|
|
if (projectInfo == null || projectInfo.getProjectId() == null) {
|
|
|
|
|
throw new ServiceException("项目信息不存在");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Long projectId = projectInfo.getProjectId();
|
|
|
|
|
// 查询是否已存在项目计划
|
|
|
|
|
LambdaQueryWrapper<ErpProjectPlan> queryWrapper = new LambdaQueryWrapper<>();
|
|
|
|
|
queryWrapper.eq(ErpProjectPlan::getProjectId, projectId);
|
|
|
|
|
queryWrapper.eq(ErpProjectPlan::getDelFlag, "0");
|
|
|
|
|
queryWrapper.orderByDesc(ErpProjectPlan::getCreateTime);
|
|
|
|
|
queryWrapper.last("LIMIT 1");
|
|
|
|
|
List<ErpProjectPlan> existingPlans = projectPlanMapper.selectList(queryWrapper);
|
|
|
|
|
|
|
|
|
|
if (CollUtil.isNotEmpty(existingPlans)) {
|
|
|
|
|
return existingPlans.get(0).getProjectPlanId();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 不存在则创建新的项目计划
|
|
|
|
|
ErpProjectPlan newPlan = new ErpProjectPlan();
|
|
|
|
|
newPlan.setProjectId(projectId);
|
|
|
|
|
newPlan.setProjectPlanStatus("3");
|
|
|
|
|
newPlan.setFlowStatus("finish");
|
|
|
|
|
newPlan.setManagerId(projectInfo.getManagerId());
|
|
|
|
|
newPlan.setChargeId(projectInfo.getChargeId());
|
|
|
|
|
|
|
|
|
|
// 生成项目计划编号
|
|
|
|
|
String projectPlanCode = remoteCodeRuleService.selectCodeRuleCode("1010");
|
|
|
|
|
if (StringUtils.isBlank(projectPlanCode)) {
|
|
|
|
|
throw new ServiceException("生成项目计划编号失败");
|
|
|
|
|
}
|
|
|
|
|
newPlan.setProjectPlanCode(projectPlanCode);
|
|
|
|
|
|
|
|
|
|
projectPlanMapper.insert(newPlan);
|
|
|
|
|
return newPlan.getProjectPlanId();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 保存前的数据校验
|
|
|
|
|
*/
|
|
|
|
|
private void validEntityBeforeSave(ErpProjectInfo entity) {
|
|
|
|
|
//TODO 做一些数据校验,如唯一约束
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 总体流程监听(例如: 草稿,撤销,退回,作废,终止,已完成等)
|
|
|
|
|
*
|
|
|
|
|
* @param processEvent 参数
|
|
|
|
|
*/
|
|
|
|
|
@EventListener(condition = "#processEvent.flowCode.startsWith('htdd')")
|
|
|
|
|
public void processHandler(ProcessEvent processEvent) {
|
|
|
|
|
TenantHelper.dynamic(processEvent.getTenantId(), () -> {
|
|
|
|
|
log.info("当前任务执行了{}", processEvent.toString());
|
|
|
|
|
ErpProjectInfo projectInfo = projectInfoMapper.selectById(
|
|
|
|
|
Convert.toLong(processEvent.getBusinessId())
|
|
|
|
|
);
|
|
|
|
|
if (projectInfo != null) {
|
|
|
|
|
projectInfo.setFlowStatus(processEvent.getStatus());
|
|
|
|
|
Map<String, Object> params = processEvent.getParams();
|
|
|
|
|
if (MapUtil.isNotEmpty(params)) {
|
|
|
|
|
// 办理人
|
|
|
|
|
if (params.containsKey("handler")) {
|
|
|
|
|
// TODO: 处理办理人
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// 更新流程状态
|
|
|
|
|
if (Objects.equals(processEvent.getStatus(), BusinessStatusEnum.WAITING.getStatus())) {
|
|
|
|
|
projectInfo.setProjectStatus(OAStatusEnum.APPROVING.getStatus());
|
|
|
|
|
} else if (Objects.equals(processEvent.getStatus(), BusinessStatusEnum.FINISH.getStatus())) {
|
|
|
|
|
projectInfo.setProjectStatus(OAStatusEnum.COMPLETED.getStatus());
|
|
|
|
|
} else if (Objects.equals(processEvent.getStatus(), BusinessStatusEnum.INVALID.getStatus())
|
|
|
|
|
|| Objects.equals(processEvent.getStatus(), BusinessStatusEnum.TERMINATION.getStatus())) {
|
|
|
|
|
projectInfo.setProjectStatus(OAStatusEnum.INVALID.getStatus());
|
|
|
|
|
} else if (Objects.equals(processEvent.getStatus(), BusinessStatusEnum.BACK.getStatus())
|
|
|
|
|
|| Objects.equals(processEvent.getStatus(), BusinessStatusEnum.CANCEL.getStatus())) {
|
|
|
|
|
projectInfo.setProjectStatus(OAStatusEnum.DRAFT.getStatus());
|
|
|
|
|
} else if (Objects.equals(processEvent.getStatus(), BusinessStatusEnum.DRAFT.getStatus())) {
|
|
|
|
|
projectInfo.setProjectStatus(OAStatusEnum.DRAFT.getStatus());
|
|
|
|
|
}
|
|
|
|
|
projectInfoMapper.updateById(projectInfo);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|