package com.ruoyi.sales.service.impl; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.ruoyi.approve.mapper.ApproveProcessMapper; import com.ruoyi.approve.pojo.ApproveProcess; import com.ruoyi.approve.service.impl.ApproveProcessServiceImpl; import com.ruoyi.common.enums.FileNameType; import com.ruoyi.common.enums.StockOutQualifiedRecordTypeEnum; import com.ruoyi.common.utils.SecurityUtils; import com.ruoyi.framework.security.LoginUser; import com.ruoyi.other.service.impl.TempFileServiceImpl; import com.ruoyi.procurementrecord.utils.StockUtils; import com.ruoyi.project.system.domain.SysDept; import com.ruoyi.project.system.mapper.SysDeptMapper; import com.ruoyi.project.system.mapper.SysUserMapper; import com.ruoyi.sales.dto.SalesLedgerProductDto; import com.ruoyi.sales.dto.ShippingInfoDto; import com.ruoyi.sales.mapper.SalesLedgerMapper; import com.ruoyi.sales.mapper.SalesLedgerProductMapper; import com.ruoyi.sales.mapper.ShipmentApprovalMapper; import com.ruoyi.sales.mapper.ShippingInfoMapper; import com.ruoyi.sales.pojo.SalesLedger; import com.ruoyi.sales.pojo.SalesLedgerProduct; import com.ruoyi.sales.pojo.ShipmentApproval; import com.ruoyi.sales.pojo.ShippingInfo; import com.ruoyi.sales.service.ShippingInfoService; import lombok.extern.slf4j.Slf4j; import org.apache.commons.collections4.CollectionUtils; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import java.io.IOException; import java.time.LocalDate; import java.time.LocalDateTime; import java.util.ArrayList; import java.util.Collections; import java.util.Date; import java.util.List; /** * @author :yys * @date : 2025/10/22 9:33 */ @Service @Slf4j public class ShippingInfoServiceImpl extends ServiceImpl implements ShippingInfoService { @Autowired private ShippingInfoMapper shippingInfoMapper; @Autowired private TempFileServiceImpl tempFileService; @Autowired private SalesLedgerProductMapper salesLedgerProductMapper; @Autowired private StockUtils stockUtils; @Autowired private CommonFileServiceImpl commonFileService; @Autowired private ApproveProcessServiceImpl approveProcessService; @Autowired private ApproveProcessMapper approveProcessMapper; @Autowired private ShipmentApprovalMapper shipmentApprovalMapper; @Autowired private SalesLedgerMapper salesLedgerMapper; @Autowired private SysUserMapper sysUserMapper; @Autowired private SysDeptMapper sysDeptMapper; @Override public IPage listPage(Page page, ShippingInfo req) { IPage listPage = shippingInfoMapper.listPage(page, req); listPage.getRecords().forEach(item ->{ item.setCommonFileList(commonFileService.getFileListByBusinessId(item.getId(), FileNameType.SHIP.getValue())); }); return listPage; } @Override public boolean deductStock(ShippingInfoDto req) throws IOException { ShippingInfo byId = this.getById(req.getId()); if (byId == null) { throw new RuntimeException("发货信息不存在"); } //扣减库存 if(!"已发货".equals(byId.getStatus())){ SalesLedgerProduct salesLedgerProduct = salesLedgerProductMapper.selectById(byId.getSalesLedgerProductId()); stockUtils.substractStock(salesLedgerProduct.getProductModelId(), salesLedgerProduct.getQuantity(), StockOutQualifiedRecordTypeEnum.SALE_SHIP_STOCK_OUT.getCode(), req.getId()); } byId.setExpressNumber(req.getExpressNumber()); byId.setExpressCompany(req.getExpressCompany()); byId.setStatus("已发货"); byId.setShippingCarNumber(req.getShippingCarNumber()); byId.setShippingDate(req.getShippingDate()); boolean update = this.updateById(byId); // 迁移文件 if(CollectionUtils.isNotEmpty(req.getTempFileIds())){ tempFileService.migrateTempFilesToFormal(req.getId(), req.getTempFileIds(), FileNameType.SHIP.getValue()); } return update ; } @Override public boolean delete(List ids) { List shippingInfos = shippingInfoMapper.selectList(new LambdaQueryWrapper() .in(ShippingInfo::getId, ids)); if(CollectionUtils.isEmpty(shippingInfos)) return false; // 删除附件 commonFileService.deleteByBusinessIds(ids, FileNameType.SHIP.getValue()); // 扣已发货库存 for (ShippingInfo shippingInfo : shippingInfos) { if("已发货".equals(shippingInfo.getStatus())) { stockUtils.deleteStockOutRecord(shippingInfo.getId(), StockOutQualifiedRecordTypeEnum.SALE_SHIP_STOCK_OUT.getCode()); } } // 删除发货审批 if(CollectionUtils.isNotEmpty(shippingInfos)){ for (ShippingInfo shippingInfo : shippingInfos){ ApproveProcess one = approveProcessService.getOne(new LambdaQueryWrapper() .like(ApproveProcess::getApproveReason, shippingInfo.getShippingNo())); if(one != null){ approveProcessService.delByIds(Collections.singletonList(one.getId())); } } } return this.removeBatchByIds(ids); } @Override public List getReturnManagementDtoById(Long shippingId) { return shippingInfoMapper.getReturnManagementDtoById(shippingId ); } @Override public List getShippingInfoByCustomerName(String customerName) { return shippingInfoMapper.getShippingInfoByCustomerName(customerName); } /** * 一键发货 - 自动审批通过并出库 * 创建审批记录,批准人为"月光" */ @Override @Transactional public boolean oneClickShipping(ShippingInfoDto req) throws IOException { LoginUser loginUser = SecurityUtils.getLoginUser(); // 获取销售台账产品信息 SalesLedgerProduct salesLedgerProduct = salesLedgerProductMapper.selectById(req.getSalesLedgerProductId()); if (salesLedgerProduct == null) { throw new RuntimeException("销售产品不存在"); } // 获取销售台账信息 SalesLedger salesLedger = salesLedgerMapper.selectById(req.getSalesLedgerId()); // 查询"月光"用户 com.ruoyi.project.system.domain.SysUser moonlightUser = sysUserMapper.selectOne( new LambdaQueryWrapper() .eq(com.ruoyi.project.system.domain.SysUser::getNickName, "月光") .last("limit 1") ); // 生成发货单号 String shippingNo = com.ruoyi.common.utils.OrderUtils.countTodayByCreateTime(shippingInfoMapper, "SH", "shipping_no"); // 创建发货记录,状态直接为"已发货" ShippingInfo shippingInfo = new ShippingInfo(); shippingInfo.setShippingNo(shippingNo); shippingInfo.setSalesLedgerId(req.getSalesLedgerId()); shippingInfo.setSalesLedgerProductId(req.getSalesLedgerProductId()); shippingInfo.setType(req.getType()); shippingInfo.setShippingDate(req.getShippingDate()); shippingInfo.setShippingCarNumber(req.getShippingCarNumber()); shippingInfo.setExpressCompany(req.getExpressCompany()); shippingInfo.setExpressNumber(req.getExpressNumber()); shippingInfo.setStatus("已发货"); boolean save = this.save(shippingInfo); if (save) { // 创建审批流程记录,状态为审核完成(2),批准人为"月光" String salesContractNo = salesLedger != null ? salesLedger.getSalesContractNo() : ""; StringBuilder approveReason = new StringBuilder(); approveReason.append("发货单号:").append(shippingNo); if (salesContractNo != null && !salesContractNo.isEmpty()) { approveReason.append("\n销售单号:").append(salesContractNo); } approveReason.append("\n").append(req.getType()); if ("货车".equals(req.getType()) && req.getShippingCarNumber() != null && !req.getShippingCarNumber().isEmpty()) { approveReason.append(":").append(req.getShippingCarNumber()); } else if ("快递".equals(req.getType()) && req.getExpressCompany() != null && !req.getExpressCompany().isEmpty()) { approveReason.append(":").append(req.getExpressCompany()); } ApproveProcess approveProcess = new ApproveProcess(); String approveId = com.ruoyi.common.utils.OrderUtils.countTodayByCreateTime(approveProcessMapper, "", "approve_id"); approveProcess.setApproveId(approveId); approveProcess.setApproveUser(loginUser.getUserId()); approveProcess.setApproveUserName(loginUser.getNickName()); approveProcess.setApproveDeptId(loginUser.getCurrentDeptId()); SysDept sysDept = sysDeptMapper.selectDeptById(loginUser.getCurrentDeptId()); approveProcess.setApproveDeptName(sysDept.getDeptName()); approveProcess.setApproveReason(approveReason.toString()); approveProcess.setApproveUserIds(moonlightUser != null ? String.valueOf(moonlightUser.getUserId()) : ""); approveProcess.setApproveUserNames(moonlightUser != null ? moonlightUser.getNickName() : "月光"); approveProcess.setApproveUserCurrentId(moonlightUser != null ? moonlightUser.getUserId() : null); approveProcess.setApproveUserCurrentName(moonlightUser != null ? moonlightUser.getNickName() : "月光"); approveProcess.setApproveStatus(2); // 审核完成 approveProcess.setApproveType(7); // 发货审批 approveProcess.setApproveDelete(0); approveProcess.setApproveTime(new Date()); // 申请时间 approveProcess.setTenantId(loginUser.getCurrentDeptId()); approveProcess.setCreateTime(LocalDateTime.now()); approveProcess.setEndDate(new Date()); approveProcessService.save(approveProcess); // 直接扣减库存 stockUtils.substractStock( salesLedgerProduct.getProductModelId(), salesLedgerProduct.getQuantity(), StockOutQualifiedRecordTypeEnum.SALE_SHIP_STOCK_OUT.getCode(), shippingInfo.getId() ); // 迁移附件 if (CollectionUtils.isNotEmpty(req.getTempFileIds())) { tempFileService.migrateTempFilesToFormal(shippingInfo.getId(), req.getTempFileIds(), FileNameType.SHIP.getValue()); } } return save; } /** * 批量一键发货 - 将销售台账下所有未发货的产品全部发货 * 创建审批记录,批准人为"月光" */ @Override @Transactional public boolean batchOneClickShipping(Long salesLedgerId, ShippingInfoDto req) throws IOException { LoginUser loginUser = SecurityUtils.getLoginUser(); // 查询"月光"用户 com.ruoyi.project.system.domain.SysUser moonlightUser = sysUserMapper.selectOne( new LambdaQueryWrapper() .eq(com.ruoyi.project.system.domain.SysUser::getNickName, "月光") .last("limit 1") ); // 获取销售台账信息 SalesLedger salesLedger = salesLedgerMapper.selectById(salesLedgerId); String salesContractNo = salesLedger != null ? salesLedger.getSalesContractNo() : ""; // 查询该销售台账下所有产品 List products = salesLedgerProductMapper.selectList( new LambdaQueryWrapper() .eq(SalesLedgerProduct::getSalesLedgerId, salesLedgerId) .eq(SalesLedgerProduct::getType, 1) // 销售类型 ); if (CollectionUtils.isEmpty(products)) { throw new RuntimeException("该销售台账下没有产品"); } // 过滤出未发货的产品(没有发货记录或发货状态不是"已发货") List unshippedProducts = new ArrayList<>(); for (SalesLedgerProduct product : products) { ShippingInfo existingShipping = shippingInfoMapper.selectOne( new LambdaQueryWrapper() .eq(ShippingInfo::getSalesLedgerProductId, product.getId()) .eq(ShippingInfo::getStatus, "已发货") .last("limit 1") ); if (existingShipping == null) { unshippedProducts.add(product); } } if (CollectionUtils.isEmpty(unshippedProducts)) { throw new RuntimeException("该销售台账下所有产品已发货"); } // 收集所有发货单号用于合并审批 StringBuilder allShippingNos = new StringBuilder(); for (SalesLedgerProduct product : unshippedProducts) { // 生成发货单号 String shippingNo = com.ruoyi.common.utils.OrderUtils.countTodayByCreateTime(shippingInfoMapper, "SH", "shipping_no"); // 创建发货记录 ShippingInfo shippingInfo = new ShippingInfo(); shippingInfo.setShippingNo(shippingNo); shippingInfo.setSalesLedgerId(salesLedgerId); shippingInfo.setSalesLedgerProductId(product.getId()); shippingInfo.setType(req.getType()); shippingInfo.setShippingDate(req.getShippingDate()); shippingInfo.setShippingCarNumber(req.getShippingCarNumber()); shippingInfo.setExpressCompany(req.getExpressCompany()); shippingInfo.setExpressNumber(req.getExpressNumber()); shippingInfo.setStatus("已发货"); boolean save = this.save(shippingInfo); if (save) { // 收集发货单号 if (allShippingNos.length() > 0) { allShippingNos.append("\n"); } allShippingNos.append("发货单号:").append(shippingNo); // 扣减库存 stockUtils.substractStock( product.getProductModelId(), product.getQuantity(), StockOutQualifiedRecordTypeEnum.SALE_SHIP_STOCK_OUT.getCode(), shippingInfo.getId() ); } } // 创建合并的审批流程记录 if (allShippingNos.length() > 0) { StringBuilder approveReason = new StringBuilder(); approveReason.append(allShippingNos.toString()); if (salesContractNo != null && !salesContractNo.isEmpty()) { approveReason.append("\n销售单号:").append(salesContractNo); } approveReason.append("\n").append(req.getType()); if ("货车".equals(req.getType()) && req.getShippingCarNumber() != null && !req.getShippingCarNumber().isEmpty()) { approveReason.append(":").append(req.getShippingCarNumber()); } else if ("快递".equals(req.getType()) && req.getExpressCompany() != null && !req.getExpressCompany().isEmpty()) { approveReason.append(":").append(req.getExpressCompany()); } ApproveProcess approveProcess = new ApproveProcess(); String approveId = com.ruoyi.common.utils.OrderUtils.countTodayByCreateTime(approveProcessMapper, "", "approve_id"); approveProcess.setApproveId(approveId); approveProcess.setApproveUser(loginUser.getUserId()); approveProcess.setApproveUserName(loginUser.getNickName()); approveProcess.setApproveDeptId(loginUser.getCurrentDeptId()); SysDept sysDept = sysDeptMapper.selectDeptById(loginUser.getCurrentDeptId()); approveProcess.setApproveDeptName(sysDept.getDeptName()); approveProcess.setApproveReason(approveReason.toString()); approveProcess.setApproveUserIds(moonlightUser != null ? String.valueOf(moonlightUser.getUserId()) : ""); approveProcess.setApproveUserNames(moonlightUser != null ? moonlightUser.getNickName() : "月光"); approveProcess.setApproveUserCurrentId(moonlightUser != null ? moonlightUser.getUserId() : null); approveProcess.setApproveUserCurrentName(moonlightUser != null ? moonlightUser.getNickName() : "月光"); approveProcess.setApproveStatus(2); // 审核完成 approveProcess.setApproveType(7); // 发货审批 approveProcess.setApproveDelete(0); approveProcess.setApproveTime(new Date()); // 申请时间 approveProcess.setTenantId(loginUser.getCurrentDeptId()); approveProcess.setCreateTime(LocalDateTime.now()); approveProcess.setEndDate(new Date()); approveProcessService.save(approveProcess); } return true; } }