| package com.vincent.rsf.server.manager.service.impl; | 
|   | 
| import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | 
| import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper; | 
| import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; | 
| import com.vincent.rsf.framework.common.Cools; | 
| import com.vincent.rsf.framework.common.R; | 
| import com.vincent.rsf.framework.exception.CoolException; | 
| import com.vincent.rsf.server.api.utils.LocUtils; | 
| import com.vincent.rsf.server.common.constant.Constants; | 
| import com.vincent.rsf.server.manager.controller.dto.ExistDto; | 
| import com.vincent.rsf.server.manager.controller.dto.OrderOutItemDto; | 
| import com.vincent.rsf.server.manager.controller.params.*; | 
| import com.vincent.rsf.server.manager.enums.*; | 
| import com.vincent.rsf.server.manager.entity.*; | 
| import com.vincent.rsf.server.manager.mapper.AsnOrderMapper; | 
| import com.vincent.rsf.server.manager.service.*; | 
| import com.vincent.rsf.server.manager.utils.LocManageUtil; | 
| import com.vincent.rsf.server.manager.utils.OptimalAlgorithmUtil; | 
| import com.vincent.rsf.server.system.constant.SerialRuleCode; | 
| import com.vincent.rsf.server.system.utils.SerialRuleUtils; | 
| import org.apache.commons.lang3.StringUtils; | 
| import org.slf4j.Logger; | 
| import org.slf4j.LoggerFactory; | 
| import org.springframework.beans.BeanUtils; | 
| import org.springframework.beans.factory.annotation.Autowired; | 
| import org.springframework.stereotype.Service; | 
| import org.springframework.transaction.annotation.Transactional; | 
|   | 
| import java.math.BigDecimal; | 
| import java.util.*; | 
| import java.util.stream.Collectors; | 
|   | 
| import com.vincent.rsf.server.manager.enums.WaveRuleType; | 
|   | 
| /** | 
|  * @author Ryan | 
|  * @description | 
|  * @throws | 
|  * @return | 
|  * @time 2025/3/7 08:02 | 
|  */ | 
| @Service("outStockServiceImpl") | 
| public class OutStockServiceImpl extends ServiceImpl<AsnOrderMapper, WkOrder> implements OutStockService { | 
|   | 
|     public Logger logger = LoggerFactory.getLogger(this.getClass()); | 
|   | 
|   | 
|     @Autowired | 
|     private AsnOrderItemService asnOrderItemService; | 
|     @Autowired | 
|     private AsnOrderLogService asnOrderLogService; | 
|     @Autowired | 
|     private AsnOrderItemLogService asnOrderItemLogService; | 
|     @Autowired | 
|     private DeliveryItemService deliveryItemService; | 
|     @Autowired | 
|     private DeliveryService deliveryService; | 
|     @Autowired | 
|     private MatnrService matnrService; | 
|     @Autowired | 
|     private WaveService waveService; | 
|     @Autowired | 
|     private WaveItemService waveItemService; | 
|     @Autowired | 
|     private OutStockItemService outStockItemService; | 
|     @Autowired | 
|     private OutStockService outStockService; | 
|     @Autowired | 
|     private WaveRuleService waveRuleService; | 
|     @Autowired | 
|     private LocItemService locItemService; | 
|     @Autowired | 
|     private DeviceSiteService deviceSiteService; | 
|     @Autowired | 
|     private LocService locService; | 
|     @Autowired | 
|     private WaveOrderRelaServiceImpl waveOrderRelaService; | 
|   | 
|   | 
|     /** | 
|      * @param | 
|      * @return | 
|      * @author Ryan | 
|      * @description 更新或保存明细 | 
|      * @time 2025/4/7 13:28 | 
|      */ | 
|     @Transactional(rollbackFor = Exception.class) | 
|     public void svaeOrUpdateOrderItem(AsnOrderAndItemsParams params, Long loginUserId) throws Exception { | 
|         WkOrder orders = params.getOrders(); | 
|         params.getItems().forEach(item -> { | 
|             item.put("orderId", orders.getId()); | 
|             item.put("orderCode", orders.getCode()); | 
|             item.put("poCode", orders.getPoCode()); | 
|             item.put("createBy", loginUserId); | 
|             item.put("updateBy", loginUserId); | 
|             if (!asnOrderItemService.fieldsSave(item, loginUserId)) { | 
|                 throw new CoolException("明细保存失败!!"); | 
|             } | 
|         }); | 
|         List<WkOrderItem> orderItems = asnOrderItemService.list(new LambdaQueryWrapper<WkOrderItem>() | 
|                 .eq(WkOrderItem::getOrderId, params.getOrders().getId())); | 
|         Double sum = orderItems.stream().mapToDouble(WkOrderItem::getAnfme).sum(); | 
|         orders.setAnfme(sum); | 
|         if (!this.updateById(orders)) { | 
|             throw new CoolException("计划收货数量修改失败!!"); | 
|         } | 
|     } | 
|   | 
|   | 
|     /** | 
|      * @param | 
|      * @return | 
|      * @author Ryan | 
|      * @description 取消出库单据 | 
|      * @time 2025/4/22 10:40 | 
|      */ | 
|     @Override | 
|     @Transactional(rollbackFor = Exception.class) | 
|     public R cancelOutOrder(String id) { | 
|         if (Cools.isEmpty(id)) { | 
|             throw new CoolException("参数不能为空!!"); | 
|         } | 
|         WkOrder order = this.getById(id); | 
|         if (Objects.isNull(order)) { | 
|             throw new CoolException("单据不存在!!"); | 
|         } | 
|         if (!order.getExceStatus().equals(AsnExceStatus.OUT_STOCK_STATUS_TASK_INIT.val)) { | 
|             throw new CoolException("当前单据状态为" + AsnExceStatus.getExceStatus(order.getExceStatus()) + ", 不可执行取消操作!!"); | 
|         } | 
|   | 
|         List<WkOrderItem> orderItems = outStockItemService.list(new LambdaQueryWrapper<WkOrderItem>().eq(WkOrderItem::getOrderId, id)); | 
|         if (!orderItems.isEmpty()) { | 
|             for (WkOrderItem orderItem : orderItems) { | 
|                 if (!Objects.isNull(orderItem.getPoDetlId())) { | 
|                     DeliveryItem deliveryItem = deliveryItemService.getById(orderItem.getPoDetlId()); | 
|                     Double workQty = Math.round((deliveryItem.getWorkQty() - orderItem.getAnfme()) * 10000) / 10000.0; | 
|                     deliveryItem.setWorkQty(workQty.compareTo(0.0) >= 0 ? workQty : 0); | 
|                     if (!deliveryItemService.updateById(deliveryItem)) { | 
|                         throw new CoolException("DO单明细更新失败!!"); | 
|                     } | 
|   | 
|                     Delivery delivery = deliveryService.getOne(new LambdaQueryWrapper<Delivery>().eq(Delivery::getCode, orderItem.getPoCode())); | 
|                     if (!Objects.isNull(delivery)) { | 
|                         Double wkQty = Math.round((delivery.getWorkQty() - delivery.getAnfme()) * 10000) / 10000.0; | 
|                         delivery.setWorkQty(wkQty.compareTo(0.0) >= 0 ? wkQty : 0).setExceStatus(POExceStatus.PO_EXCE_STATUS_UN_EXCE.val); | 
|                         if (!deliveryService.updateById(delivery)) { | 
|                             throw new CoolException("DO单据修改失败!!"); | 
|                         } | 
|                     } | 
|                 } | 
|             } | 
|         } | 
|   | 
|         if (!this.remove(new LambdaQueryWrapper<WkOrder>().eq(WkOrder::getId, id))) { | 
|             throw new CoolException("主单删除失败!!"); | 
|         } | 
|         outStockItemService.remove(new LambdaQueryWrapper<WkOrderItem>().eq(WkOrderItem::getOrderId, id)); | 
|         return R.ok("操作成功"); | 
|     } | 
|   | 
|     /** | 
|      * @param | 
|      * @param loginUserId | 
|      * @return | 
|      * @author Ryan | 
|      * @description 通过DO单生成出库单 | 
|      * @time 2025/4/23 16:24 | 
|      */ | 
|     @Override | 
|     @Transactional(rollbackFor = Exception.class) | 
|     public R genOutStock(List<DeliveryItem> items, Long loginUserId) { | 
|         if (items.isEmpty()) { | 
|             throw new CoolException("参数不能为空!!"); | 
|         } | 
|         Map<Long, List<DeliveryItem>> listMap = items.stream().collect(Collectors.groupingBy(DeliveryItem::getDeliveryId)); | 
|         listMap.keySet().forEach(key -> { | 
|             Delivery delivery = deliveryService.getById(key); | 
|             if (Objects.isNull(delivery)) { | 
|                 throw new CoolException("单据不存在!!"); | 
|             } | 
|             WkOrder order = new WkOrder(); | 
|             BeanUtils.copyProperties(delivery, order); | 
|             String ruleCode = SerialRuleUtils.generateRuleCode(SerialRuleCode.SYS_OUT_STOCK_CODE, order); | 
|             if (StringUtils.isBlank(ruleCode)) { | 
|                 throw new CoolException("编码规则错误:请检查 「SYS_OUT_STOCK_CODE」编码是否设置成功"); | 
|             } | 
|             order.setExceStatus(AsnExceStatus.OUT_STOCK_STATUS_TASK_INIT.val) | 
|                     .setType(delivery.getType()) | 
|                     .setWkType(delivery.getWkType()) | 
|                     .setCode(ruleCode) | 
|                     .setPoId(delivery.getId()) | 
|                     .setId(null) | 
|                     .setUpdateBy(loginUserId) | 
|                     .setCreateBy(loginUserId) | 
|                     .setCreateTime(new Date()) | 
|                     .setUpdateTime(new Date()) | 
|                     .setPoCode(delivery.getCode()); | 
|             if (!this.save(order)) { | 
|                 throw new CoolException("主单保存失败!!"); | 
|             } | 
|             List<WkOrderItem> orderItems = new ArrayList<>(); | 
|             listMap.get(key).forEach(item -> { | 
|                 DeliveryItem deliveryItem = deliveryItemService.getById(item.getId()); | 
|                 WkOrderItem orderItem = new WkOrderItem(); | 
|                 if (item.getAnfme().compareTo(0.0) <= 0) { | 
|                     throw new CoolException("出库数量不能小于或等于零!!"); | 
|                 } | 
|                 Double anfme = Math.round((deliveryItem.getAnfme() - item.getAnfme()) * 10000) / 10000.0; | 
|                 if (anfme.compareTo(0.0) < 0) { | 
|                     throw new CoolException("出库数量不足!!"); | 
|                 } | 
|   | 
|                 BeanUtils.copyProperties(item, orderItem); | 
|                 orderItem.setId(null) | 
|                         .setPoCode(order.getPoCode()) | 
|                         .setMaktx(item.getMaktx()) | 
|                         .setMatnrCode(item.getMatnrCode()) | 
|                         .setFieldsIndex(item.getFieldsIndex()) | 
|                         .setAnfme(item.getAnfme()) | 
|                         .setWorkQty(0.0) | 
|                         .setOrderId(order.getId()) | 
|                         .setOrderCode(order.getCode()) | 
|                         .setStockUnit(item.getUnit()) | 
|                         .setPurUnit(item.getUnit()) | 
|                         .setPlatWorkCode(item.getPlatWorkCode()) | 
|                         .setPlatOrderCode(item.getPlatOrderCode()) | 
|                         .setProjectCode(item.getProjectCode()) | 
|                         .setPlatItemId(item.getPlatItemId()) | 
|                         .setUpdateBy(loginUserId) | 
|                         .setCreateBy(loginUserId) | 
|                         .setPoDetlId(item.getId()); | 
|                 orderItems.add(orderItem); | 
|   | 
|                 if (!deliveryItemService.update(new LambdaUpdateWrapper<DeliveryItem>() | 
|                         .set(DeliveryItem::getWorkQty, item.getAnfme()) | 
|                         .eq(DeliveryItem::getId, item.getId()))) { | 
|                     throw new CoolException("DO单明细修改失败!!"); | 
|                 } | 
|             }); | 
|   | 
|             Double sum = orderItems.stream().mapToDouble(WkOrderItem::getAnfme).sum(); | 
|             //修改计划数量 | 
|             order.setAnfme(sum).setWorkQty(0.0); | 
|             if (!this.saveOrUpdate(order)) { | 
|                 throw new CoolException("主单数量修改失败!!"); | 
|             } | 
|             if (!asnOrderItemService.saveBatch(orderItems)) { | 
|                 throw new CoolException("明细保存失败!!"); | 
|             } | 
|   | 
|             Short exceStatus = POExceStatus.PO_EXCE_STATUS_SECTION_DONE.val; | 
|             if (delivery.getAnfme().compareTo(order.getAnfme()) <= 0) { | 
|                 exceStatus = AsnExceStatus.ASN_EXCE_STATUS_TASK_DONE.val; | 
|             } | 
|   | 
|             Double wkQty = Math.round((delivery.getWorkQty() + sum) * 10000) / 10000.0; | 
|             if (!deliveryService.update(new LambdaUpdateWrapper<Delivery>() | 
|                     .set(Delivery::getExceStatus, exceStatus) | 
|                     .set(Delivery::getWorkQty, wkQty) | 
|                     .eq(Delivery::getId, key))) { | 
|                 throw new CoolException("主单修改失败!!"); | 
|             } | 
|         }); | 
|         return R.ok(); | 
|     } | 
|   | 
|     /** | 
|      * @param | 
|      * @return | 
|      * @author Ryan | 
|      * @description 生成波次 | 
|      * @time 2025/4/24 15:04 | 
|      */ | 
|     @Override | 
|     @Transactional(rollbackFor = Exception.class) | 
|     public R generateWaves(GenWaveParams params) { | 
|         if (Objects.isNull(params.getIds()) || params.getIds().isEmpty()) { | 
|             throw new CoolException("参数不能为空!!"); | 
|         } | 
|         List<WkOrder> orders = this.list(new LambdaQueryWrapper<WkOrder>() | 
|                 .eq(WkOrder::getExceStatus, AsnExceStatus.OUT_STOCK_STATUS_TASK_INIT.val) | 
|                 .in(WkOrder::getId, params.getIds())); | 
|         if (orders.isEmpty()) { | 
|             throw new CoolException("当前单据状态不能执行波次生成操作!!"); | 
|         } | 
|         Double sum = orders.stream().mapToDouble(WkOrder::getAnfme).sum(); | 
|         Double workQty = orders.stream().mapToDouble(WkOrder::getWorkQty).sum(); | 
|         Double anfme = Math.round((sum - workQty) * 10000) / 10000.0; | 
|         Wave wave = new Wave(); | 
|         String ruleCode = SerialRuleUtils.generateRuleCode(SerialRuleCode.SYS_WAVE_TYPE, null); | 
|         if (StringUtils.isBlank(ruleCode)) { | 
|             throw new CoolException("编码规则错误:请要查看「SYS_WAVE_TYPE」是否设置成功!!"); | 
|         } | 
|         wave.setOrderNum(params.getIds().size()) | 
|                 .setType(Short.parseShort("1")) | 
|                 .setCode(ruleCode) | 
|                 .setExceStatus(WaveExceStatus.WAVE_EXCE_STATUS_INIT.val) | 
|                 .setAnfme(anfme); | 
|         if (!waveService.save(wave)) { | 
|             throw new CoolException("波次保存失败!!"); | 
|         } | 
|   | 
|         List<Long> list = orders.stream().map(WkOrder::getId).collect(Collectors.toList()); | 
|         List<WkOrderItem> orderItems = asnOrderItemService | 
|                 .list(new LambdaQueryWrapper<WkOrderItem>() | 
|                         .in(WkOrderItem::getOrderId, list).apply("anfme > work_qty")); | 
|         if (orderItems.isEmpty()) { | 
|             throw new CoolException("单据不存在!!"); | 
|         } | 
|   | 
|         //合并物料,生成波次明细 | 
|         List<WaveItem> waveItems = mergeWave(orderItems, wave); | 
|          | 
|         if (!waveItemService.saveBatch(waveItems)) { | 
|             throw new CoolException("波次明细保存失败!!"); | 
|         } | 
|         double sum1 = waveItems.stream().mapToDouble(WaveItem::getAnfme).sum(); | 
|         wave.setAnfme(sum1).setGroupQty(waveItems.size()); | 
|         if (!waveService.saveOrUpdate(wave)) { | 
|             throw new CoolException("主单修改失败!!"); | 
|         } | 
|         for (int i = 0; i < orderItems.size(); i++) { | 
|             orderItems.get(i).setWorkQty(orderItems.get(i).getAnfme()); | 
|         } | 
|   | 
|         /** | 
|          *订单信息存储至逻辑关联表 | 
|          */ | 
|         for (WaveItem item : waveItems) { | 
|             List<WkOrderItem> items = orderItems.stream() | 
|                     .filter(orderItem -> item.getMatnrId() | 
|                             .equals(orderItem.getMatnrId())) | 
|                     .collect(Collectors.toList()); | 
|   | 
|             items.forEach(orderItem -> { | 
|                 WaveOrderRela orderRela = new WaveOrderRela(); | 
|                 orderRela.setId(null) | 
|                         .setOrderId(orderItem.getOrderId()) | 
|                         .setOrderItemId(orderItem.getId()) | 
|                         .setWaveId(wave.getId()) | 
|                         .setWaveItemId(item.getId()); | 
|                 if (!waveOrderRelaService.saveOrUpdate(orderRela)) { | 
|                     throw new CoolException("<UNK>"); | 
|                 } | 
|             }); | 
|         } | 
|          | 
|          | 
|         if (!asnOrderItemService.saveOrUpdateBatch(orderItems)) { | 
|             throw new CoolException("出库单执行数量修改失败!!"); | 
|         } | 
|         for (WkOrder order : orders) { | 
|             Double wkQty = Math.round((order.getWorkQty() + order.getAnfme()) * 10000) / 10000.0; | 
|             if (!this.update(new LambdaUpdateWrapper<WkOrder>() | 
|                     .set(WkOrder::getWaveId, wave.getId()) | 
|                     .set(WkOrder::getWorkQty, wkQty) | 
|                     .set(WkOrder::getExceStatus, AsnExceStatus.OUT_STOCK_STATUS_TASK_WAVE.val) | 
|                     .eq(WkOrder::getId, order.getId()))) { | 
|                 throw new CoolException("执行状态修改修改失败!!"); | 
|             } | 
|         } | 
|   | 
|         return R.ok("操作完成!!"); | 
|     } | 
|   | 
|     /** | 
|      * @param | 
|      * @return | 
|      * @author Ryan | 
|      * @description 保存出库主单及明细 | 
|      * @time 2025/4/29 13:47 | 
|      */ | 
|     @Override | 
|     @Transactional(rollbackFor = Exception.class) | 
|     public R saveOutStock(AsnOrderAndItemsParams params, Long loginUserId) { | 
|         if (Objects.isNull(params.getOrders())) { | 
|             throw new CoolException("主单信息不能为空"); | 
|         } | 
|         WkOrder orders = params.getOrders(); | 
|         if (StringUtils.isBlank(orders.getWkType())) { | 
|             throw new CoolException("业务类型不能为空!!"); | 
|         } | 
|   | 
|         String ruleCode = SerialRuleUtils.generateRuleCode(SerialRuleCode.SYS_OUT_STOCK_CODE, orders); | 
|         if (StringUtils.isBlank(ruleCode)) { | 
|             throw new CoolException("编码规则错误:请检查「SYS_OUT_STOCK_CODE」是否设置正确!!"); | 
|         } | 
|         orders.setCode(ruleCode) | 
|                 .setType(OrderType.ORDER_OUT.type) | 
|                 .setExceStatus(AsnExceStatus.OUT_STOCK_STATUS_TASK_INIT.val) | 
|                 .setUpdateBy(loginUserId) | 
|                 .setCreateBy(loginUserId); | 
|         if (!this.save(orders)) { | 
|             throw new CoolException("主单保存失败!!"); | 
|         } | 
|         if (params.getItems().isEmpty()) { | 
|             throw new CoolException("收货通知单明细不能为空!!"); | 
|         } | 
|         params.setOrders(orders); | 
|         try { | 
|             svaeOrUpdateOrderItem(params, loginUserId); | 
|         } catch (Exception e) { | 
|             throw new CoolException(e.getMessage()); | 
|         } | 
|         return R.ok(); | 
|     } | 
|   | 
|     /** | 
|      * @param | 
|      * @return | 
|      * @author Ryan | 
|      * @description 修改主单及明细 | 
|      * @time 2025/4/29 13:47 | 
|      */ | 
|     @Override | 
|     public R updateOrderItem(AsnOrderAndItemsParams params, Long loginUserId) { | 
|         WkOrder orders = params.getOrders(); | 
|         if (Objects.isNull(orders)) { | 
|             throw new CoolException("主单信息不能为空!!"); | 
|         } | 
|         if (Objects.isNull(orders.getId())) { | 
|             throw new CoolException("数据错误:单据ID不能为空!!"); | 
|         } | 
|         if (!this.updateById(orders)) { | 
|             throw new CoolException("主单修改失败!!"); | 
|         } | 
|         if (Objects.isNull(params.getItems()) || params.getItems().isEmpty()) { | 
|             throw new CoolException("明细参数不能为空!!"); | 
|         } | 
|         try { | 
|             svaeOrUpdateOrderItem(params, loginUserId); | 
|         } catch (Exception e) { | 
|             throw new CoolException(e.getMessage()); | 
|         } | 
|         return R.ok(); | 
|     } | 
|   | 
|     @Override | 
|     public R getOrderOutTaskItem(OrderOutTaskParam param) { | 
|         if (Cools.isEmpty(param.getWaveId())) { | 
|             throw new CoolException("策略参数为空"); | 
|         } | 
|         if (Cools.isEmpty(param.getOrderId())) { | 
|             throw new CoolException("单据ID为空"); | 
|         } | 
|         WaveRule waveRule = waveRuleService.getOne(new LambdaQueryWrapper<WaveRule>() | 
|                 .eq(WaveRule::getId, param.getWaveId())); | 
|         if (Cools.isEmpty(waveRule)) { | 
|             throw new CoolException("未找到当前策略"); | 
|         } | 
|         List<OrderOutItemDto> locItems = null; | 
|         locItems = getOutOrderList(param.getOrderId(), waveRule); | 
|         return R.ok(locItems); | 
|     } | 
|   | 
|     /** | 
|      * 生成出库任务 | 
|      * | 
|      * @param params | 
|      * @param outId | 
|      * @return | 
|      */ | 
|     @Override | 
|     @Transactional(rollbackFor = Exception.class) | 
|     public R genOutStockTask(List<OutStockToTaskParams> params, Long loginUserId, Long outId) { | 
|         if (params.isEmpty()) { | 
|             throw new CoolException("参数不能为空!!"); | 
|         } | 
|         //优先生成浅库位任务 | 
|         List<OutStockToTaskParams> Items = params.stream() | 
|                 .sorted(Comparator.comparing(OutStockToTaskParams::getLocCode) | 
|                         .thenComparing(item -> { | 
|             return LocUtils.isShallowLoc(item.getLocCode()) ? 1 : 0; | 
|         }).reversed()).collect(Collectors.toList()); | 
|   | 
|         for (OutStockToTaskParams param : Items) { | 
|             if (Objects.isNull(param) || StringUtils.isBlank(param.getLocCode())) { | 
|                 continue; | 
|             } | 
|             Loc loc = locService.getOne(new LambdaQueryWrapper<Loc>().eq(Loc::getCode, param.getLocCode()).eq(Loc::getBarcode, param.getBarcode())); | 
|             if (!Objects.isNull(loc)) { | 
|                 List<LocItem> locItems = new ArrayList<>(); | 
|                 LocItem locItem = locItemService.getById(param.getId()); | 
|   | 
|                 WkOrderItem orderItem = outStockItemService.getOne(new LambdaQueryWrapper<WkOrderItem>() | 
|                         .eq(WkOrderItem::getOrderId, outId) | 
|                         .eq(StringUtils.isNotBlank(locItem.getBatch()), WkOrderItem::getSplrBatch, locItem.getBatch()) | 
|                         .eq(StringUtils.isNotBlank(locItem.getFieldsIndex()), WkOrderItem::getFieldsIndex, locItem.getFieldsIndex()) | 
|                         .eq(WkOrderItem::getMatnrId, locItem.getMatnrId())); | 
|   | 
|                 if (Objects.isNull(orderItem)) { | 
|                     throw new CoolException("单据明细不存在!!"); | 
|                 } | 
|   | 
|                 locItem.setOutQty(param.getOutQty()) | 
|                         .setBatch(param.getBatch()) | 
|                         .setSourceId(outId) | 
|                         .setSourceCode(orderItem.getOrderCode()) | 
|                         .setSource(orderItem.getId()); | 
|                 locItems.add(locItem); | 
|   | 
|                 LocToTaskParams taskParams = new LocToTaskParams(); | 
|                 taskParams.setType(Constants.TASK_TYPE_ORDER_OUT_STOCK) | 
|                         .setOrgLoc(loc.getCode()) | 
|                         .setItems(locItems) | 
|                         .setSourceId(outId) | 
|                         .setSiteNo(param.getSiteNo()); | 
|                 try { | 
|                     //生成出库任务 | 
|                     locItemService.generateTask(TaskResouceType.TASK_RESOUCE_ORDER_TYPE.val, taskParams, loginUserId); | 
|                 } catch (Exception e) { | 
|                     logger.error("UNK", e); | 
|                     throw new CoolException(e.getMessage()); | 
|                 } | 
|   | 
|   | 
|                 Double workQty = Math.round((orderItem.getWorkQty() + locItem.getOutQty()) * 10000) / 10000.0; | 
|   | 
|                 orderItem.setUpdateBy(loginUserId).setUpdateTime(new Date()).setWorkQty(workQty); | 
|   | 
|                 if (!outStockItemService.updateById(orderItem)) { | 
|                     throw new CoolException("单据明细修改失败!!"); | 
|                 } | 
|             } | 
|         } | 
|   | 
|         Double sum = Items.stream().mapToDouble(OutStockToTaskParams::getOutQty).sum(); | 
|         //更新出库单明细及主单 | 
|         WkOrder outOrder = outStockService.getById(outId); | 
|         if (Objects.isNull(outOrder)) { | 
|             throw new CoolException("出库单据不存在!!"); | 
|         } | 
|         Double workQty = Math.round((outOrder.getWorkQty() + sum) * 10000) / 10000.0; | 
|   | 
|         outOrder.setWorkQty(workQty).setExceStatus(AsnExceStatus.OUT_STOCK_STATUS_TASK_CREATE.val); | 
|   | 
|         if (!outStockService.updateById(outOrder)) { | 
|             throw new CoolException("出库单状态修改失败!!"); | 
|         } | 
|   | 
|         return R.ok(); | 
|     } | 
|   | 
|     /** | 
|      * 获取出库站点 | 
|      * | 
|      * @return | 
|      */ | 
|     @Override | 
|     public R getSiteNos() { | 
|         List<Integer> list = Arrays.asList(TaskType.TASK_TYPE_MERGE_OUT.type, | 
|                 TaskType.TASK_TYPE_OUT.type, | 
|                 TaskType.TASK_TYPE_MERGE_OUT.type, | 
|                 TaskType.TASK_TYPE_PICK_AGAIN_OUT.type); | 
|         List<DeviceSite> sites = deviceSiteService.list(new LambdaQueryWrapper<DeviceSite>().in(DeviceSite::getType, list).groupBy(DeviceSite::getSite)); | 
|         return R.ok(sites); | 
|     } | 
|   | 
|     @Override | 
|     @Transactional(rollbackFor = Exception.class) | 
|     public R cancelOutOrderByItems(List<WkOrderItem> orderItems) { | 
|         Map<Long, List<WkOrderItem>> listMap = orderItems.stream().collect(Collectors.groupingBy(WkOrderItem::getOrderId)); | 
|         for (Long key : listMap.keySet()) { | 
|             WkOrder order = this.getById(key); | 
|             if (Objects.isNull(order)) { | 
|                 throw new CoolException("单据不存在!!"); | 
|             } | 
|             List<WkOrderItem> items = listMap.get(key); | 
|             if (!items.isEmpty()) { | 
|                 for (WkOrderItem orderItem : items) { | 
|                     DeliveryItem deliveryItem = deliveryItemService.getById(orderItem.getPoDetlId()); | 
|                     Double workQty = Math.round((deliveryItem.getWorkQty() - orderItem.getAnfme()) * 10000) / 10000.0; | 
|                     deliveryItem.setWorkQty(workQty.compareTo(0.0) >= 0 ? workQty : 0); | 
|                     if (!deliveryItemService.updateById(deliveryItem)) { | 
|                         throw new CoolException("DO单明细更新失败!!"); | 
|                     } | 
|   | 
|                     Delivery delivery = deliveryService.getOne(new LambdaQueryWrapper<Delivery>().eq(Delivery::getCode, orderItem.getPoCode())); | 
|                     if (!Objects.isNull(delivery)) { | 
|                         Double wkQty = Math.round((delivery.getWorkQty() - delivery.getAnfme()) * 10000) / 10000.0; | 
|                         delivery.setWorkQty(wkQty.compareTo(0.0) >= 0 ? wkQty : 0).setExceStatus(POExceStatus.PO_EXCE_STATUS_UN_EXCE.val); | 
|                         if (!deliveryService.updateById(delivery)) { | 
|                             throw new CoolException("DO单据修改失败!!"); | 
|                         } | 
|                     } | 
|                 } | 
|             } | 
|   | 
|             if (!this.remove(new LambdaQueryWrapper<WkOrder>().eq(WkOrder::getId, key))) { | 
|                 throw new CoolException("主单删除失败!!"); | 
|             } | 
|             if (!outStockItemService.remove(new LambdaQueryWrapper<WkOrderItem>() | 
|                     .eq(WkOrderItem::getOrderId, key))) { | 
|                 throw new CoolException("单据明细删除失败!!"); | 
|             } | 
|         } | 
|         return R.ok("操作成功"); | 
|     } | 
|   | 
|   | 
|     private List<OrderOutItemDto> getOutOrderList(Long orderId, WaveRule waveRule) { | 
|         List<WkOrderItem> wkOrderItems = asnOrderItemService.list(new LambdaQueryWrapper<WkOrderItem>() | 
|                 .eq(WkOrderItem::getOrderId, orderId)); | 
|         List<OrderOutItemDto> list = new ArrayList<>(); | 
|         Set<ExistDto> existDtos = new HashSet<>(); | 
|         for (WkOrderItem wkOrderItem : wkOrderItems) { | 
|             BigDecimal issued = new BigDecimal(wkOrderItem.getAnfme().toString()) | 
|                     .subtract(new BigDecimal(wkOrderItem.getWorkQty().toString()) | 
|                     ); | 
|             if (issued.doubleValue() <= 0) { | 
|                 continue; | 
|             } | 
|             List<LocItem> locItems = new ArrayList<>(); | 
|             if (WaveRuleType.Efficiency_First.type.equals(waveRule.getType())) { | 
|                 locItems = LocManageUtil.getEfficiencyFirstItemList(wkOrderItem.getMatnrCode(), wkOrderItem.getSplrBatch(), wkOrderItem.getAnfme()); | 
|             } else if (WaveRuleType.First_In_First_Out.type.equals(waveRule.getType())) { | 
|                 locItems = LocManageUtil.getFirstInFirstOutItemList(wkOrderItem.getMatnrCode(), wkOrderItem.getSplrBatch(), wkOrderItem.getAnfme()); | 
|             } else { | 
|                 locItems = LocManageUtil.getFirstInFirstOutItemList(wkOrderItem.getMatnrCode(), wkOrderItem.getSplrBatch(), wkOrderItem.getAnfme()); | 
|             } | 
|             for (LocItem locItem : locItems) { | 
|                 Loc loc = locService.getById(locItem.getLocId()); | 
|                 List<LocItem> itemList = locItemService.list(new LambdaQueryWrapper<LocItem>().eq(LocItem::getLocCode, locItem.getLocCode())); | 
|                 if (issued.doubleValue() > 0) { | 
|                     ExistDto existDto = new ExistDto().setBatch(locItem.getBatch()).setMatnr(locItem.getMatnrCode()).setLocNo(locItem.getLocCode()); | 
|                     if (existDtos.add(existDto)) { | 
|                         locItem.setOutQty(issued.doubleValue() >= locItem.getAnfme() ? locItem.getAnfme() : issued.doubleValue()); | 
|                         locItem.setBarcode(loc.getBarcode()); | 
|                         OrderOutItemDto orderOutItemDto = new OrderOutItemDto(); | 
|                         orderOutItemDto.setLocItem(locItem); | 
|   | 
|                         List<DeviceSite> deviceSites = deviceSiteService.list(new LambdaQueryWrapper<DeviceSite>() | 
|                                 .eq(DeviceSite::getChannel, loc.getChannel()) | 
|                                 .eq(DeviceSite::getType, issued.doubleValue() >= locItem.getAnfme() && itemList.size() == 1 ? TaskType.TASK_TYPE_OUT.type : TaskType.TASK_TYPE_PICK_AGAIN_OUT.type) | 
|                         ); | 
|   | 
|                         if (!deviceSites.isEmpty()) { | 
|                             List<OrderOutItemDto.staListDto> maps = new ArrayList<>(); | 
|                             for (DeviceSite sta : deviceSites) { | 
|                                 OrderOutItemDto.staListDto staListDto = new OrderOutItemDto.staListDto(); | 
|                                 staListDto.setStaNo(sta.getSite()); | 
|                                 staListDto.setStaName(sta.getSite()); | 
|                                 maps.add(staListDto); | 
|                             } | 
|                             orderOutItemDto.setStaNos(maps); | 
|                             //默认获取第一站点 | 
|                             DeviceSite deviceSite = deviceSites.stream().findFirst().get(); | 
|                             orderOutItemDto.setSiteNo(deviceSite.getSite()); | 
|                         } | 
|   | 
|                         list.add(orderOutItemDto); | 
|   | 
|                         issued = issued.subtract(new BigDecimal(locItem.getAnfme().toString())); | 
|                     } | 
|                 } | 
|             } | 
|             if (issued.doubleValue() > 0) { | 
|                 LocItem locItem = new LocItem() | 
|                         .setId(new Random().nextLong()) | 
|                         .setMatnrCode(wkOrderItem.getMatnrCode()) | 
|                         .setMaktx(wkOrderItem.getMaktx()) | 
|                         .setAnfme(0.00) | 
|                         .setWorkQty(issued.doubleValue()) | 
|                         .setOutQty(issued.doubleValue()) | 
|                         .setUnit(wkOrderItem.getStockUnit()) | 
|                         .setBatch(wkOrderItem.getSplrBatch()); | 
|                 OrderOutItemDto orderOutItemDto = new OrderOutItemDto(); | 
|                 orderOutItemDto.setLocItem(locItem); | 
|                 list.add(orderOutItemDto); | 
|             } | 
|         } | 
|   | 
|         return list; | 
|   | 
|     } | 
|   | 
|     /** | 
|      * @param | 
|      * @param wave | 
|      * @return | 
|      * @author Ryan | 
|      * @description 合并生成波次 | 
|      * @time 2025/4/25 10:07 | 
|      */ | 
|     private List<WaveItem> mergeWave(List<WkOrderItem> orderItems, Wave wave) { | 
|         List<WaveItem> items = new ArrayList<>(); | 
|         orderItems.forEach(order -> { | 
|             Double anfme = Math.round((order.getAnfme() - order.getWorkQty()) * 10000) / 10000.0; | 
|             WaveItem item = new WaveItem(); | 
|             BeanUtils.copyProperties(order, item); | 
|                 item.setId(null) | 
|                     .setAnfme(anfme) | 
|                     .setMatnrId(order.getMatnrId()) | 
|                     .setMaktx(order.getMaktx()) | 
|                     .setWaveId(wave.getId()) | 
|                     .setWaveCode(wave.getCode()); | 
|             items.add(item); | 
|         }); | 
|         List<WaveItem> waveItems = OptimalAlgorithmUtil.groupAndMerge(items, | 
|                 (p1, p2) -> new WaveItem( | 
|                         p1.getWaveId(), | 
|                         p1.getWaveCode(), | 
|                         p1.getMatnrId(), | 
|                         p1.getMaktx(), | 
|                         p1.getMatnrCode(), | 
|                         p1.getBatch(), | 
|                         p1.getSplrBatch(), | 
|                         p1.getOrderCode(), | 
|                         p1.getOrderId(), | 
|                         p1.getOrderItemId(), | 
|                         p1.getUnit(), | 
|                         p1.getTrackCode(), | 
|                         p1.getFieldsIndex(), | 
|                         Math.round((p1.getAnfme() + p2.getAnfme()) * 10000) / 10000.0, | 
|                         p1.getWorkQty(), | 
|                         p1.getTenantId(), | 
|                         p1.getStatus(), | 
|                         p1.getDeleted(), | 
|                         p1.getCreateTime(), | 
|                         p1.getCreateBy(), | 
|                         p1.getUpdateTime(), | 
|                         p1.getUpdateBy(), | 
|                         p1.getMemo() | 
|                 ), | 
|                 WaveItem::getSplrBatch, WaveItem::getMatnrCode, WaveItem::getFieldsIndex | 
|         ); | 
|   | 
|         return waveItems; | 
|     } | 
| } |