| | |
| | | import com.baomidou.mybatisplus.extension.plugins.pagination.Page; |
| | | import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; |
| | | import com.ruoyi.account.service.AccountIncomeService; |
| | | import com.ruoyi.approve.service.IApproveProcessService; |
| | | import com.ruoyi.approve.vo.ApproveProcessVO; |
| | | import com.ruoyi.approve.pojo.ApproveProcess; |
| | | import com.ruoyi.common.enums.ApproveTypeEnum; |
| | | import com.ruoyi.basic.mapper.CustomerMapper; |
| | | import com.ruoyi.basic.mapper.ProductMapper; |
| | | import com.ruoyi.basic.mapper.ProductModelMapper; |
| | |
| | | private static final String LOCK_PREFIX = "contract_no_lock:"; |
| | | private static final long LOCK_WAIT_TIMEOUT = 10; // 锁等待超时时间(秒) |
| | | private static final long LOCK_EXPIRE_TIME = 30; // 锁自动过期时间(秒) |
| | | private static final int INBOUND_BIZ_TYPE_WEB = 1; |
| | | private static final int INBOUND_BIZ_TYPE_SCAN_QUALIFIED = 2; |
| | | private static final int INBOUND_BIZ_TYPE_SCAN_UNQUALIFIED = 3; |
| | | private final AccountIncomeService accountIncomeService; |
| | | private final SalesLedgerMapper salesLedgerMapper; |
| | | private final CustomerMapper customerMapper; |
| | |
| | | private final StockInRecordService stockInRecordService; |
| | | private final StockOutRecordService stockOutRecordService; |
| | | private final StockUtils stockUtils; |
| | | @Autowired |
| | | private IApproveProcessService approveProcessService; |
| | | |
| | | @Autowired |
| | | private SysDeptMapper sysDeptMapper; |
| | |
| | | |
| | | @Override |
| | | public IPage<SalesLedger> selectSalesLedgerListPage(Page page, SalesLedgerDto salesLedgerDto) { |
| | | return salesLedgerMapper.selectSalesLedgerListPage(page, salesLedgerDto); |
| | | IPage<SalesLedger> iPage = salesLedgerMapper.selectSalesLedgerListPage(page, salesLedgerDto); |
| | | |
| | | if (CollectionUtils.isEmpty(iPage.getRecords())) { |
| | | return iPage; |
| | | } |
| | | |
| | | List<Long> salesLedgerIds = iPage.getRecords().stream().map(SalesLedger::getId).collect(Collectors.toList()); |
| | | |
| | | boolean hasWidthHeightFilter = salesLedgerDto.getWidth() != null || salesLedgerDto.getHeight() != null; |
| | | Map<Long, List<SalesLedgerProduct>> matchedProductsMap = Collections.emptyMap(); |
| | | if (hasWidthHeightFilter) { |
| | | LambdaQueryWrapper<SalesLedgerProduct> productQueryWrapper = new LambdaQueryWrapper<SalesLedgerProduct>() |
| | | .in(SalesLedgerProduct::getSalesLedgerId, salesLedgerIds) |
| | | .eq(SalesLedgerProduct::getType, 1); |
| | | if (salesLedgerDto.getWidth() != null) { |
| | | productQueryWrapper.eq(SalesLedgerProduct::getWidth, salesLedgerDto.getWidth()); |
| | | } |
| | | if (salesLedgerDto.getHeight() != null) { |
| | | productQueryWrapper.eq(SalesLedgerProduct::getHeight, salesLedgerDto.getHeight()); |
| | | } |
| | | List<SalesLedgerProduct> matchedProducts = salesLedgerProductMapper.selectList(productQueryWrapper); |
| | | matchedProductsMap = CollectionUtils.isEmpty(matchedProducts) ? Collections.emptyMap() |
| | | : matchedProducts.stream().collect(Collectors.groupingBy(SalesLedgerProduct::getSalesLedgerId)); |
| | | } |
| | | |
| | | List<SalesLedgerProductTotalsDto> productTotals = salesLedgerProductMapper.selectSalesLedgerProductTotals(salesLedgerIds, 1); |
| | | Map<Long, SalesLedgerProductTotalsDto> productTotalsMap = CollectionUtils.isEmpty(productTotals) |
| | | ? Collections.emptyMap() |
| | | : productTotals.stream() |
| | | .filter(t -> t.getSalesLedgerId() != null) |
| | | .collect(Collectors.toMap(SalesLedgerProductTotalsDto::getSalesLedgerId, Function.identity(), (a, b) -> a)); |
| | | |
| | | List<InvoiceLedgerDto> invoiceLedgerDtoList = invoiceLedgerMapper.invoicedTotal(salesLedgerIds); |
| | | if (CollectionUtils.isEmpty(invoiceLedgerDtoList)) { |
| | | invoiceLedgerDtoList = Collections.emptyList(); |
| | | } |
| | | |
| | | Map<Long, BigDecimal> invoiceTotals = invoiceLedgerDtoList.stream() |
| | | .filter(dto -> dto.getSalesLedgerId() != null && dto.getInvoiceTotal() != null) |
| | | .collect(Collectors.toMap( |
| | | dto -> dto.getSalesLedgerId().longValue(), |
| | | InvoiceLedgerDto::getInvoiceTotal, |
| | | BigDecimal::add |
| | | )); |
| | | |
| | | List<ReceiptPayment> receiptPayments = Collections.emptyList(); |
| | | if (!CollectionUtils.isEmpty(salesLedgerIds)) { |
| | | receiptPayments = receiptPaymentMapper.selectList(new LambdaQueryWrapper<ReceiptPayment>() |
| | | .in(ReceiptPayment::getSalesLedgerId, salesLedgerIds)); |
| | | } |
| | | |
| | | Map<Long, BigDecimal> receiptTotals = new HashMap<>(); |
| | | if (!CollectionUtils.isEmpty(receiptPayments)) { |
| | | for (ReceiptPayment receiptPayment : receiptPayments) { |
| | | if (receiptPayment.getSalesLedgerId() != null && receiptPayment.getReceiptPaymentAmount() != null) { |
| | | receiptTotals.merge(receiptPayment.getSalesLedgerId(), receiptPayment.getReceiptPaymentAmount(), BigDecimal::add); |
| | | } |
| | | } |
| | | } |
| | | |
| | | for (SalesLedger salesLedger : iPage.getRecords()) { |
| | | Long ledgerId = salesLedger.getId(); |
| | | |
| | | SalesLedgerProductTotalsDto totals = productTotalsMap.get(ledgerId); |
| | | if (totals != null) { |
| | | salesLedger.setProductTotalQuantity(totals.getTotalQuantity() != null ? totals.getTotalQuantity() : BigDecimal.ZERO); |
| | | salesLedger.setProductTotalArea(totals.getTotalArea() != null ? totals.getTotalArea() : BigDecimal.ZERO); |
| | | } else { |
| | | salesLedger.setProductTotalQuantity(BigDecimal.ZERO); |
| | | salesLedger.setProductTotalArea(BigDecimal.ZERO); |
| | | } |
| | | if (hasWidthHeightFilter) { |
| | | salesLedger.setMatchedProducts(matchedProductsMap.getOrDefault(ledgerId, Collections.emptyList())); |
| | | } |
| | | |
| | | BigDecimal contractAmount = salesLedger.getContractAmount() == null ? BigDecimal.ZERO : salesLedger.getContractAmount(); |
| | | BigDecimal invoiceTotal = invoiceTotals.getOrDefault(ledgerId, BigDecimal.ZERO); |
| | | BigDecimal receiptPaymentAmountTotal = receiptTotals.getOrDefault(ledgerId, BigDecimal.ZERO); |
| | | |
| | | BigDecimal noInvoiceAmountTotal = contractAmount.subtract(invoiceTotal); |
| | | if (noInvoiceAmountTotal.compareTo(BigDecimal.ZERO) < 0) { |
| | | noInvoiceAmountTotal = BigDecimal.ZERO; |
| | | } |
| | | |
| | | BigDecimal noReceiptPaymentAmountTotal = invoiceTotal.subtract(receiptPaymentAmountTotal); |
| | | if (noReceiptPaymentAmountTotal.compareTo(BigDecimal.ZERO) < 0) { |
| | | noReceiptPaymentAmountTotal = BigDecimal.ZERO; |
| | | } |
| | | |
| | | salesLedger.setNoInvoiceAmountTotal(noInvoiceAmountTotal); |
| | | salesLedger.setInvoiceTotal(invoiceTotal); |
| | | salesLedger.setReceiptPaymentAmountTotal(receiptPaymentAmountTotal); |
| | | salesLedger.setNoReceiptAmount(noReceiptPaymentAmountTotal); |
| | | |
| | | boolean hasInvoiceOperation = invoiceTotal.compareTo(BigDecimal.ZERO) > 0; |
| | | boolean hasReceiptOperation = receiptPaymentAmountTotal.compareTo(BigDecimal.ZERO) > 0; |
| | | salesLedger.setIsEdit(!(hasInvoiceOperation || hasReceiptOperation)); |
| | | } |
| | | |
| | | if (salesLedgerDto.getStatus() != null && salesLedgerDto.getStatus()) { |
| | | iPage.getRecords().removeIf(salesLedger -> |
| | | Objects.equals(salesLedger.getNoInvoiceAmountTotal(), new BigDecimal("0.00"))); |
| | | iPage.setTotal(iPage.getRecords().size()); |
| | | } |
| | | |
| | | return iPage; |
| | | } |
| | | |
| | | @Override |
| | |
| | | if (!updateList.isEmpty()) { |
| | | for (SalesLedgerProduct product : updateList) { |
| | | product.setType(type.getCode()); |
| | | product.setProductStockStatus(0); |
| | | SalesLedgerProduct db = salesLedgerProductMapper.selectById(product.getId()); |
| | | if (db != null) { |
| | | BigDecimal stockedQty = product.getStockedQuantity() != null ? product.getStockedQuantity() : db.getStockedQuantity(); |
| | | BigDecimal orderQty = product.getQuantity() != null ? product.getQuantity() : db.getQuantity(); |
| | | product.setStockedQuantity(stockedQty); |
| | | product.setProductStockStatus(calculateProductStockStatus(stockedQty, orderQty)); |
| | | } else { |
| | | product.setProductStockStatus(0); |
| | | } |
| | | product.fillRemainingQuantity(); |
| | | salesLedgerProductMapper.updateById(product); |
| | | // 清空销售产品绑定的加工 |
| | |
| | | salesLedgerProduct.setNoInvoiceNum(salesLedgerProduct.getQuantity()); |
| | | salesLedgerProduct.setNoInvoiceAmount(salesLedgerProduct.getTaxInclusiveTotalPrice()); |
| | | salesLedgerProduct.setPendingInvoiceTotal(salesLedgerProduct.getTaxInclusiveTotalPrice()); |
| | | salesLedgerProduct.setProductStockStatus(0); |
| | | BigDecimal stockedQty = salesLedgerProduct.getStockedQuantity(); |
| | | BigDecimal orderQty = salesLedgerProduct.getQuantity(); |
| | | salesLedgerProduct.setProductStockStatus(calculateProductStockStatus(stockedQty, orderQty)); |
| | | salesLedgerProduct.fillRemainingQuantity(); |
| | | ProductModel productModel = productModelMapper.selectById(salesLedgerProduct.getProductModelId()); |
| | | if (productModel == null) { |
| | | throw new ServiceException("新增销售台账失败,销售产品不存在"); |
| | | } |
| | | salesLedgerProduct.setUnit(productModel.getUnit()); |
| | | salesLedgerProductMapper.insert(salesLedgerProduct); |
| | | // 绑定产品额外加工 |
| | | // 清空销售产品绑定的加工 |
| | |
| | | // salesLedgerProductServiceImpl.addProductionData(salesLedgerProduct); |
| | | } |
| | | } |
| | | refreshSalesLedgerStockStatus(salesLedgerId); |
| | | } |
| | | |
| | | private int calculateProductStockStatus(BigDecimal stockedQty, BigDecimal orderQty) { |
| | | BigDecimal stocked = stockedQty == null ? BigDecimal.ZERO : stockedQty; |
| | | BigDecimal order = orderQty == null ? BigDecimal.ZERO : orderQty; |
| | | if (stocked.compareTo(BigDecimal.ZERO) <= 0) { |
| | | return 0; |
| | | } |
| | | if (order.compareTo(BigDecimal.ZERO) > 0 && stocked.compareTo(order) < 0) { |
| | | return 1; |
| | | } |
| | | return 2; |
| | | } |
| | | |
| | | private void refreshSalesLedgerStockStatus(Long salesLedgerId) { |
| | | if (salesLedgerId == null) return; |
| | | SalesLedger ledger = baseMapper.selectById(salesLedgerId); |
| | | if (ledger == null) return; |
| | | List<SalesLedgerProduct> allProducts = salesLedgerProductMapper.selectList( |
| | | Wrappers.<SalesLedgerProduct>lambdaQuery() |
| | | .eq(SalesLedgerProduct::getSalesLedgerId, salesLedgerId) |
| | | .eq(SalesLedgerProduct::getType, SaleEnum.SALE.getCode())); |
| | | if (CollectionUtils.isEmpty(allProducts)) { |
| | | ledger.setStockStatus(0); |
| | | baseMapper.updateById(ledger); |
| | | return; |
| | | } |
| | | boolean anyInbound = allProducts.stream().anyMatch(p -> { |
| | | BigDecimal sq = p.getStockedQuantity(); |
| | | return sq != null && sq.compareTo(BigDecimal.ZERO) > 0; |
| | | }); |
| | | boolean allFull = allProducts.stream().allMatch(p -> Objects.equals(p.getProductStockStatus(), 2)); |
| | | ledger.setStockStatus(allFull ? 2 : (anyInbound ? 1 : 0)); |
| | | baseMapper.updateById(ledger); |
| | | } |
| | | |
| | | private SalesLedger convertToEntity(SalesLedgerDto dto) { |
| | |
| | | |
| | | // 查询所有产品 |
| | | List<SalesLedgerProduct> allProducts = salesLedgerProductMapper.selectList( |
| | | new LambdaQueryWrapper<SalesLedgerProduct>().in(SalesLedgerProduct::getSalesLedgerId, salesLedgerIds)); |
| | | new LambdaQueryWrapper<SalesLedgerProduct>() |
| | | .in(SalesLedgerProduct::getSalesLedgerId, salesLedgerIds) |
| | | .eq(SalesLedgerProduct::getType, 1)); |
| | | |
| | | if (CollectionUtils.isNotEmpty(allProducts)) { |
| | | Map<Long, SalesLedger> ledgerMap = ledgers.stream() |
| | | .collect(Collectors.toMap(SalesLedger::getId, Function.identity())); |
| | | |
| | | Map<Long, List<SalesLedgerProduct>> groupedData = new LinkedHashMap<>(); |
| | | for (SalesLedgerProduct p : allProducts) { |
| | | groupedData.computeIfAbsent(p.getSalesLedgerId(), k -> new ArrayList<>()).add(p); |
| | | } |
| | | |
| | | List<SalesInvoicesDto.InvoiceOrderGroupDto> groups = new ArrayList<>(); |
| | | BigDecimal totalQty = BigDecimal.ZERO; |
| | | BigDecimal totalArea = BigDecimal.ZERO; |
| | | |
| | | for (Map.Entry<Long, List<SalesLedgerProduct>> ledgerEntry : groupedData.entrySet()) { |
| | | SalesLedger ledger = ledgerMap.get(ledgerEntry.getKey()); |
| | | String orderNo = ledger != null ? ledger.getSalesContractNo() : ""; |
| | | List<SalesLedgerProduct> products = ledgerEntry.getValue(); |
| | | Map<String, List<SalesLedgerProduct>> groupedData = new LinkedHashMap<>(); |
| | | Map<String, Set<String>> groupOrderNos = new LinkedHashMap<>(); |
| | | for (SalesLedgerProduct p : allProducts) { |
| | | String productCategory = StringUtils.defaultString(p.getProductCategory(), ""); |
| | | String specificationModel = StringUtils.defaultString(p.getSpecificationModel(), ""); |
| | | String groupKey = productCategory + "||" + specificationModel; |
| | | groupedData.computeIfAbsent(groupKey, k -> new ArrayList<>()).add(p); |
| | | |
| | | SalesLedger ledger = ledgerMap.get(p.getSalesLedgerId()); |
| | | String orderNo = ledger != null ? StringUtils.defaultString(ledger.getSalesContractNo(), "") : ""; |
| | | if (StringUtils.isNotEmpty(orderNo)) { |
| | | groupOrderNos.computeIfAbsent(groupKey, k -> new LinkedHashSet<>()).add(orderNo); |
| | | } |
| | | } |
| | | |
| | | for (Map.Entry<String, List<SalesLedgerProduct>> productEntry : groupedData.entrySet()) { |
| | | String key = productEntry.getKey(); |
| | | String[] parts = key.split("\\|\\|", -1); |
| | | String productName = parts.length > 0 ? parts[0] : ""; |
| | | String specificationModel = parts.length > 1 ? parts[1] : ""; |
| | | List<SalesLedgerProduct> products = productEntry.getValue(); |
| | | |
| | | SalesInvoicesDto.InvoiceOrderGroupDto group = new SalesInvoicesDto.InvoiceOrderGroupDto(); |
| | | group.setSalesContractNo(orderNo); |
| | | if (CollectionUtils.isNotEmpty(products)) { |
| | | group.setProductName(products.get(0).getProductCategory()); |
| | | } |
| | | Set<String> orderNos = groupOrderNos.getOrDefault(key, Collections.emptySet()); |
| | | group.setSalesContractNo(String.join(",", orderNos)); |
| | | group.setProductName(productName); |
| | | group.setSpecificationModel(specificationModel); |
| | | |
| | | List<SalesInvoicesDto.InvoiceItemDto> itemDtos = new ArrayList<>(); |
| | | Map<String, SalesInvoicesDto.InvoiceItemDto> mergedItems = new LinkedHashMap<>(); |
| | | BigDecimal groupQty = BigDecimal.ZERO; |
| | | BigDecimal groupArea = BigDecimal.ZERO; |
| | | |
| | | for (SalesLedgerProduct p : products) { |
| | | SalesInvoicesDto.InvoiceItemDto item = new SalesInvoicesDto.InvoiceItemDto(); |
| | | item.setFloorCode(p.getFloorCode()); |
| | | item.setWidthHeight((p.getWidth() != null ? p.getWidth().stripTrailingZeros().toPlainString() : "0") + |
| | | " * " + (p.getHeight() != null ? p.getHeight().stripTrailingZeros().toPlainString() : "0")); |
| | | item.setQuantity(p.getQuantity()); |
| | | String widthHeight = (p.getWidth() != null ? p.getWidth().stripTrailingZeros().toPlainString() : "0") + |
| | | " * " + (p.getHeight() != null ? p.getHeight().stripTrailingZeros().toPlainString() : "0"); |
| | | String floorCode = StringUtils.defaultString(p.getFloorCode(), ""); |
| | | String remark = StringUtils.defaultString(p.getRemark(), ""); |
| | | String processReq = StringUtils.defaultString(p.getProcessRequirement(), ""); |
| | | String itemKey = floorCode + "||" + widthHeight + "||" + remark + "||" + processReq; |
| | | |
| | | BigDecimal qty = p.getQuantity() != null ? p.getQuantity() : BigDecimal.ZERO; |
| | | |
| | | // 面积 |
| | | BigDecimal area = p.getSettleTotalArea() != null ? p.getSettleTotalArea() : p.getActualTotalArea(); |
| | | if (area == null && p.getWidth() != null && p.getHeight() != null && p.getQuantity() != null) { |
| | | area = p.getWidth().multiply(p.getHeight()).multiply(p.getQuantity()).divide(new BigDecimal(1000000), 2, RoundingMode.HALF_UP); |
| | | } |
| | | item.setArea(area); |
| | | item.setRemark(p.getRemark()); |
| | | item.setProcessRequirement(p.getProcessRequirement()); |
| | | area = area != null ? area : BigDecimal.ZERO; |
| | | |
| | | itemDtos.add(item); |
| | | groupQty = groupQty.add(p.getQuantity() != null ? p.getQuantity() : BigDecimal.ZERO); |
| | | groupArea = groupArea.add(area != null ? area : BigDecimal.ZERO); |
| | | SalesInvoicesDto.InvoiceItemDto item = mergedItems.get(itemKey); |
| | | if (item == null) { |
| | | item = new SalesInvoicesDto.InvoiceItemDto(); |
| | | item.setFloorCode(floorCode); |
| | | item.setWidthHeight(widthHeight); |
| | | item.setSpecificationModel(p.getSpecificationModel()); |
| | | item.setQuantity(qty); |
| | | item.setArea(area); |
| | | item.setRemark(remark); |
| | | item.setProcessRequirement(processReq); |
| | | mergedItems.put(itemKey, item); |
| | | } else { |
| | | item.setQuantity((item.getQuantity() != null ? item.getQuantity() : BigDecimal.ZERO).add(qty)); |
| | | item.setArea((item.getArea() != null ? item.getArea() : BigDecimal.ZERO).add(area)); |
| | | } |
| | | |
| | | groupQty = groupQty.add(qty); |
| | | groupArea = groupArea.add(area); |
| | | } |
| | | |
| | | group.setItems(itemDtos); |
| | | group.setItems(new ArrayList<>(mergedItems.values())); |
| | | group.setGroupTotalQuantity(groupQty); |
| | | group.setGroupTotalArea(groupArea.setScale(2, RoundingMode.HALF_UP)); |
| | | groups.add(group); |
| | |
| | | SalesLabelDto dto = new SalesLabelDto(); |
| | | dto.setCustomerName(salesLedger.getCustomerName()); |
| | | dto.setSalesContractNo(salesLedger.getSalesContractNo()); |
| | | dto.setProductName(p.getProductCategory()); |
| | | dto.setProductName(p.getSpecificationModel()); |
| | | |
| | | // 宽*高=数量 |
| | | String specification = (p.getWidth() != null ? p.getWidth().stripTrailingZeros().toPlainString() : "0") + "*" + |
| | |
| | | if (ledger.getStockStatus() == null) { |
| | | throw new ServiceException("入库失败,销售订单入库状态异常"); |
| | | } |
| | | if (ledger.getStockStatus() == 3) { |
| | | throw new ServiceException("入库失败,该销售订单正在入库审批中,请勿重复提交"); |
| | | } |
| | | if (ledger.getStockStatus() == 2) { |
| | | throw new ServiceException("入库失败,该销售订单已入库,请勿重复入库"); |
| | | } |
| | |
| | | if (salesLedgerProducts == null || salesLedgerProducts.isEmpty()) { |
| | | throw new ServiceException("入库失败,未查询到该销售订单的销售产品"); |
| | | } |
| | | String approveUserIds = resolveApproveUserIds(dto.getApproveUserIds(), ledger.getId(), INBOUND_BIZ_TYPE_WEB); |
| | | if (StringUtils.isEmpty(approveUserIds)) { |
| | | throw new ServiceException("入库失败,请选择审批人"); |
| | | } |
| | | |
| | | String productIds = products.stream().map(String::valueOf).collect(Collectors.joining(",")); |
| | | LoginUser loginUser = SecurityUtils.getLoginUser(); |
| | | ApproveProcessVO approveProcessVO = new ApproveProcessVO(); |
| | | approveProcessVO.setApproveType(ApproveTypeEnum.STOCK_IN.getCode()); |
| | | approveProcessVO.setApproveDeptId(loginUser.getCurrentDeptId()); |
| | | approveProcessVO.setApproveReason("入库审批:" + ledger.getSalesContractNo()); |
| | | approveProcessVO.setApproveRemark("salesStock:" + ledger.getId() + ":" + productIds); |
| | | approveProcessVO.setApproveUserIds(approveUserIds); |
| | | approveProcessVO.setApproveUser(loginUser.getUserId()); |
| | | approveProcessVO.setApproveTime(LocalDate.now().toString()); |
| | | try { |
| | | approveProcessService.addApprove(approveProcessVO); |
| | | } catch (Exception e) { |
| | | throw new ServiceException("入库审批发起失败:" + e.getMessage()); |
| | | } |
| | | // 审批中 |
| | | ledger.setStockStatus(3); |
| | | baseMapper.updateById(ledger); |
| | | } |
| | | |
| | | @Override |
| | | public void executeSalesStockApproved(Long salesLedgerId, List<Long> products) { |
| | | SalesProductStockDto dto = new SalesProductStockDto(); |
| | | dto.setSalesLedgerId(salesLedgerId); |
| | | dto.setSalesLedgerProducts(products); |
| | | |
| | | SalesLedger ledger = baseMapper.selectById(dto.getSalesLedgerId()); |
| | | if (ledger == null) { |
| | | throw new ServiceException("入库失败,销售订单不存在"); |
| | | } |
| | | List<SalesLedgerProduct> salesLedgerProducts = salesLedgerProductMapper.selectList(Wrappers.<SalesLedgerProduct>lambdaQuery().in(SalesLedgerProduct::getId, products)); |
| | | for (SalesLedgerProduct product : salesLedgerProducts) { |
| | | if (!Objects.equals(product.getSalesLedgerId(), ledger.getId())) { |
| | | throw new ServiceException("入库失败,存在不属于当前销售订单的产品"); |
| | |
| | | } |
| | | inboundQtyByLineId.merge(inbound.getId(), inboundQty, BigDecimal::add); |
| | | } |
| | | List<SalesLedgerProduct> selectedProducts = salesLedgerProductMapper.selectList(Wrappers.<SalesLedgerProduct>lambdaQuery() |
| | | .in(SalesLedgerProduct::getId, inboundQtyByLineId.keySet())); |
| | | if (CollectionUtils.isEmpty(selectedProducts)) { |
| | | throw new ServiceException("入库失败,未查询到入库产品"); |
| | | } |
| | | if (selectedProducts.size() != inboundQtyByLineId.size()) { |
| | | throw new ServiceException("入库失败,部分产品不存在"); |
| | | } |
| | | for (SalesLedgerProduct selectedProduct : selectedProducts) { |
| | | if (!Objects.equals(selectedProduct.getSalesLedgerId(), salesLedger.getId())) { |
| | | throw new ServiceException("入库失败,销售产品与订单不匹配"); |
| | | } |
| | | if (!Objects.equals(selectedProduct.getType(), SaleEnum.SALE.getCode())) { |
| | | throw new ServiceException("入库失败,仅支持销售订单产品行"); |
| | | } |
| | | if (selectedProduct.getProductModelId() == null) { |
| | | throw new ServiceException("入库失败,产品规格未维护,无法入库"); |
| | | } |
| | | } |
| | | String approveUserIds = resolveApproveUserIds(dto.getApproveUserIds(), salesLedger.getId(), INBOUND_BIZ_TYPE_SCAN_QUALIFIED); |
| | | if (StringUtils.isEmpty(approveUserIds)) { |
| | | throw new ServiceException("入库失败,请选择审批人"); |
| | | } |
| | | String lines = inboundQtyByLineId.entrySet().stream() |
| | | .map(e -> e.getKey() + "@" + e.getValue().stripTrailingZeros().toPlainString()) |
| | | .collect(Collectors.joining(",")); |
| | | String reason = "销售扫码合格入库审批:" + salesLedger.getSalesContractNo(); |
| | | String remark = "scanQualified:" + salesLedger.getId() + ":" + lines; |
| | | ApproveProcess exist = approveProcessService.getOne(new LambdaQueryWrapper<ApproveProcess>() |
| | | .eq(ApproveProcess::getApproveType, ApproveTypeEnum.STOCK_IN.getCode()) |
| | | .eq(ApproveProcess::getApproveRemark, remark) |
| | | .eq(ApproveProcess::getApproveDelete, 0) |
| | | .orderByDesc(ApproveProcess::getCreateTime) |
| | | .last("limit 1")); |
| | | if (exist != null && !Objects.equals(exist.getApproveStatus(), 3)) { |
| | | throw new ServiceException("入库失败,该申请已提交审批"); |
| | | } |
| | | LoginUser loginUser = SecurityUtils.getLoginUser(); |
| | | ApproveProcessVO approveProcessVO = new ApproveProcessVO(); |
| | | approveProcessVO.setApproveType(ApproveTypeEnum.STOCK_IN.getCode()); |
| | | approveProcessVO.setApproveDeptId(loginUser.getCurrentDeptId()); |
| | | approveProcessVO.setApproveReason(reason); |
| | | approveProcessVO.setApproveRemark(remark); |
| | | approveProcessVO.setApproveUserIds(approveUserIds); |
| | | approveProcessVO.setApproveUser(loginUser.getUserId()); |
| | | approveProcessVO.setApproveTime(LocalDate.now().toString()); |
| | | try { |
| | | approveProcessService.addApprove(approveProcessVO); |
| | | } catch (Exception e) { |
| | | throw new ServiceException("入库审批发起失败:" + e.getMessage()); |
| | | } |
| | | } |
| | | |
| | | @Override |
| | | @Transactional(rollbackFor = Exception.class) |
| | | public void executeSalesScanInboundApproved(Long salesLedgerId, Map<Long, BigDecimal> inboundQtyByLineId) { |
| | | SalesLedger salesLedger = baseMapper.selectById(salesLedgerId); |
| | | if (salesLedger == null) { |
| | | throw new ServiceException("入库失败,销售订单不存在"); |
| | | } |
| | | Long ledgerId = salesLedger.getId(); |
| | | for (Map.Entry<Long, BigDecimal> entry : inboundQtyByLineId.entrySet()) { |
| | | Long salesLedgerProductId = entry.getKey(); |
| | | BigDecimal inboundThisLine = entry.getValue(); |
| | | if (inboundThisLine.compareTo(BigDecimal.ZERO) == 0) { |
| | | if (inboundThisLine == null || inboundThisLine.compareTo(BigDecimal.ZERO) <= 0) { |
| | | continue; |
| | | } |
| | | SalesLedgerProduct dbProduct = salesLedgerProductMapper.selectById(salesLedgerProductId); |
| | |
| | | } |
| | | inboundQtyByLineId.merge(inbound.getId(), inboundQty, BigDecimal::add); |
| | | } |
| | | List<SalesLedgerProduct> selectedProducts = salesLedgerProductMapper.selectList(Wrappers.<SalesLedgerProduct>lambdaQuery() |
| | | .in(SalesLedgerProduct::getId, inboundQtyByLineId.keySet())); |
| | | if (CollectionUtils.isEmpty(selectedProducts)) { |
| | | throw new ServiceException("不合格入库失败,未查询到入库产品"); |
| | | } |
| | | if (selectedProducts.size() != inboundQtyByLineId.size()) { |
| | | throw new ServiceException("不合格入库失败,部分产品不存在"); |
| | | } |
| | | for (SalesLedgerProduct selectedProduct : selectedProducts) { |
| | | if (!Objects.equals(selectedProduct.getSalesLedgerId(), salesLedger.getId())) { |
| | | throw new ServiceException("不合格入库失败,销售产品与订单不匹配"); |
| | | } |
| | | if (!Objects.equals(selectedProduct.getType(), SaleEnum.SALE.getCode())) { |
| | | throw new ServiceException("不合格入库失败,仅支持销售订单产品行"); |
| | | } |
| | | if (selectedProduct.getProductModelId() == null) { |
| | | throw new ServiceException("不合格入库失败,产品规格未维护,无法入库"); |
| | | } |
| | | } |
| | | String approveUserIds = resolveApproveUserIds(dto.getApproveUserIds(), salesLedger.getId(), INBOUND_BIZ_TYPE_SCAN_UNQUALIFIED); |
| | | if (StringUtils.isEmpty(approveUserIds)) { |
| | | throw new ServiceException("不合格入库失败,请选择审批人"); |
| | | } |
| | | String lines = inboundQtyByLineId.entrySet().stream() |
| | | .map(e -> e.getKey() + "@" + e.getValue().stripTrailingZeros().toPlainString()) |
| | | .collect(Collectors.joining(",")); |
| | | String reason = "销售扫码不合格入库审批:" + salesLedger.getSalesContractNo(); |
| | | String remark = "scanUnqualified:" + salesLedger.getId() + ":" + lines; |
| | | ApproveProcess exist = approveProcessService.getOne(new LambdaQueryWrapper<ApproveProcess>() |
| | | .eq(ApproveProcess::getApproveType, ApproveTypeEnum.STOCK_IN.getCode()) |
| | | .eq(ApproveProcess::getApproveRemark, remark) |
| | | .eq(ApproveProcess::getApproveDelete, 0) |
| | | .orderByDesc(ApproveProcess::getCreateTime) |
| | | .last("limit 1")); |
| | | if (exist != null && !Objects.equals(exist.getApproveStatus(), 3)) { |
| | | throw new ServiceException("不合格入库失败,该申请已提交审批"); |
| | | } |
| | | LoginUser loginUser = SecurityUtils.getLoginUser(); |
| | | ApproveProcessVO approveProcessVO = new ApproveProcessVO(); |
| | | approveProcessVO.setApproveType(ApproveTypeEnum.STOCK_IN.getCode()); |
| | | approveProcessVO.setApproveDeptId(loginUser.getCurrentDeptId()); |
| | | approveProcessVO.setApproveReason(reason); |
| | | approveProcessVO.setApproveRemark(remark); |
| | | approveProcessVO.setApproveUserIds(approveUserIds); |
| | | approveProcessVO.setApproveUser(loginUser.getUserId()); |
| | | approveProcessVO.setApproveTime(LocalDate.now().toString()); |
| | | try { |
| | | approveProcessService.addApprove(approveProcessVO); |
| | | } catch (Exception e) { |
| | | throw new ServiceException("不合格入库审批发起失败:" + e.getMessage()); |
| | | } |
| | | } |
| | | |
| | | @Override |
| | | @Transactional(rollbackFor = Exception.class) |
| | | public void executeSalesScanInboundUnqualifiedApproved(Long salesLedgerId, Map<Long, BigDecimal> inboundQtyByLineId) { |
| | | SalesLedger salesLedger = baseMapper.selectById(salesLedgerId); |
| | | if (salesLedger == null) { |
| | | throw new ServiceException("不合格入库失败,销售订单不存在"); |
| | | } |
| | | Long ledgerId = salesLedger.getId(); |
| | | for (Map.Entry<Long, BigDecimal> entry : inboundQtyByLineId.entrySet()) { |
| | | Long salesLedgerProductId = entry.getKey(); |
| | | BigDecimal inboundThisLine = entry.getValue(); |
| | | if (inboundThisLine.compareTo(BigDecimal.ZERO) == 0) { |
| | | if (inboundThisLine == null || inboundThisLine.compareTo(BigDecimal.ZERO) <= 0) { |
| | | continue; |
| | | } |
| | | SalesLedgerProduct dbProduct = salesLedgerProductMapper.selectById(salesLedgerProductId); |
| | |
| | | } |
| | | stockUtils.addUnStock(ledgerId, dbProduct.getId(), dbProduct.getProductModelId(), inboundThisLine, |
| | | StockInUnQualifiedRecordTypeEnum.SALES_SCAN_UNSTOCK_IN.getCode(), dbProduct.getId()); |
| | | |
| | | BigDecimal oldUnStocked = dbProduct.getUnqualifiedStockedQuantity() == null ? BigDecimal.ZERO : dbProduct.getUnqualifiedStockedQuantity(); |
| | | dbProduct.setUnqualifiedStockedQuantity(oldUnStocked.add(inboundThisLine)); |
| | | dbProduct.fillRemainingQuantity(); |
| | |
| | | } |
| | | } |
| | | |
| | | private String resolveApproveUserIds(String approveUserIds, Long salesLedgerId, Integer bizType) { |
| | | if (StringUtils.isNotEmpty(approveUserIds)) { |
| | | return approveUserIds; |
| | | } |
| | | LambdaQueryWrapper<ApproveProcess> wrapper = new LambdaQueryWrapper<ApproveProcess>() |
| | | .eq(ApproveProcess::getApproveType, ApproveTypeEnum.STOCK_IN.getCode()) |
| | | .eq(ApproveProcess::getApproveDelete, 0) |
| | | .orderByDesc(ApproveProcess::getCreateTime) |
| | | .last("limit 1"); |
| | | if (Objects.equals(bizType, INBOUND_BIZ_TYPE_WEB)) { |
| | | wrapper.likeRight(ApproveProcess::getApproveReason, "入库审批:") |
| | | .likeRight(ApproveProcess::getApproveRemark, "salesStock:" + salesLedgerId + ":"); |
| | | } else if (Objects.equals(bizType, INBOUND_BIZ_TYPE_SCAN_QUALIFIED)) { |
| | | wrapper.likeRight(ApproveProcess::getApproveReason, "销售扫码合格入库审批:") |
| | | .likeRight(ApproveProcess::getApproveRemark, "scanQualified:" + salesLedgerId + ":"); |
| | | } else if (Objects.equals(bizType, INBOUND_BIZ_TYPE_SCAN_UNQUALIFIED)) { |
| | | wrapper.likeRight(ApproveProcess::getApproveReason, "销售扫码不合格入库审批:") |
| | | .likeRight(ApproveProcess::getApproveRemark, "scanUnqualified:" + salesLedgerId + ":"); |
| | | } else { |
| | | return null; |
| | | } |
| | | ApproveProcess latest = approveProcessService.getOne(wrapper); |
| | | return latest == null ? null : latest.getApproveUserIds(); |
| | | } |
| | | |
| | | @Override |
| | | @Transactional(rollbackFor = Exception.class) |
| | | public void scanOutbound(SalesScanInboundDto dto) { |