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<QualityInspectMapper, QualityInspect> 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<QualityInspectParam> qualityInspectParams = qualityInspectParamService.list(Wrappers.<QualityInspectParam>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<QualityInspectParam> inspectParams = qualityInspectParamService.list(Wrappers.<QualityInspectParam>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<QualityInspectParam> paramList = qualityInspectParamService.list(Wrappers.<QualityInspectParam>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<String, Object>() {{
|
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.<QualityInspectParam>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<QualityInspect> qualityInspectListPage(Page page, QualityInspect qualityInspect) {
|
return qualityInspectMapper.qualityInspectListPage(page, qualityInspect);
|
}
|
|
@Override
|
public void qualityInspectExport(HttpServletResponse response, QualityInspect qualityInspect) {
|
List<QualityInspect> qualityInspects = qualityInspectMapper.qualityInspectExport(qualityInspect);
|
ExcelUtil<QualityInspect> util = new ExcelUtil<QualityInspect>(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<SalesLedgerProduct> products = salesLedgerProductMapper.selectList(new LambdaQueryWrapper<SalesLedgerProduct>()
|
.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<SalesLedgerProduct> lines = salesLedgerProductMapper.selectList(new LambdaQueryWrapper<SalesLedgerProduct>()
|
.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);
|
}
|
}
|
|
|
}
|