| | |
| | | package com.by4cloud.platformx.business.service.impl; |
| | | |
| | | import cn.hutool.core.bean.BeanUtil; |
| | | import cn.hutool.core.date.DateUtil; |
| | | import cn.hutool.core.util.ArrayUtil; |
| | | import cn.hutool.core.util.ObjUtil; |
| | | import cn.hutool.core.util.StrUtil; |
| | | import com.baomidou.mybatisplus.core.toolkit.Wrappers; |
| | | import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; |
| | | import com.by4cloud.platformx.business.entity.OutBound; |
| | | import com.by4cloud.platformx.business.mapper.OutBoundMapper; |
| | | import com.by4cloud.platformx.business.dto.OutBoundAddDTO; |
| | | import com.by4cloud.platformx.business.dto.OutSubjectMatterAddDTO; |
| | | import com.by4cloud.platformx.business.entity.*; |
| | | import com.by4cloud.platformx.business.mapper.*; |
| | | import com.by4cloud.platformx.business.service.OutBoundService; |
| | | import com.by4cloud.platformx.common.core.util.R; |
| | | import com.github.yulichang.wrapper.MPJLambdaWrapper; |
| | | import lombok.RequiredArgsConstructor; |
| | | import org.springframework.stereotype.Service; |
| | | |
| | | import java.math.BigDecimal; |
| | | import java.math.RoundingMode; |
| | | import java.util.ArrayList; |
| | | import java.util.Date; |
| | | import java.util.List; |
| | | import java.util.Map; |
| | | import java.util.stream.Collectors; |
| | | |
| | | /** |
| | | * EPR出库记录 |
| | |
| | | * @date 2026-04-29 11:32:05 |
| | | */ |
| | | @Service |
| | | @RequiredArgsConstructor |
| | | public class OutBoundServiceImpl extends ServiceImpl<OutBoundMapper, OutBound> implements OutBoundService { |
| | | |
| | | private final ContractPaymentScheduleMapper scheduleMapper; |
| | | private final ContractSubjectMatterMapper contractSubjectMatterMapper; |
| | | private final PaymentConfirmMapper paymentConfirmMapper; |
| | | private final ContractMapper contractMapper; |
| | | private final ContractPaymentScheduleProcessMapper scheduleProcessMapper; |
| | | |
| | | @Override |
| | | public boolean save(OutBound entity) { |
| | | entity.setOutBoundTime(new Date()); |
| | | return super.save(entity); |
| | | public R add(OutBoundAddDTO addDTO) { |
| | | if (ArrayUtil.isEmpty(addDTO.getSubjectMatterList().toArray())) { |
| | | return R.failed("出库标的物不能为空"); |
| | | } |
| | | MPJLambdaWrapper<ContractSubjectMatter> wrapper = new MPJLambdaWrapper<ContractSubjectMatter>() |
| | | .selectAll(ContractSubjectMatter.class) |
| | | .ne(ContractSubjectMatter::getDeliveryStatus, 2) |
| | | .exists(Contract.class, contractQuery -> |
| | | contractQuery.select(ContractSubjectMatter::getId) |
| | | .eq(Contract::getId, ContractSubjectMatter::getContractId) |
| | | .eq(StrUtil.isNotBlank(addDTO.getBusGuestName()), Contract::getPartyA, addDTO.getBusGuestName()) |
| | | .eq(ObjUtil.isNotNull(addDTO.getBusGuestId()), Contract::getPartyAId, addDTO.getBusGuestId()) |
| | | ) |
| | | .orderByAsc(ContractSubjectMatter::getCreateTime); |
| | | List<ContractSubjectMatter> subjectMatterList = contractSubjectMatterMapper.selectList(wrapper); |
| | | if (ArrayUtil.isEmpty(subjectMatterList.toArray())) { |
| | | return R.failed("没有查询到相关合同订单"); |
| | | } |
| | | Double outTotal = subjectMatterList.stream().mapToDouble(item -> { |
| | | // 优先判断 remainNum |
| | | if (item.getRemainingQuantity() != null && item.getRemainingQuantity().compareTo(new BigDecimal("0")) > 0) { |
| | | return item.getRemainingQuantity().doubleValue(); |
| | | } else { |
| | | // remainNum 为 null 或 0 时,使用 quantity |
| | | return item.getQuantity() == null ? 0.0 : item.getQuantity().doubleValue(); |
| | | } |
| | | }).sum(); |
| | | BigDecimal orderNum = addDTO.getSubjectMatterList().stream().map(OutSubjectMatterAddDTO::getOutBoundNum) |
| | | .filter(num -> num != null) // 过滤掉 null 值,防止 NullPointerException |
| | | .reduce(BigDecimal.ZERO, BigDecimal::add); |
| | | if (new BigDecimal(outTotal).compareTo(orderNum) < 0) { |
| | | return R.failed("出库数量超出合同订单数量"); |
| | | } |
| | | //当前入库标的物 |
| | | List<ContractSubjectMatter> currentOutList = new ArrayList<>(); |
| | | addDTO.getSubjectMatterList().stream().forEach(outSubjectMatterAddDTO -> { |
| | | BigDecimal remainNum = outSubjectMatterAddDTO.getOutBoundNum(); |
| | | for (ContractSubjectMatter subjectMatter : subjectMatterList) { |
| | | |
| | | if (subjectMatter.getDeliveredQuantity().compareTo(new BigDecimal("0")) > 0) { |
| | | remainNum = remainNum.subtract(subjectMatter.getRemainingQuantity()); |
| | | //部分交付状态下 全部出库 |
| | | if (remainNum.compareTo(new BigDecimal("0")) >= 0) { |
| | | BigDecimal currentOut = subjectMatter.getRemainingQuantity(); |
| | | subjectMatter.setDeliveredQuantity(subjectMatter.getQuantity()); |
| | | subjectMatter.setRemainingQuantity(new BigDecimal("0")); |
| | | subjectMatter.setActualDeliveryDate(DateUtil.today()); |
| | | subjectMatter.setDeliveryStatus(2); |
| | | subjectMatter.setLastDeliveredQuantity(currentOut); |
| | | contractSubjectMatterMapper.updateById(subjectMatter); |
| | | currentOutList.add(subjectMatter); |
| | | } else { |
| | | BigDecimal currentOut = remainNum.add(subjectMatter.getRemainingQuantity()); |
| | | subjectMatter.setDeliveredQuantity(subjectMatter.getQuantity().add(remainNum)); |
| | | subjectMatter.setRemainingQuantity(remainNum.multiply(new BigDecimal("-1"))); |
| | | subjectMatter.setActualDeliveryDate(DateUtil.today()); |
| | | subjectMatter.setDeliveryStatus(1); |
| | | subjectMatter.setLastDeliveredQuantity(currentOut); |
| | | contractSubjectMatterMapper.updateById(subjectMatter); |
| | | currentOutList.add(subjectMatter); |
| | | break; |
| | | } |
| | | |
| | | } else { |
| | | //未交付状态下 |
| | | remainNum = remainNum.subtract(subjectMatter.getQuantity()); |
| | | if (remainNum.compareTo(new BigDecimal("0")) >= 0) { |
| | | subjectMatter.setDeliveredQuantity(subjectMatter.getQuantity()); |
| | | subjectMatter.setRemainingQuantity(new BigDecimal("0")); |
| | | subjectMatter.setActualDeliveryDate(DateUtil.today()); |
| | | subjectMatter.setDeliveryStatus(2); |
| | | contractSubjectMatterMapper.updateById(subjectMatter); |
| | | subjectMatter.setLastDeliveredQuantity(subjectMatter.getQuantity()); |
| | | currentOutList.add(subjectMatter); |
| | | } else { |
| | | BigDecimal currentOut = remainNum.add(subjectMatter.getQuantity()); |
| | | subjectMatter.setDeliveredQuantity(subjectMatter.getQuantity().add(remainNum)); |
| | | subjectMatter.setRemainingQuantity(remainNum.multiply(new BigDecimal("-1"))); |
| | | subjectMatter.setActualDeliveryDate(DateUtil.today()); |
| | | subjectMatter.setDeliveryStatus(1); |
| | | subjectMatter.setLastDeliveredQuantity(currentOut); |
| | | contractSubjectMatterMapper.updateById(subjectMatter); |
| | | currentOutList.add(subjectMatter); |
| | | break; |
| | | } |
| | | } |
| | | } |
| | | OutBound outBound = BeanUtil.copyProperties(outSubjectMatterAddDTO, OutBound.class); |
| | | outBound.setBusGuestName(addDTO.getBusGuestName()); |
| | | outBound.setBusGuestId(addDTO.getBusGuestId()); |
| | | outBound.setOutBoundTime(addDTO.getOutBoundTime()); |
| | | baseMapper.insert(outBound); |
| | | }); |
| | | |
| | | //出库应收款明细 |
| | | if (ArrayUtil.isNotEmpty(currentOutList.toArray())) { |
| | | //根据合同id拆分应收明细 |
| | | Map<Long, List<ContractSubjectMatter>> conSubMatMap = currentOutList.stream().collect(Collectors.groupingBy(ContractSubjectMatter::getContractId)); |
| | | |
| | | conSubMatMap.forEach((key, value) -> { |
| | | //查询合同是否有出库阶段 |
| | | ContractPaymentSchedule schedule = scheduleMapper.selectOne(Wrappers.<ContractPaymentSchedule>lambdaQuery().eq(ContractPaymentSchedule::getContractId, key) |
| | | .eq(ContractPaymentSchedule::getStageName, "发货前")); |
| | | //当前出库金额 |
| | | BigDecimal outPirce = value.stream() |
| | | .filter(item -> item.getLastDeliveredQuantity() != null && item.getUnitPrice() != null) |
| | | .map(item -> item.getLastDeliveredQuantity().multiply(item.getUnitPrice())) |
| | | .reduce(BigDecimal.ZERO, BigDecimal::add); |
| | | //合同金额 |
| | | Contract contract = contractMapper.selectById(key); |
| | | //出库是否完成 |
| | | if (outPirce.compareTo(contract.getAmount())<0){ |
| | | contract.setBillingStatus("1"); |
| | | contractMapper.updateById(contract); |
| | | } |
| | | if (outPirce.compareTo(contract.getAmount())==0){ |
| | | contract.setBillingStatus("2"); |
| | | contractMapper.updateById(contract); |
| | | } |
| | | if (ObjUtil.isNotNull(schedule)) { |
| | | |
| | | //按出库资金比例 计算发货前应收更新资金明细 |
| | | PaymentConfirm paymentConfirm = new PaymentConfirm(); |
| | | paymentConfirm.setBusinessType("发货前应收"); |
| | | paymentConfirm.setBusGuestId(contract.getPartyAId()); |
| | | paymentConfirm.setBusGuestName(contract.getPartyA()); |
| | | paymentConfirm.setContractId(contract.getId()); |
| | | paymentConfirm.setContractName(contract.getContractName()); |
| | | paymentConfirm.setContractNo(contract.getContractNo()); |
| | | paymentConfirm.setScheduleId(schedule.getId()); |
| | | paymentConfirm.setScheduleName(schedule.getStageName()); |
| | | paymentConfirm.setConfirmTime(addDTO.getOutBoundTime()); |
| | | paymentConfirm.setTransationAmount(outPirce.divide(contract.getAmount(), 10, RoundingMode.HALF_UP).multiply(schedule.getPlannedAmount())); |
| | | paymentConfirm.setReceivableAmount(paymentConfirm.getTransationAmount()); |
| | | paymentConfirm.setTotalAmount(paymentConfirm.getTransationAmount().multiply(new BigDecimal("-1"))); |
| | | //最近应收 |
| | | PaymentConfirm lastConfirm = paymentConfirmMapper.selectOne(Wrappers.<PaymentConfirm>lambdaQuery().eq(PaymentConfirm::getContractId, contract.getId()) |
| | | .orderByDesc(PaymentConfirm::getCreateTime).last("limit 1")); |
| | | if (ObjUtil.isNotNull(lastConfirm)) { |
| | | paymentConfirm.setTotalAmount(paymentConfirm.getTotalAmount().add(lastConfirm.getTotalAmount())); |
| | | paymentConfirmMapper.insert(paymentConfirm); |
| | | } else { |
| | | paymentConfirmMapper.insert(paymentConfirm); |
| | | } |
| | | //更新发货前收款生效时间 |
| | | if (ObjUtil.isNull(schedule.getEffectiveDate())){ |
| | | schedule.setEffectiveDate(DateUtil.offsetDay(addDTO.getOutBoundTime(),schedule.getAgreedDays())); |
| | | //查询是否有后续阶段 |
| | | List<ContractPaymentSchedule> afterSchedule = scheduleMapper.selectList(Wrappers.<ContractPaymentSchedule>lambdaQuery().eq(ContractPaymentSchedule::getContractId, key) |
| | | .gt(ContractPaymentSchedule::getStageOrder, schedule.getStageOrder())); |
| | | if (ArrayUtil.isEmpty(afterSchedule.toArray())){ |
| | | schedule.setEffectiveEndDate(contract.getExpirationDate()); |
| | | } |
| | | scheduleMapper.updateById(schedule); |
| | | //查询是否有之前阶段 |
| | | ContractPaymentSchedule beforeSchedule = scheduleMapper.selectOne(Wrappers.<ContractPaymentSchedule>lambdaQuery().eq(ContractPaymentSchedule::getContractId, schedule.getContractId()) |
| | | .lt(ContractPaymentSchedule::getStageOrder, schedule.getStageOrder()).orderByDesc(ContractPaymentSchedule::getCreateTime).last("limit 1")); |
| | | if (ObjUtil.isNotNull(beforeSchedule)){ |
| | | beforeSchedule.setEffectiveEndDate(DateUtil.offsetDay(addDTO.getOutBoundTime(),beforeSchedule.getAgreedDays())); |
| | | scheduleMapper.updateById(beforeSchedule); |
| | | } |
| | | } |
| | | //更新履约 |
| | | ContractPaymentScheduleProcess contractPaymentScheduleProcess = new ContractPaymentScheduleProcess(); |
| | | contractPaymentScheduleProcess.setContractId(schedule.getContractId()); |
| | | contractPaymentScheduleProcess.setContractName(schedule.getContractName()); |
| | | contractPaymentScheduleProcess.setScheduleId(schedule.getId()); |
| | | contractPaymentScheduleProcess.setScheduleName(schedule.getStageName()); |
| | | contractPaymentScheduleProcess.setProcessDate(addDTO.getOutBoundTime()); |
| | | scheduleProcessMapper.insert(contractPaymentScheduleProcess); |
| | | //查询是否有后续阶段 |
| | | List<ContractPaymentSchedule> afterSchedule = scheduleMapper.selectList(Wrappers.<ContractPaymentSchedule>lambdaQuery() |
| | | .eq(ContractPaymentSchedule::getContractId, schedule.getContractId()) |
| | | .gt(ContractPaymentSchedule::getStageOrder, schedule.getStageOrder())); |
| | | if (ArrayUtil.isNotEmpty(afterSchedule.toArray())&&afterSchedule.size()==1){ |
| | | |
| | | //最后阶段生效时间 |
| | | ContractPaymentSchedule endSchedule = afterSchedule.get(0); |
| | | endSchedule.setEffectiveDate(DateUtil.offsetDay(addDTO.getOutBoundTime(),endSchedule.getAgreedDays())); |
| | | endSchedule.setEffectiveEndDate(contract.getExpirationDate()); |
| | | scheduleMapper.updateById(endSchedule); |
| | | //当前阶段生效时间 |
| | | schedule.setEffectiveDate(DateUtil.offsetDay(addDTO.getOutBoundTime(),schedule.getAgreedDays())); |
| | | schedule.setEffectiveEndDate(endSchedule.getEffectiveDate()); |
| | | scheduleMapper.updateById(schedule); |
| | | //最后阶段应收 |
| | | PaymentConfirm newConfim = new PaymentConfirm(); |
| | | newConfim.setBusinessType(endSchedule.getStageName()+"应收"); |
| | | newConfim.setBusGuestId(contract.getPartyAId()); |
| | | newConfim.setBusGuestName(contract.getPartyA()); |
| | | newConfim.setContractId(contract.getId()); |
| | | newConfim.setContractName(contract.getContractName()); |
| | | newConfim.setContractNo(contract.getContractNo()); |
| | | newConfim.setScheduleId(endSchedule.getId()); |
| | | newConfim.setScheduleName(schedule.getStageName()); |
| | | newConfim.setConfirmTime(addDTO.getOutBoundTime()); |
| | | newConfim.setTransationAmount(endSchedule.getPlannedAmount()); |
| | | newConfim.setReceivableAmount(endSchedule.getPlannedAmount()); |
| | | PaymentConfirm lastNewConfirm = paymentConfirmMapper.selectOne(Wrappers.<PaymentConfirm>lambdaQuery().eq(PaymentConfirm::getContractId,contract.getId()) |
| | | .orderByDesc(PaymentConfirm::getCreateTime).last("limit 1")); |
| | | BigDecimal lastNewTotal = new BigDecimal("0"); |
| | | if(ObjUtil.isNotNull(lastNewConfirm)){ |
| | | lastNewTotal = lastNewConfirm.getTotalAmount(); |
| | | } |
| | | newConfim.setTotalAmount(lastNewTotal.subtract(newConfim.getReceivableAmount())); |
| | | paymentConfirmMapper.insert(newConfim); |
| | | } |
| | | if(ArrayUtil.isEmpty(afterSchedule.toArray())){ |
| | | schedule.setEffectiveEndDate(contract.getExpirationDate()); |
| | | scheduleMapper.updateById(schedule); |
| | | } |
| | | } |
| | | }); |
| | | } |
| | | |
| | | return R.ok(); |
| | | } |
| | | } |