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);
}
}
}