package com.ruoyi.production.service.impl; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.ruoyi.common.exception.ServiceException; import com.ruoyi.production.bean.dto.ProductionBomStructureDto; import com.ruoyi.production.bean.vo.ProductionBomStructureVo; import com.ruoyi.production.mapper.ProductionBomStructureMapper; import com.ruoyi.production.mapper.ProductionOperationTaskMapper; import com.ruoyi.production.mapper.ProductionOrderBomMapper; import com.ruoyi.production.mapper.ProductionOrderMapper; import com.ruoyi.production.mapper.ProductionOrderRoutingMapper; import com.ruoyi.production.mapper.ProductionOrderRoutingOperationMapper; import com.ruoyi.production.mapper.ProductionOrderRoutingOperationParamMapper; import com.ruoyi.production.mapper.ProductionProductMainMapper; import com.ruoyi.production.pojo.ProductionBomStructure; import com.ruoyi.production.pojo.ProductionOperationTask; import com.ruoyi.production.pojo.ProductionOrder; import com.ruoyi.production.pojo.ProductionOrderBom; import com.ruoyi.production.pojo.ProductionOrderRouting; import com.ruoyi.production.pojo.ProductionOrderRoutingOperation; import com.ruoyi.production.pojo.ProductionOrderRoutingOperationParam; import com.ruoyi.production.pojo.ProductionProductMain; import com.ruoyi.production.service.ProductionBomStructureService; import com.ruoyi.technology.mapper.TechnologyOperationMapper; import com.ruoyi.technology.mapper.TechnologyOperationParamMapper; import com.ruoyi.technology.mapper.TechnologyParamMapper; import com.ruoyi.technology.pojo.TechnologyOperation; import com.ruoyi.technology.pojo.TechnologyOperationParam; import com.ruoyi.technology.pojo.TechnologyParam; import lombok.RequiredArgsConstructor; import org.springframework.beans.BeanUtils; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import java.math.BigDecimal; import java.time.LocalDate; import java.time.format.DateTimeFormatter; import java.util.*; import java.util.stream.Collectors; /** *

* 生产订单BOM产品结构 服务实现类 *

* * @author 芯导软件(江苏)有限公司 * @since 2026-04-21 03:55:52 */ @Service @RequiredArgsConstructor() public class ProductionBomStructureServiceImpl extends ServiceImpl implements ProductionBomStructureService { private final ProductionBomStructureMapper productionBomStructureMapper; private final ProductionOrderBomMapper productionOrderBomMapper; private final ProductionOrderMapper productionOrderMapper; private final ProductionOrderRoutingMapper productionOrderRoutingMapper; private final ProductionOrderRoutingOperationMapper productionOrderRoutingOperationMapper; private final ProductionOrderRoutingOperationParamMapper productionOrderRoutingOperationParamMapper; private final ProductionOperationTaskMapper productionOperationTaskMapper; private final ProductionProductMainMapper productionProductMainMapper; private final TechnologyOperationMapper technologyOperationMapper; private final TechnologyOperationParamMapper technologyOperationParamMapper; private final TechnologyParamMapper technologyParamMapper; /** * 根据BOM查询并组装结构树。 */ @Override public List listByBomId(Long bomId) { // 按BOMID查询生产结构数据 List list = productionBomStructureMapper.listByBomId(bomId); Map map = new HashMap<>(); for (ProductionBomStructureVo node : list) { node.setChildren(new ArrayList<>()); map.put(node.getId(), node); } List tree = new ArrayList<>(); for (ProductionBomStructureVo node : list) { Long parentId = node.getParentId(); if (parentId == null || parentId == 0L) { tree.add(node); continue; } ProductionBomStructureVo parent = map.get(parentId); if (parent != null) { parent.getChildren().add(node); } } return tree; } @Override @Transactional(rollbackFor = Exception.class) public Boolean addProductionBomStructure(ProductionBomStructureDto dto) { // 新增生产BOM结构 // 读取当前订单BOM主键,并把前端树结构拍平成列表 Long orderBomId = dto.getProductionOrderBomId(); List flatDtoList = new ArrayList<>(); flattenTree(dto.getChildren(), flatDtoList); // 查询数据库已有结构,用于后续做增删改对比 List dbList = this.list(new LambdaQueryWrapper() .eq(ProductionBomStructure::getProductionOrderBomId, orderBomId)); // 收集前端仍然存在的节点ID Set frontendIds = new HashSet<>(); for (ProductionBomStructureDto item : flatDtoList) { if (item.getId() != null) { frontendIds.add(item.getId()); } } // 计算需要删除的节点(数据库有、前端已删除) Set deleteIds = new HashSet<>(); for (ProductionBomStructure dbItem : dbList) { if (!frontendIds.contains(dbItem.getId())) { deleteIds.add(dbItem.getId()); } } // 先删掉前端已经移除的节点 if (!deleteIds.isEmpty()) { this.removeByIds(deleteIds); } // 按是否有ID拆分为新增和更新,同时缓存新增节点的临时ID映射 List insertList = new ArrayList<>(); List updateList = new ArrayList<>(); Map tempEntityMap = new HashMap<>(); for (ProductionBomStructureDto item : flatDtoList) { ProductionBomStructure entity = new ProductionBomStructure(); BeanUtils.copyProperties(item, entity); entity.setProductionOrderBomId(orderBomId); if (item.getId() == null) { entity.setParentId(null); insertList.add(entity); tempEntityMap.put(item.getTempId(), entity); } else { updateList.add(entity); } } // 批量新增,拿到数据库生成的真实ID if (!insertList.isEmpty()) { this.saveBatch(insertList); } // 新增节点二次回写父ID(前端传的是临时父ID) List parentFixList = new ArrayList<>(); for (ProductionBomStructureDto item : flatDtoList) { if (item.getId() == null && item.getParentTempId() != null) { ProductionBomStructure child = tempEntityMap.get(item.getTempId()); if (child == null) { continue; } ProductionBomStructure parent = tempEntityMap.get(item.getParentTempId()); // 父节点是本次新增时,直接用新增后的真实ID;否则回退为前端传入父ID Long realParentId = parent != null ? parent.getId() : Long.valueOf(item.getParentTempId()); child.setParentId(realParentId); parentFixList.add(child); } } // 回写新增节点的父子关系 if (!parentFixList.isEmpty()) { this.updateBatchById(parentFixList); } // 批量更新已有节点 if (!updateList.isEmpty()) { this.updateBatchById(updateList); } syncDemandedQuantityAndTaskPlanQuantity(orderBomId, dto.getProductionOrderId()); return true; } private void syncDemandedQuantityAndTaskPlanQuantity(Long orderBomId, Long productionOrderId) { if (orderBomId == null) { return; } ProductionOrderBom orderBom = productionOrderBomMapper.selectById(orderBomId); if (orderBom == null) { return; } Long currentProductionOrderId = productionOrderId != null ? productionOrderId : orderBom.getProductionOrderId(); if (currentProductionOrderId == null) { return; } ProductionOrder productionOrder = productionOrderMapper.selectById(currentProductionOrderId); if (productionOrder == null) { return; } BigDecimal orderQuantity = defaultDecimal(productionOrder.getQuantity()); List structureList = this.list( Wrappers.lambdaQuery() .eq(ProductionBomStructure::getProductionOrderBomId, orderBomId) .orderByAsc(ProductionBomStructure::getId)); //同步需求数量 syncStructureDemandedQuantity(structureList, orderQuantity); Long rootProductModelId = orderBom.getProductModelId() != null ? orderBom.getProductModelId() : productionOrder.getProductModelId(); //同步生产工艺路线 syncRoutingOperationsByBom(currentProductionOrderId, productionOrder, orderBom, structureList, rootProductModelId); //同步工单 syncTaskPlanQuantity( currentProductionOrderId, structureList, orderQuantity, rootProductModelId); } private void syncStructureDemandedQuantity(List structureList, BigDecimal orderQuantity) { if (structureList == null || structureList.isEmpty()) { return; } List updateList = new ArrayList<>(); for (ProductionBomStructure structure : structureList) { if (structure == null || structure.getId() == null) { continue; } BigDecimal demandedQuantity = defaultDecimal(structure.getUnitQuantity()).multiply(orderQuantity); if (compareDecimal(structure.getDemandedQuantity(), demandedQuantity) == 0) { continue; } ProductionBomStructure update = new ProductionBomStructure(); update.setId(structure.getId()); update.setDemandedQuantity(demandedQuantity); updateList.add(update); structure.setDemandedQuantity(demandedQuantity); } if (!updateList.isEmpty()) { this.updateBatchById(updateList); } } private void syncTaskPlanQuantity(Long productionOrderId, List structureList, BigDecimal orderQuantity, Long rootProductModelId) { List taskList = productionOperationTaskMapper.selectList( Wrappers.lambdaQuery() .eq(ProductionOperationTask::getProductionOrderId, productionOrderId) .orderByAsc(ProductionOperationTask::getId)); if (taskList == null || taskList.isEmpty()) { return; } Set routingOperationIds = taskList.stream() .map(ProductionOperationTask::getProductionOrderRoutingOperationId) .filter(Objects::nonNull) .collect(Collectors.toSet()); if (routingOperationIds.isEmpty()) { return; } Map routingOperationMap = productionOrderRoutingOperationMapper .selectBatchIds(routingOperationIds) .stream() .filter(item -> item != null && item.getId() != null) .collect(Collectors.toMap(ProductionOrderRoutingOperation::getId, item -> item, (left, right) -> left)); // Keep task plan quantities aligned with the same order BOM snapshot demand used during snapshot creation. Map demandedQuantityMap = buildOperationDemandedQuantityMap(structureList, rootProductModelId); for (ProductionOperationTask task : taskList) { if (task == null || task.getId() == null || task.getProductionOrderRoutingOperationId() == null) { continue; } ProductionOrderRoutingOperation routingOperation = routingOperationMap.get(task.getProductionOrderRoutingOperationId()); if (routingOperation == null) { continue; } BigDecimal planQuantity = resolveTaskPlanQuantity( routingOperation, demandedQuantityMap, orderQuantity, rootProductModelId); if (compareDecimal(task.getPlanQuantity(), planQuantity) == 0) { continue; } ProductionOperationTask update = new ProductionOperationTask(); update.setId(task.getId()); update.setPlanQuantity(planQuantity); productionOperationTaskMapper.updateById(update); } } private void syncRoutingOperationsByBom(Long productionOrderId, ProductionOrder productionOrder, ProductionOrderBom orderBom, List structureList, Long rootProductModelId) { ProductionOrderRouting orderRouting = getOrCreateOrderRoutingSnapshot(productionOrderId, productionOrder, orderBom, rootProductModelId); List desiredOperationList = buildDesiredRoutingOperationList(structureList, rootProductModelId); List existingOperationList = productionOrderRoutingOperationMapper.selectList( Wrappers.lambdaQuery() .eq(ProductionOrderRoutingOperation::getOrderRoutingId, orderRouting.getId()) .eq(ProductionOrderRoutingOperation::getProductionOrderId, productionOrderId) .orderByAsc(ProductionOrderRoutingOperation::getDragSort) .orderByAsc(ProductionOrderRoutingOperation::getId)); Map> existingBucketMap = buildExistingRoutingOperationBucketMap(existingOperationList); List finalOperationList = new ArrayList<>(); for (ProductionOrderRoutingOperation desiredOperation : desiredOperationList) { String bucketKey = buildRoutingOperationBucketKey( desiredOperation.getTechnologyOperationId(), desiredOperation.getProductModelId()); Deque matchedQueue = existingBucketMap.get(bucketKey); ProductionOrderRoutingOperation matchedOperation = matchedQueue == null ? null : matchedQueue.pollFirst(); if (matchedOperation == null) { matchedOperation = insertRoutingOperationSnapshot(orderRouting.getId(), productionOrderId, desiredOperation); } else { updateRoutingOperationSnapshotIfNecessary(matchedOperation, orderRouting.getId(), productionOrderId, desiredOperation); } finalOperationList.add(matchedOperation); } for (Deque queue : existingBucketMap.values()) { while (queue != null && !queue.isEmpty()) { removeRoutingOperationSnapshot(queue.pollFirst()); } } syncRoutingOperationTasks(productionOrderId, finalOperationList); } private ProductionOrderRouting getOrCreateOrderRoutingSnapshot(Long productionOrderId, ProductionOrder productionOrder, ProductionOrderBom orderBom, Long rootProductModelId) { ProductionOrderRouting orderRouting = productionOrderRoutingMapper.selectOne( Wrappers.lambdaQuery() .eq(ProductionOrderRouting::getProductionOrderId, productionOrderId) .orderByDesc(ProductionOrderRouting::getId) .last("limit 1")); if (orderRouting == null) { orderRouting = new ProductionOrderRouting(); orderRouting.setProductionOrderId(productionOrderId); orderRouting.setProductModelId(rootProductModelId); orderRouting.setTechnologyRoutingId(productionOrder == null ? null : productionOrder.getTechnologyRoutingId()); orderRouting.setBomId(orderBom == null ? null : orderBom.getBomId()); orderRouting.setOrderBomId(orderBom == null ? null : orderBom.getId()); productionOrderRoutingMapper.insert(orderRouting); return orderRouting; } ProductionOrderRouting update = new ProductionOrderRouting(); update.setId(orderRouting.getId()); boolean changed = false; if (!Objects.equals(orderRouting.getProductModelId(), rootProductModelId)) { update.setProductModelId(rootProductModelId); orderRouting.setProductModelId(rootProductModelId); changed = true; } Long technologyRoutingId = productionOrder == null ? null : productionOrder.getTechnologyRoutingId(); if (!Objects.equals(orderRouting.getTechnologyRoutingId(), technologyRoutingId)) { update.setTechnologyRoutingId(technologyRoutingId); orderRouting.setTechnologyRoutingId(technologyRoutingId); changed = true; } Long bomId = orderBom == null ? null : orderBom.getBomId(); if (!Objects.equals(orderRouting.getBomId(), bomId)) { update.setBomId(bomId); orderRouting.setBomId(bomId); changed = true; } Long orderBomId = orderBom == null ? null : orderBom.getId(); if (!Objects.equals(orderRouting.getOrderBomId(), orderBomId)) { update.setOrderBomId(orderBomId); orderRouting.setOrderBomId(orderBomId); changed = true; } if (changed) { productionOrderRoutingMapper.updateById(update); } return orderRouting; } private List buildDesiredRoutingOperationList(List structureList, Long rootProductModelId) { if (structureList == null || structureList.isEmpty()) { return Collections.emptyList(); } Map structureById = structureList.stream() .filter(item -> item != null && item.getId() != null) .collect(Collectors.toMap(ProductionBomStructure::getId, item -> item, (left, right) -> left)); Map uniqueOperationMap = new LinkedHashMap<>(); for (ProductionBomStructure bomStructure : structureList) { if (bomStructure == null || bomStructure.getTechnologyOperationId() == null) { continue; } Long outputProductModelId = resolveOutputProductModelId(resolveOperationOutputNode(bomStructure, structureById), rootProductModelId); uniqueOperationMap.putIfAbsent(buildBomOperationDedupKey(bomStructure, outputProductModelId), bomStructure); } List desiredOperationList = new ArrayList<>(); int dragSort = 1; for (ProductionBomStructure bomStructure : uniqueOperationMap.values()) { Long outputProductModelId = resolveOutputProductModelId(resolveOperationOutputNode(bomStructure, structureById), rootProductModelId); TechnologyOperation technologyOperation = getTechnologyOperation(bomStructure.getTechnologyOperationId()); ProductionOrderRoutingOperation routingOperation = new ProductionOrderRoutingOperation(); routingOperation.setProductModelId(outputProductModelId); routingOperation.setTechnologyOperationId(bomStructure.getTechnologyOperationId()); routingOperation.setOperationName(technologyOperation == null ? null : technologyOperation.getName()); routingOperation.setIsQuality(technologyOperation == null ? null : technologyOperation.getIsQuality()); routingOperation.setIsProduction(technologyOperation == null ? null : technologyOperation.getIsProduction()); routingOperation.setType(technologyOperation == null ? null : technologyOperation.getType()); routingOperation.setDragSort(dragSort++); desiredOperationList.add(routingOperation); } return desiredOperationList; } private Map> buildExistingRoutingOperationBucketMap(List existingOperationList) { Map> existingBucketMap = new LinkedHashMap<>(); if (existingOperationList == null || existingOperationList.isEmpty()) { return existingBucketMap; } for (ProductionOrderRoutingOperation routingOperation : existingOperationList) { String bucketKey = buildRoutingOperationBucketKey( routingOperation.getTechnologyOperationId(), routingOperation.getProductModelId()); existingBucketMap.computeIfAbsent(bucketKey, key -> new ArrayDeque<>()).addLast(routingOperation); } return existingBucketMap; } private ProductionOrderRoutingOperation insertRoutingOperationSnapshot(Long orderRoutingId, Long productionOrderId, ProductionOrderRoutingOperation desiredOperation) { ProductionOrderRoutingOperation insert = new ProductionOrderRoutingOperation(); insert.setOrderRoutingId(orderRoutingId); insert.setProductionOrderId(productionOrderId); insert.setProductModelId(desiredOperation.getProductModelId()); insert.setTechnologyOperationId(desiredOperation.getTechnologyOperationId()); insert.setOperationName(desiredOperation.getOperationName()); insert.setIsQuality(desiredOperation.getIsQuality()); insert.setIsProduction(desiredOperation.getIsProduction()); insert.setType(desiredOperation.getType()); insert.setDragSort(desiredOperation.getDragSort()); productionOrderRoutingOperationMapper.insert(insert); syncRoutingOperationParams(insert.getId(), productionOrderId, insert.getTechnologyOperationId()); return insert; } private void updateRoutingOperationSnapshotIfNecessary(ProductionOrderRoutingOperation currentOperation, Long orderRoutingId, Long productionOrderId, ProductionOrderRoutingOperation desiredOperation) { if (currentOperation == null || currentOperation.getId() == null) { return; } ProductionOrderRoutingOperation update = new ProductionOrderRoutingOperation(); update.setId(currentOperation.getId()); boolean changed = false; if (!Objects.equals(currentOperation.getOrderRoutingId(), orderRoutingId)) { update.setOrderRoutingId(orderRoutingId); currentOperation.setOrderRoutingId(orderRoutingId); changed = true; } if (!Objects.equals(currentOperation.getProductionOrderId(), productionOrderId)) { update.setProductionOrderId(productionOrderId); currentOperation.setProductionOrderId(productionOrderId); changed = true; } if (!Objects.equals(currentOperation.getProductModelId(), desiredOperation.getProductModelId())) { update.setProductModelId(desiredOperation.getProductModelId()); currentOperation.setProductModelId(desiredOperation.getProductModelId()); changed = true; } if (!Objects.equals(currentOperation.getTechnologyOperationId(), desiredOperation.getTechnologyOperationId())) { update.setTechnologyOperationId(desiredOperation.getTechnologyOperationId()); currentOperation.setTechnologyOperationId(desiredOperation.getTechnologyOperationId()); changed = true; } if (!Objects.equals(currentOperation.getOperationName(), desiredOperation.getOperationName())) { update.setOperationName(desiredOperation.getOperationName()); currentOperation.setOperationName(desiredOperation.getOperationName()); changed = true; } if (!Objects.equals(currentOperation.getIsQuality(), desiredOperation.getIsQuality())) { update.setIsQuality(desiredOperation.getIsQuality()); currentOperation.setIsQuality(desiredOperation.getIsQuality()); changed = true; } if (!Objects.equals(currentOperation.getIsProduction(), desiredOperation.getIsProduction())) { update.setIsProduction(desiredOperation.getIsProduction()); currentOperation.setIsProduction(desiredOperation.getIsProduction()); changed = true; } if (!Objects.equals(currentOperation.getType(), desiredOperation.getType())) { update.setType(desiredOperation.getType()); currentOperation.setType(desiredOperation.getType()); changed = true; } if (!Objects.equals(currentOperation.getDragSort(), desiredOperation.getDragSort())) { update.setDragSort(desiredOperation.getDragSort()); currentOperation.setDragSort(desiredOperation.getDragSort()); changed = true; } if (changed) { productionOrderRoutingOperationMapper.updateById(update); } } private void removeRoutingOperationSnapshot(ProductionOrderRoutingOperation routingOperation) { if (routingOperation == null || routingOperation.getId() == null) { return; } ProductionOperationTask task = productionOperationTaskMapper.selectOne( Wrappers.lambdaQuery() .eq(ProductionOperationTask::getProductionOrderRoutingOperationId, routingOperation.getId()) .last("limit 1")); if (task != null) { validateTaskCanRemove(task); productionOperationTaskMapper.deleteById(task.getId()); } productionOrderRoutingOperationParamMapper.delete( Wrappers.lambdaQuery() .eq(ProductionOrderRoutingOperationParam::getProductionOrderRoutingOperationId, routingOperation.getId())); productionOrderRoutingOperationMapper.deleteById(routingOperation.getId()); } private void syncRoutingOperationTasks(Long productionOrderId, List routingOperationList) { if (routingOperationList == null || routingOperationList.isEmpty()) { return; } List routingOperationIdList = routingOperationList.stream() .map(ProductionOrderRoutingOperation::getId) .filter(Objects::nonNull) .collect(Collectors.toList()); if (routingOperationIdList.isEmpty()) { return; } Map taskByRoutingOperationId = productionOperationTaskMapper.selectList( Wrappers.lambdaQuery() .in(ProductionOperationTask::getProductionOrderRoutingOperationId, routingOperationIdList) .orderByAsc(ProductionOperationTask::getId)) .stream() .filter(item -> item != null && item.getProductionOrderRoutingOperationId() != null) .collect(Collectors.toMap( ProductionOperationTask::getProductionOrderRoutingOperationId, item -> item, (left, right) -> left, LinkedHashMap::new)); for (int i = 0; i < routingOperationList.size(); i++) { ProductionOrderRoutingOperation routingOperation = routingOperationList.get(i); if (routingOperation == null || routingOperation.getId() == null) { continue; } boolean shouldHaveTask = i == routingOperationList.size() - 1 || Boolean.TRUE.equals(routingOperation.getIsProduction()); ProductionOperationTask existingTask = taskByRoutingOperationId.get(routingOperation.getId()); if (shouldHaveTask) { if (existingTask == null) { ProductionOperationTask task = new ProductionOperationTask(); task.setProductionOrderId(productionOrderId); task.setProductionOrderRoutingOperationId(routingOperation.getId()); task.setPlanQuantity(BigDecimal.ZERO); task.setCompleteQuantity(BigDecimal.ZERO); task.setWorkOrderNo(generateNextTaskNo()); task.setStatus(2); productionOperationTaskMapper.insert(task); } continue; } if (existingTask != null) { validateTaskCanRemove(existingTask); productionOperationTaskMapper.deleteById(existingTask.getId()); } } } private void validateTaskCanRemove(ProductionOperationTask task) { if (task == null || task.getId() == null) { return; } if (defaultDecimal(task.getCompleteQuantity()).compareTo(BigDecimal.ZERO) > 0) { throw new ServiceException("工序已产生报工记录,无法根据 BOM 变更删除对应工序快照"); } long reportCount = productionProductMainMapper.selectCount( Wrappers.lambdaQuery() .eq(ProductionProductMain::getProductionOperationTaskId, task.getId())); if (reportCount > 0) { throw new ServiceException("工序已产生报工记录,无法根据 BOM 变更删除对应工单"); } } private void syncRoutingOperationParams(Long routingOperationId, Long productionOrderId, Long technologyOperationId) { if (routingOperationId == null || technologyOperationId == null) { return; } List operationParamList = technologyOperationParamMapper.selectList( Wrappers.lambdaQuery() .eq(TechnologyOperationParam::getTechnologyOperationId, technologyOperationId) .orderByAsc(TechnologyOperationParam::getId)); for (TechnologyOperationParam operationParam : operationParamList) { TechnologyParam technologyParam = technologyParamMapper.selectById(operationParam.getTechnologyParamId()); if (technologyParam == null) { continue; } ProductionOrderRoutingOperationParam snapshot = new ProductionOrderRoutingOperationParam(); snapshot.setProductionOrderId(productionOrderId); snapshot.setProductionOrderRoutingOperationId(routingOperationId); snapshot.setTechnologyOperationId(operationParam.getTechnologyOperationId()); snapshot.setTechnologyOperationParamId(operationParam.getId()); snapshot.setParamId(technologyParam.getId()); snapshot.setParamCode(technologyParam.getParamCode()); snapshot.setParamName(technologyParam.getParamName()); snapshot.setParamType(technologyParam.getParamType()); snapshot.setParamFormat(technologyParam.getParamFormat()); snapshot.setUnit(technologyParam.getUnit()); snapshot.setIsRequired(technologyParam.getIsRequired()); snapshot.setRemark(technologyParam.getRemark()); snapshot.setStandardValue(operationParam.getStandardValue()); productionOrderRoutingOperationParamMapper.insert(snapshot); } } private Map buildOperationDemandedQuantityMap(List structureList, Long rootProductModelId) { if (structureList == null || structureList.isEmpty()) { return Collections.emptyMap(); } Map structureById = structureList.stream() .filter(item -> item != null && item.getId() != null) .collect(Collectors.toMap(ProductionBomStructure::getId, item -> item, (left, right) -> left)); Map demandedQuantityMap = new HashMap<>(); Set mergedOutputNodeKeySet = new HashSet<>(); for (ProductionBomStructure bomStructure : structureList) { if (bomStructure == null || bomStructure.getTechnologyOperationId() == null) { continue; } // Resolve the output node first, then read the output node demand for the task plan quantity. ProductionBomStructure outputNode = resolveOperationOutputNode(bomStructure, structureById); Long outputProductModelId = resolveOutputProductModelId(outputNode, rootProductModelId); if (outputProductModelId == null) { continue; } String mergedOutputNodeKey = buildOperationOutputNodeKey( bomStructure.getTechnologyOperationId(), outputNode == null ? null : outputNode.getId(), outputProductModelId); if (!mergedOutputNodeKeySet.add(mergedOutputNodeKey)) { continue; } // Multiple input rows can point to the same output node, so only count that output demand once. String key = buildOperationDemandedQuantityKey(bomStructure.getTechnologyOperationId(), outputProductModelId); demandedQuantityMap.merge(key, defaultDecimal(outputNode == null ? null : outputNode.getDemandedQuantity()), BigDecimal::add); } return demandedQuantityMap; } private BigDecimal resolveTaskPlanQuantity(ProductionOrderRoutingOperation routingOperation, Map demandedQuantityMap, BigDecimal orderQuantity, Long rootProductModelId) { if (routingOperation == null || demandedQuantityMap == null || demandedQuantityMap.isEmpty()) { return orderQuantity; } Long outputProductModelId = routingOperation.getProductModelId() != null ? routingOperation.getProductModelId() : rootProductModelId; String key = buildOperationDemandedQuantityKey( routingOperation.getTechnologyOperationId(), outputProductModelId); BigDecimal planQuantity = demandedQuantityMap.get(key); return planQuantity != null ? planQuantity : orderQuantity; } private String buildOperationDemandedQuantityKey(Long operationId, Long outputProductModelId) { return String.valueOf(operationId) + "#" + String.valueOf(outputProductModelId); } private String buildRoutingOperationBucketKey(Long operationId, Long outputProductModelId) { return String.valueOf(operationId) + "#" + String.valueOf(outputProductModelId); } private String buildBomOperationDedupKey(ProductionBomStructure bomStructure, Long outputProductModelId) { Long operationId = bomStructure == null ? null : bomStructure.getTechnologyOperationId(); Long parentId = bomStructure == null ? null : bomStructure.getParentId(); return operationId + "#" + outputProductModelId + "#" + parentId; } private String buildOperationOutputNodeKey(Long operationId, Long outputNodeId, Long outputProductModelId) { return String.valueOf(operationId) + "#" + String.valueOf(outputNodeId) + "#" + String.valueOf(outputProductModelId); } private ProductionBomStructure resolveOperationOutputNode(ProductionBomStructure bomStructure, Map structureById) { if (bomStructure == null) { return null; } // The root node is the first output node; other rows use their direct parent as the current operation output. if (bomStructure.getParentId() == null) { return bomStructure; } ProductionBomStructure parent = structureById.get(bomStructure.getParentId()); return parent != null ? parent : bomStructure; } private Long resolveOutputProductModelId(ProductionBomStructure outputNode, Long rootProductModelId) { if (outputNode == null) { return rootProductModelId; } return outputNode.getProductModelId() != null ? outputNode.getProductModelId() : rootProductModelId; } private TechnologyOperation getTechnologyOperation(Long technologyOperationId) { if (technologyOperationId == null) { return null; } return technologyOperationMapper.selectById(technologyOperationId); } private String generateNextTaskNo() { String datePrefix = LocalDate.now().format(DateTimeFormatter.ofPattern("yyyyMMdd")); ProductionOperationTask latestTask = productionOperationTaskMapper.selectOne( Wrappers.lambdaQuery() .likeRight(ProductionOperationTask::getWorkOrderNo, "GD" + datePrefix) .orderByDesc(ProductionOperationTask::getWorkOrderNo) .last("limit 1")); int sequenceNumber = 1; if (latestTask != null && latestTask.getWorkOrderNo() != null && latestTask.getWorkOrderNo().startsWith("GD" + datePrefix)) { try { sequenceNumber = Integer.parseInt(latestTask.getWorkOrderNo().substring(("GD" + datePrefix).length())) + 1; } catch (NumberFormatException ignored) { sequenceNumber = 1; } } return "GD" + String.format("%s%03d", datePrefix, sequenceNumber); } private BigDecimal defaultDecimal(BigDecimal value) { return value == null ? BigDecimal.ZERO : value; } private int compareDecimal(BigDecimal left, BigDecimal right) { return defaultDecimal(left).compareTo(defaultDecimal(right)); } /** * 将树形结构拍平成列表,便于统一保存。 */ private void flattenTree(List source, List result) { // 扁平化处理树 if (source == null) { return; } for (ProductionBomStructureDto node : source) { result.add(node); flattenTree(node.getChildren(), result); } } }