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.manager.enums.OrderType; import com.vincent.rsf.server.manager.controller.params.AsnOrderAndItemsParams; import com.vincent.rsf.server.manager.entity.*; import com.vincent.rsf.server.manager.enums.AsnExceStatus; import com.vincent.rsf.server.manager.enums.POExceStatus; import com.vincent.rsf.server.manager.enums.WaveExceStatus; import com.vincent.rsf.server.manager.mapper.AsnOrderMapper; import com.vincent.rsf.server.manager.service.*; 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.springframework.beans.BeanUtils; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import java.util.*; import java.util.stream.Collectors; /** * @author Ryan * @description * @throws * @return * @time 2025/3/7 08:02 */ @Service("outStockServiceImpl") public class OutStockServiceImpl extends ServiceImpl implements OutStockService { @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; /** * @param * @return * @author Ryan * @description 更新或保存明细 * @time 2025/4/7 13:28 */ @Transactional(rollbackFor = Exception.class) public void svaeOrUpdateOrderItem(AsnOrderAndItemsParams params, Long loginUserId) throws Exception { AsnOrder orders = params.getOrders(); params.getItems().forEach(item -> { item.put("asnId", orders.getId()); item.put("asnCode", orders.getCode()); item.put("poCode", orders.getPoCode()); item.put("createBy", loginUserId); item.put("updateBy", loginUserId); if (!asnOrderItemService.fieldsSave(item, loginUserId)) { throw new CoolException("明细保存失败!!"); } }); List orderItems = asnOrderItemService.list(new LambdaQueryWrapper() .eq(AsnOrderItem::getAsnId, params.getOrders().getId())); Double sum = orderItems.stream().mapToDouble(AsnOrderItem::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) { //TODO 出库单取消流程,QMS(单据取消)->DO单->出库单->波次->判断是否全单据->全单据下发取消任务至WCS,非全单数据取消删除流程所有关联数据 if (Cools.isEmpty(id)) { throw new CoolException("参数不能为空!!"); } AsnOrder 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 orderItems = outStockItemService.list(new LambdaQueryWrapper().eq(AsnOrderItem::getAsnId, id)); if (!orderItems.isEmpty()) { for (AsnOrderItem 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().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().eq(AsnOrder::getId, id))) { throw new CoolException("主单删除失败!!"); } if (!outStockItemService.remove(new LambdaQueryWrapper().eq(AsnOrderItem::getAsnId, id))) { throw new CoolException("单据明细删除失败!!"); } 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 ids, Long loginUserId) { if (Objects.isNull(ids) || ids.isEmpty()) { throw new CoolException("参数不能为空!!"); } List items = deliveryItemService.list(new LambdaQueryWrapper().in(DeliveryItem::getId, ids)); if (items.isEmpty()) { throw new CoolException("单据不存在!!"); } Map> listMap = items.stream().collect(Collectors.groupingBy(DeliveryItem::getDeliveryId)); //获取第一个ID Long deliveryId = items.stream().findFirst().get().getDeliveryId(); Delivery delivery = deliveryService.getById(deliveryId); if (Objects.isNull(delivery)) { throw new CoolException("主单据不存在!!"); } AsnOrder order = new AsnOrder(); BeanUtils.copyProperties(delivery, order); String ruleCode = SerialRuleUtils.generateRuleCode(SerialRuleCode.SYS_OUT_STOCK_CODE, order); if (Objects.isNull(ruleCode) || 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) .setId(null) .setUpdateBy(loginUserId) .setCreateBy(loginUserId); if (!this.save(order)) { throw new CoolException("主单保存失败!!"); } listMap.keySet().forEach(key -> { Delivery delivery1 = deliveryService.getById(key); List orderItems = new ArrayList<>(); listMap.get(key).forEach(item -> { AsnOrderItem orderItem = new AsnOrderItem(); Double anfme = Math.round((item.getAnfme() - item.getWorkQty() - item.getQty()) * 10000) / 10000.0; BeanUtils.copyProperties(item, orderItem); orderItem.setId(null) .setPoCode(delivery1.getCode()) .setMaktx(item.getMaktx()) .setMatnrCode(item.getMatnrCode()) .setFieldsIndex(item.getFieldsIndex()) .setAnfme(anfme) .setWorkQty(0.0) .setAsnId(order.getId()) .setAsnCode(order.getCode()) .setStockUnit(item.getUnit()) .setPurUnit(item.getUnit()) .setPlatWorkCode(item.getPlatWorkCode()) .setPlatOrderCode(item.getPlatOrderCode()) .setProjectCode(item.getProjectCode()) .setPlatItemId(item.getPlatItemId()) .setFieldsIndex(item.getFieldsIndex()) .setUpdateBy(loginUserId) .setCreateBy(loginUserId) .setPoDetlId(item.getId()); orderItems.add(orderItem); if (!deliveryItemService.update(new LambdaUpdateWrapper() .set(DeliveryItem::getWorkQty, item.getAnfme()) .eq(DeliveryItem::getId, item.getId()))) { throw new CoolException("DO单明细修改失败!!"); } }); Double sum = orderItems.stream().mapToDouble(AsnOrderItem::getAnfme).sum(); if (!asnOrderItemService.saveBatch(orderItems)) { throw new CoolException("明细保存失败!!"); } Short exceStatus = POExceStatus.PO_EXCE_STATUS_SECTION_DONE.val; if (delivery1.getAnfme().compareTo(sum) <= 0) { exceStatus = AsnExceStatus.ASN_EXCE_STATUS_TASK_DONE.val; } if (!deliveryService.update(new LambdaUpdateWrapper() .set(Delivery::getExceStatus, exceStatus) .set(Delivery::getWorkQty, sum) .eq(Delivery::getId, key))) { throw new CoolException("主单修改失败!!"); } }); Double sum = items.stream().mapToDouble(DeliveryItem::getAnfme).sum(); //修改计划数量 order.setAnfme(sum).setWorkQty(0.0); if (!this.saveOrUpdate(order)) { 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(List ids) { if (Objects.isNull(ids) || ids.isEmpty()) { throw new CoolException("参数不能为空!!"); } List orders = this.list(new LambdaQueryWrapper() .in(AsnOrder::getId, ids) .eq(AsnOrder::getExceStatus, AsnExceStatus.OUT_STOCK_STATUS_TASK_INIT.val)); if (orders.isEmpty()) { throw new CoolException("当前单据状态不能执行波次生成操作!!"); } double sum = orders.stream().mapToDouble(AsnOrder::getAnfme).sum(); Wave wave = new Wave(); String ruleCode = SerialRuleUtils.generateRuleCode(SerialRuleCode.SYS_WAVE_TYPE, null); if (Objects.isNull(ruleCode) || StringUtils.isBlank(ruleCode)) { throw new CoolException("编码规则错误:请要查看「SYS_WAVE_TYPE」是否设置成功!!"); } wave.setOrderNum(ids.size()) .setType(Short.parseShort("1")) .setCode(ruleCode) .setExceStatus(WaveExceStatus.WAVE_EXCE_STATUS_INIT.val) .setAnfme(sum); if (!waveService.save(wave)) { throw new CoolException("波次保存失败!!"); } List list = orders.stream().map(AsnOrder::getId).collect(Collectors.toList()); List orderItems = asnOrderItemService .list(new LambdaQueryWrapper() .in(AsnOrderItem::getAsnId, list)); if (orderItems.isEmpty()) { throw new CoolException("单据不存在!!"); } List waveItems = mergeWave(orderItems, wave); if (!waveItemService.saveBatch(waveItems)) { throw new CoolException("波次明细保存失败!!"); } double sum1 = waveItems.stream().mapToDouble(WaveItem::getAnfme).sum(); wave.setAnfme(sum1); if (!waveService.saveOrUpdate(wave)) { throw new CoolException("主单修改失败!!"); } for (int i = 0; i < orderItems.size(); i++) { orderItems.get(i).setWorkQty(orderItems.get(i).getAnfme()); } if (!asnOrderItemService.saveOrUpdateBatch(orderItems)) { throw new CoolException("出库单执行数量修改失败!!"); } double sum2 = orderItems.stream().mapToDouble(AsnOrderItem::getWorkQty).sum(); if (!this.update(new LambdaUpdateWrapper() .set(AsnOrder::getWaveId, wave.getId()) .set(AsnOrder::getWorkQty, sum2) .set(AsnOrder::getExceStatus, AsnExceStatus.OUT_STOCK_STATUS_TASK_WAVE.val) .in(AsnOrder::getId, ids))) { 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("主单信息不能为空"); } AsnOrder orders = params.getOrders(); if (Objects.isNull(orders)) { throw new CoolException("单据不能为空!!"); } if (StringUtils.isBlank(orders.getWkType())) { throw new CoolException("业务类型不能为空!!"); } String ruleCode = SerialRuleUtils.generateRuleCode(SerialRuleCode.SYS_OUT_STOCK_CODE, orders); if (Objects.isNull(ruleCode) || 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) { AsnOrder 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(); } /** * @param * @param wave * @return * @author Ryan * @description 合并生成波次 * @time 2025/4/25 10:07 */ private List mergeWave(List orderItems, Wave wave) { List items = new ArrayList<>(); orderItems.forEach(order -> { WaveItem item = new WaveItem(); BeanUtils.copyProperties(order, item); item.setOrderItemId(order.getId()) .setId(null) .setOrderCode(order.getAsnCode()) .setOrderId(order.getAsnId()) .setMatnrId(order.getMatnrId()) .setMaktx(order.getMaktx()) .setWaveId(wave.getId()) .setWaveCode(wave.getCode()); items.add(item); }); List 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; } }