package com.ruoyi.quality.service.impl; import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.toolkit.ObjectUtils; import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.deepoove.poi.XWPFTemplate; import com.deepoove.poi.config.Configure; import com.ruoyi.common.enums.StockInQualifiedRecordTypeEnum; import com.ruoyi.common.utils.HackLoopTableRenderPolicy; import com.ruoyi.common.utils.poi.ExcelUtil; import com.ruoyi.procurementrecord.service.ProcurementRecordService; import com.ruoyi.procurementrecord.utils.StockUtils; import com.ruoyi.quality.dto.QualityInspectDto; import com.ruoyi.quality.mapper.QualityInspectMapper; import com.ruoyi.quality.mapper.QualityTestStandardMapper; import com.ruoyi.quality.mapper.QualityUnqualifiedMapper; import com.ruoyi.quality.pojo.QualityInspect; import com.ruoyi.quality.pojo.QualityInspectParam; import com.ruoyi.quality.pojo.QualityUnqualified; import com.ruoyi.quality.service.IQualityInspectParamService; import com.ruoyi.quality.service.IQualityInspectService; import com.ruoyi.purchase.mapper.PurchaseLedgerMapper; import com.ruoyi.purchase.pojo.PurchaseLedger; import com.ruoyi.sales.mapper.SalesLedgerProductMapper; import com.ruoyi.sales.pojo.SalesLedgerProduct; import lombok.AllArgsConstructor; import org.springframework.beans.BeanUtils; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import javax.servlet.http.HttpServletResponse; import java.io.InputStream; import java.io.OutputStream; import java.math.BigDecimal; import java.net.URLEncoder; import java.util.HashMap; import java.util.List; import java.util.Objects; import java.util.stream.Collectors; @AllArgsConstructor @Service @Transactional(rollbackFor = Exception.class) public class QualityInspectServiceImpl extends ServiceImpl implements IQualityInspectService { private final StockUtils stockUtils; private QualityInspectMapper qualityInspectMapper; private IQualityInspectParamService qualityInspectParamService; private QualityTestStandardMapper qualityTestStandardMapper; private QualityUnqualifiedMapper qualityUnqualifiedMapper; private SalesLedgerProductMapper salesLedgerProductMapper; private PurchaseLedgerMapper purchaseLedgerMapper; private ProcurementRecordService procurementRecordService; @Override public int add(QualityInspectDto qualityInspectDto) { QualityInspect qualityInspect = new QualityInspect(); BeanUtils.copyProperties(qualityInspectDto, qualityInspect); qualityInspect.setInspectState(0);//默认未提交 qualityInspectMapper.insert(qualityInspect); for (QualityInspectParam qualityInspectParam : qualityInspectDto.getQualityInspectParams()) { qualityInspectParam.setInspectId(qualityInspect.getId()); } qualityInspectParamService.saveBatch(qualityInspectDto.getQualityInspectParams()); return 0; } @Override public QualityInspectDto getDetailById(Integer id) { QualityInspect qualityInspect = qualityInspectMapper.selectById(id); List qualityInspectParams = qualityInspectParamService.list(Wrappers.lambdaQuery().eq(QualityInspectParam::getInspectId, id)); QualityInspectDto qualityInspectDto = new QualityInspectDto(); BeanUtils.copyProperties(qualityInspect, qualityInspectDto); qualityInspectDto.setQualityInspectParams(qualityInspectParams); return qualityInspectDto; } //提交 @Override public int submit(QualityInspect inspect) { QualityInspect qualityInspect = qualityInspectMapper.selectById(inspect.getId()); //提交前必须判断是否合格 if (ObjectUtils.isNull(qualityInspect.getCheckResult())) { throw new RuntimeException("请先判断是否合格"); } /*判断不合格*/ if (qualityInspect.getCheckResult().equals("不合格")) { QualityUnqualified qualityUnqualified = new QualityUnqualified(); BeanUtils.copyProperties(qualityInspect, qualityUnqualified); qualityUnqualified.setInspectState(0);//待处理 List inspectParams = qualityInspectParamService.list(Wrappers.lambdaQuery().eq(QualityInspectParam::getInspectId, inspect.getId())); String text = inspectParams.stream().map(QualityInspectParam::getParameterItem).collect(Collectors.joining(",")); qualityUnqualified.setDefectivePhenomena(text + "这些指标中存在不合格");//不合格现象 qualityUnqualified.setInspectId(qualityInspect.getId()); qualityUnqualifiedMapper.insert(qualityUnqualified); } else { //合格直接入库 stockUtils.addStock(qualityInspect.getProductModelId(), qualityInspect.getQuantity(), StockInQualifiedRecordTypeEnum.QUALITYINSPECT_STOCK_IN.getCode(), qualityInspect.getId()); // 采购原材料检验:合格入库后同步到采购产品“已入库数量”,与扫码入库共用一份数据口径 syncQualifiedInboundToPurchaseProducts(qualityInspect); } qualityInspect.setInspectState(1);//已提交 int updated = qualityInspectMapper.updateById(qualityInspect); refreshPurchaseLedgerStockStatusByInspect(qualityInspect.getPurchaseLedgerId()); return updated; } /*生成检验报告*/ @Override public void down(HttpServletResponse response, QualityInspect qualityInspect) { QualityInspect inspect = qualityInspectMapper.selectById(qualityInspect.getId()); String inspectType = ""; switch (inspect.getInspectType()) { case 0: inspectType = "原材料检验"; break; case 1: inspectType = "过程检验"; break; case 2: inspectType = "出厂检验"; break; } List paramList = qualityInspectParamService.list(Wrappers.lambdaQuery().eq(QualityInspectParam::getInspectId, inspect.getId())); int index = 1; for (QualityInspectParam detail : paramList) { detail.setIndex(index); index++; } InputStream inputStream = this.getClass().getResourceAsStream("/static/report-template.docx"); Configure configure = Configure.builder() .bind("paramList", new HackLoopTableRenderPolicy()) .build(); String finalInspectType = inspectType; XWPFTemplate template = XWPFTemplate.compile(inputStream, configure).render( new HashMap() {{ put("inspect", inspect); put("inspectType", finalInspectType); put("paramList", paramList); }}); try { response.setContentType("application/msword"); String fileName = URLEncoder.encode( "检验报告", "UTF-8"); response.setHeader("Access-Control-Expose-Headers", "Content-Disposition"); response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".docx"); OutputStream os = response.getOutputStream(); template.write(os); os.flush(); os.close(); inputStream.close(); } catch (Exception e) { e.printStackTrace(); throw new RuntimeException("导出失败"); } } @Override public int updateQualityInspect(QualityInspectDto qualityInspectDto) { if (ObjectUtils.isNotNull(qualityInspectDto.getQualityInspectParams())) { qualityInspectParamService.remove(Wrappers.lambdaQuery().eq(QualityInspectParam::getInspectId, qualityInspectDto.getId())); for (QualityInspectParam qualityInspectParam : qualityInspectDto.getQualityInspectParams()) { qualityInspectParam.setInspectId(qualityInspectDto.getId()); } qualityInspectParamService.saveBatch(qualityInspectDto.getQualityInspectParams()); } QualityInspect qualityInspect = new QualityInspect(); BeanUtils.copyProperties(qualityInspectDto, qualityInspect); return qualityInspectMapper.updateById(qualityInspect); } @Override public IPage qualityInspectListPage(Page page, QualityInspect qualityInspect) { return qualityInspectMapper.qualityInspectListPage(page, qualityInspect); } @Override public void qualityInspectExport(HttpServletResponse response, QualityInspect qualityInspect) { List qualityInspects = qualityInspectMapper.qualityInspectExport(qualityInspect); ExcelUtil util = new ExcelUtil(QualityInspect.class); switch (qualityInspect.getInspectType()) { case 0: util.exportExcel(response, qualityInspects, "原材料检验导出"); break; case 1: util.exportExcel(response, qualityInspects, "过程检验导出"); break; case 2: util.exportExcel(response, qualityInspects, "出厂检验导出"); break; } } private void refreshPurchaseLedgerStockStatusByInspect(Long purchaseLedgerId) { if (purchaseLedgerId == null) { return; } List products = salesLedgerProductMapper.selectList(new LambdaQueryWrapper() .eq(SalesLedgerProduct::getSalesLedgerId, purchaseLedgerId) .eq(SalesLedgerProduct::getType, 2)); if (products == null || products.isEmpty()) { return; } boolean allInbound = true; boolean anyInbound = false; for (SalesLedgerProduct product : products) { BigDecimal orderQty = product.getQuantity() == null ? BigDecimal.ZERO : product.getQuantity(); BigDecimal totalInboundQty = product.getStockedQuantity() == null ? BigDecimal.ZERO : product.getStockedQuantity(); if (totalInboundQty.compareTo(BigDecimal.ZERO) > 0) { anyInbound = true; } if (totalInboundQty.compareTo(orderQty) < 0) { allInbound = false; } } PurchaseLedger purchaseLedger = purchaseLedgerMapper.selectById(purchaseLedgerId); if (purchaseLedger == null) { return; } int targetStockStatus = allInbound ? 2 : (anyInbound ? 1 : 0); if (!Objects.equals(purchaseLedger.getStockStatus(), targetStockStatus)) { purchaseLedger.setStockStatus(targetStockStatus); purchaseLedgerMapper.updateById(purchaseLedger); } } private void syncQualifiedInboundToPurchaseProducts(QualityInspect qualityInspect) { if (qualityInspect == null) { return; } if (!Objects.equals(qualityInspect.getInspectType(), 0) || qualityInspect.getPurchaseLedgerId() == null) { return; } if (qualityInspect.getProductModelId() == null || qualityInspect.getQuantity() == null) { return; } BigDecimal inboundQty = qualityInspect.getQuantity(); if (inboundQty.compareTo(BigDecimal.ZERO) <= 0) { return; } List lines = salesLedgerProductMapper.selectList(new LambdaQueryWrapper() .eq(SalesLedgerProduct::getSalesLedgerId, qualityInspect.getPurchaseLedgerId()) .eq(SalesLedgerProduct::getType, 2) .eq(SalesLedgerProduct::getProductModelId, qualityInspect.getProductModelId()) .eq(SalesLedgerProduct::getIsChecked, true) .orderByAsc(SalesLedgerProduct::getId)); if (lines == null || lines.isEmpty()) { return; } BigDecimal remaining = inboundQty; SalesLedgerProduct fallbackLine = null; for (SalesLedgerProduct line : lines) { if (remaining.compareTo(BigDecimal.ZERO) <= 0) { break; } BigDecimal orderQty = line.getQuantity() == null ? BigDecimal.ZERO : line.getQuantity(); BigDecimal stocked = line.getStockedQuantity() == null ? BigDecimal.ZERO : line.getStockedQuantity(); BigDecimal canFill = orderQty.subtract(stocked); if (canFill.compareTo(BigDecimal.ZERO) <= 0) { fallbackLine = line; continue; } BigDecimal add = canFill.min(remaining); BigDecimal newStocked = stocked.add(add); int status; if (newStocked.compareTo(BigDecimal.ZERO) <= 0) { status = 0; } else if (orderQty.compareTo(BigDecimal.ZERO) > 0 && newStocked.compareTo(orderQty) < 0) { status = 1; } else { status = 2; } line.setStockedQuantity(newStocked); line.setProductStockStatus(status); line.fillRemainingQuantity(); salesLedgerProductMapper.updateById(line); remaining = remaining.subtract(add); fallbackLine = line; } // 允许多入库:若仍有剩余,累计到最后一行,确保 remaining_shipped_quantity 能同步增长 if (remaining.compareTo(BigDecimal.ZERO) > 0 && fallbackLine != null) { BigDecimal orderQty = fallbackLine.getQuantity() == null ? BigDecimal.ZERO : fallbackLine.getQuantity(); BigDecimal stocked = fallbackLine.getStockedQuantity() == null ? BigDecimal.ZERO : fallbackLine.getStockedQuantity(); BigDecimal newStocked = stocked.add(remaining); int status; if (newStocked.compareTo(BigDecimal.ZERO) <= 0) { status = 0; } else if (orderQty.compareTo(BigDecimal.ZERO) > 0 && newStocked.compareTo(orderQty) < 0) { status = 1; } else { status = 2; } fallbackLine.setStockedQuantity(newStocked); fallbackLine.setProductStockStatus(status); fallbackLine.fillRemainingQuantity(); salesLedgerProductMapper.updateById(fallbackLine); } } }