platformx-business-finance-biz/src/main/java/com/by4cloud/platformx/business/service/impl/OutBoundServiceImpl.java
@@ -1,6 +1,7 @@
package com.by4cloud.platformx.business.service.impl;
import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.date.DatePattern;
import cn.hutool.core.date.DateUtil;
import cn.hutool.core.util.ArrayUtil;
import cn.hutool.core.util.ObjUtil;
@@ -13,16 +14,14 @@
import com.by4cloud.platformx.business.mapper.*;
import com.by4cloud.platformx.business.service.OutBoundService;
import com.by4cloud.platformx.common.core.util.R;
import com.by4cloud.platformx.common.security.util.SecurityUtils;
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.*;
import java.util.stream.Collectors;
/**
@@ -40,6 +39,7 @@
   private final PaymentConfirmMapper paymentConfirmMapper;
   private final ContractMapper contractMapper;
   private final ContractPaymentScheduleProcessMapper scheduleProcessMapper;
   private final ContractOutBoundMapper contractOutBoundMapper;
   @Override
   public R add(OutBoundAddDTO addDTO) {
@@ -49,6 +49,7 @@
      MPJLambdaWrapper<ContractSubjectMatter> wrapper = new MPJLambdaWrapper<ContractSubjectMatter>()
            .selectAll(ContractSubjectMatter.class)
            .ne(ContractSubjectMatter::getDeliveryStatus, 2)
            .in(ContractSubjectMatter::getMaterialCode, addDTO.getSubjectMatterList().stream().map(item -> item.getSubjectMatterCode()).collect(Collectors.toList()))
            .exists(Contract.class, contractQuery ->
                  contractQuery.select(ContractSubjectMatter::getId)
                        .eq(Contract::getId, ContractSubjectMatter::getContractId)
@@ -60,110 +61,121 @@
      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)
      List<String> erpCodeList = subjectMatterList.stream().map(ContractSubjectMatter::getMaterialCode).collect(Collectors.toList());
      BigDecimal orderNum = subjectMatterList.stream().filter(item ->item.getDeliveryStatus()!=2)
            .map(item ->item.getDeliveryStatus()==0?item.getQuantity():item.getRemainingQuantity())
            .reduce(BigDecimal.ZERO, BigDecimal::add);
      BigDecimal outTotal = addDTO.getSubjectMatterList().stream()
            .filter(item->erpCodeList.contains(item.getSubjectMatterCode()))
            .map(OutSubjectMatterAddDTO::getOutBoundNum)
            .filter(num -> num != null) // 过滤掉 null 值,防止 NullPointerException
            .reduce(BigDecimal.ZERO, BigDecimal::add);
      if (new BigDecimal(outTotal).compareTo(orderNum) < 0) {
      if (outTotal.compareTo(orderNum) > 0) {
         return R.failed("出库数量超出合同订单数量");
      }
      //当前入库标的物
      List<ContractSubjectMatter> currentOutList = new ArrayList<>();
      Map<Long, Object> currentOutMap = new HashMap<>();
      //出库批次
      String batchNumber = DateUtil.format(new Date(), DatePattern.PURE_DATETIME_MS_PATTERN);
      //出库
      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.setBatchNumber(batchNumber);
         outBound.setBusGuestName(addDTO.getBusGuestName());
         outBound.setBusGuestId(addDTO.getBusGuestId());
         outBound.setOutBoundTime(addDTO.getOutBoundTime());
         baseMapper.insert(outBound);
         BigDecimal remainNum = outSubjectMatterAddDTO.getOutBoundNum();
         for (ContractSubjectMatter subjectMatter : subjectMatterList) {
            if (outSubjectMatterAddDTO.getSubjectMatterCode().equals(subjectMatter.getMaterialCode())) {
               Contract contract = contractMapper.selectById(subjectMatter.getContractId());
               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);
                     currentOutMap.put(subjectMatter.getContractId(), subjectMatter);
                     //新增合同出库历史
                     saveContractOutBound(contract, addDTO, subjectMatter,outBound.getId());
                  } else {
                     BigDecimal currentOut = remainNum.add(subjectMatter.getRemainingQuantity());
                     subjectMatter.setDeliveredQuantity(subjectMatter.getQuantity().add(remainNum));
                     subjectMatter.setRemainingQuantity(subjectMatter.getRemainingQuantity().subtract(currentOut));
                     subjectMatter.setActualDeliveryDate(DateUtil.today());
                     subjectMatter.setDeliveryStatus(1);
                     subjectMatter.setLastDeliveredQuantity(currentOut);
                     contractSubjectMatterMapper.updateById(subjectMatter);
                     currentOutMap.put(subjectMatter.getContractId(), subjectMatter);
                     //新增合同出库历史
                     saveContractOutBound(contract, addDTO, subjectMatter,outBound.getId());
                     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);
                     subjectMatter.setLastDeliveredQuantity(subjectMatter.getQuantity());
                     contractSubjectMatterMapper.updateById(subjectMatter);
                     currentOutMap.put(subjectMatter.getContractId(), subjectMatter);
                     //新增合同出库历史
                     saveContractOutBound(contract, addDTO, subjectMatter,outBound.getId());
                  } 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);
                     currentOutMap.put(subjectMatter.getContractId(), subjectMatter);
                     //新增合同出库历史
                     saveContractOutBound(contract, addDTO, subjectMatter,outBound.getId());
                     break;
                  }
               }
            }
         }
      });
      //出库应收款明细
      if (ArrayUtil.isNotEmpty(currentOutList.toArray())) {
         //根据合同id拆分应收明细
         Map<Long, List<ContractSubjectMatter>> conSubMatMap = currentOutList.stream().collect(Collectors.groupingBy(ContractSubjectMatter::getContractId));
         conSubMatMap.forEach((key, value) -> {
      if (ArrayUtil.isNotEmpty(currentOutMap.keySet())) {
         for (Long contractId : currentOutMap.keySet()) {
            //当前出库合同出库标的物
            List<ContractSubjectMatter> contractSubjectMatterList = contractSubjectMatterMapper.selectList(Wrappers.<ContractSubjectMatter>lambdaQuery()
                  .eq(ContractSubjectMatter::getContractId, contractId).ne(ContractSubjectMatter::getDeliveryStatus, 0));
            //查询合同是否有出库阶段
            ContractPaymentSchedule schedule = scheduleMapper.selectOne(Wrappers.<ContractPaymentSchedule>lambdaQuery().eq(ContractPaymentSchedule::getContractId, key)
            ContractPaymentSchedule schedule = scheduleMapper.selectOne(Wrappers.<ContractPaymentSchedule>lambdaQuery().eq(ContractPaymentSchedule::getContractId, contractId)
                  .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)) {
               //按出库资金比例 计算发货前应收更新资金明细
               //当前出库金额
               BigDecimal outPirce = contractSubjectMatterList.stream()
                     .map(item -> (item.getDeliveryStatus() == 1 ? item.getDeliveredQuantity().multiply(item.getUnitPrice()) : item.getQuantity().multiply(item.getUnitPrice())))
                     .reduce(BigDecimal.ZERO, BigDecimal::add);
               //合同金额
               Contract contract = contractMapper.selectById(contractId);
               //出库是否完成
               if (outPirce.compareTo(contract.getAmount()) < 0) {
                  contract.setBillingStatus("1");
                  contractMapper.updateById(contract);
               }
               if (outPirce.compareTo(contract.getAmount()) == 0) {
                  contract.setBillingStatus("2");
                  contractMapper.updateById(contract);
               }
               //计算发货前应收更新资金明细
               PaymentConfirm paymentConfirm = new PaymentConfirm();
               paymentConfirm.setBusinessType("发货前应收");
               paymentConfirm.setBusGuestId(contract.getPartyAId());
@@ -174,89 +186,198 @@
               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);
                  //已存在发货应收
                  if (StrUtil.equals(lastConfirm.getBusinessType(), paymentConfirm.getBusinessType())) {
                     //查询之前所有发货前应收
                     List<PaymentConfirm> fhConfirmList = paymentConfirmMapper.selectList(Wrappers.<PaymentConfirm>lambdaQuery().eq(PaymentConfirm::getContractId, contract.getId())
                           .eq(PaymentConfirm::getScheduleId,schedule.getId()).orderByDesc(PaymentConfirm::getCreateTime));
                     BigDecimal fhTotalAmount = fhConfirmList.stream().map(item->item.getTransationAmount()).reduce(BigDecimal.ZERO,BigDecimal::add);
                     //本次发货前应收 出库总额 - 所有发货前应收
                     paymentConfirm.setTransationAmount(paymentConfirm.getTransationAmount().subtract(fhTotalAmount));
                     paymentConfirm.setReceivableAmount(paymentConfirm.getTransationAmount());
                     paymentConfirm.setTotalAmount(lastConfirm.getTotalAmount().subtract(paymentConfirm.getTransationAmount()));
                     if(schedule.getPaymentRatio().compareTo(new BigDecimal("0"))>0) {
                        paymentConfirmMapper.insert(paymentConfirm);
                     }
                  } else {
                     //不存在发货应收 出库总额
                     paymentConfirm.setTotalAmount(lastConfirm.getTotalAmount().subtract(paymentConfirm.getTransationAmount()));
                     if(schedule.getPaymentRatio().compareTo(new BigDecimal("0"))>0) {
                        paymentConfirmMapper.insert(paymentConfirm);
                     }
                  }
               } else {
                  paymentConfirmMapper.insert(paymentConfirm);
                  //无之前阶段应收
                  paymentConfirm.setTotalAmount(paymentConfirm.getTransationAmount().multiply(new BigDecimal("-1")));
                  if(schedule.getPaymentRatio().compareTo(new BigDecimal("0"))>0) {
                     paymentConfirmMapper.insert(paymentConfirm);
                     //客户付款完成合同
                     List<Contract> customerCompleteContractList = contractMapper.selectList(Wrappers.<Contract>lambdaQuery().eq(Contract::getPartyAId, contract.getPartyAId())
                           .eq(Contract::getPartyBId, SecurityUtils.getUser().getCompId()).apply(" amount = paid_amount"));
                     if (ArrayUtil.isNotEmpty(customerCompleteContractList)){
                        List<BigDecimal> outAmountList = new ArrayList<>();
                        for (Contract completeContract:customerCompleteContractList
                        ) {
                           PaymentConfirm contractLastConfirm = paymentConfirmMapper.selectOne(Wrappers.<PaymentConfirm>lambdaQuery().eq(PaymentConfirm::getContractId,completeContract.getId())
                                 .orderByDesc(PaymentConfirm::getCreateTime).last("limit 1"));
                           //查询客户付款完成合同中最后明细金额 有预收转入当前合同
                           if (contractLastConfirm.getTotalAmount().compareTo(new BigDecimal("0"))>0){
                              //新增完成付款合同 资金转出
                              PaymentConfirm outConfirm = BeanUtil.copyProperties(contractLastConfirm,PaymentConfirm.class,"id","transationAmount",
                                    "advanceAmount","receivableAmount","overdueAmount","totalAmount");
                              outConfirm.setBusinessType("资金转出");
                              outConfirm.setConfirmTime(new Date());
                              outConfirm.setTransationAmount(contractLastConfirm.getTotalAmount());
                              outConfirm.setTotalAmount(new BigDecimal("0"));
                              paymentConfirmMapper.insert(outConfirm);
                              outAmountList.add(outConfirm.getTransationAmount());
                           }
                        }
                        if (ArrayUtil.isNotEmpty(outAmountList.toArray())){
                           BigDecimal currentSum = new BigDecimal("0");
                           for (BigDecimal currentAmount:outAmountList
                           ) {
                              currentSum=currentSum.add(currentAmount);
                              //新增当前合同 资金转入
                              PaymentConfirm inConfirm = BeanUtil.copyProperties(paymentConfirm,PaymentConfirm.class,"id","transationAmount",
                                    "advanceAmount","receivableAmount","overdueAmount","totalAmount");
                              inConfirm.setBusinessType("资金转入");
                              inConfirm.setConfirmTime(new Date());
                              inConfirm.setTransationAmount(currentSum.subtract(currentAmount));
                              inConfirm.setTotalAmount(paymentConfirm.getTotalAmount().add(currentSum));
                              if (inConfirm.getTotalAmount().compareTo(new BigDecimal("0"))>0){
                                 inConfirm.setAdvanceAmount(inConfirm.getTotalAmount());
                              }else if (inConfirm.getTotalAmount().compareTo(new BigDecimal("0"))<0){
                                 inConfirm.setReceivableAmount(inConfirm.getTotalAmount().multiply(new BigDecimal("-1")));
                              }
                              paymentConfirmMapper.insert(inConfirm);
                           }
                        }
                     }
                  }
               }
               //更新发货前收款生效时间
               if (ObjUtil.isNull(schedule.getEffectiveDate())){
                  schedule.setEffectiveDate(DateUtil.offsetDay(addDTO.getOutBoundTime(),schedule.getAgreedDays()));
               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)
                  List<ContractPaymentSchedule> afterSchedule = scheduleMapper.selectList(Wrappers.<ContractPaymentSchedule>lambdaQuery().eq(ContractPaymentSchedule::getContractId, contractId)
                        .gt(ContractPaymentSchedule::getStageOrder, schedule.getStageOrder()));
                  if (ArrayUtil.isEmpty(afterSchedule.toArray())){
                  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()));
                  if (ObjUtil.isNotNull(beforeSchedule)) {
                     beforeSchedule.setEffectiveEndDate(schedule.getEffectiveDate());
                     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);
               }
               //更新履约
               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){
                     .gt(ContractPaymentSchedule::getStageOrder, schedule.getStageOrder())
                     .orderByAsc(ContractPaymentSchedule::getCreateTime));
               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();
                  if (StrUtil.equals(endSchedule.getStageName(), "质保金")) {
                     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()));
                     if(endSchedule.getPaymentRatio().compareTo(new BigDecimal("0"))>0) {
                        paymentConfirmMapper.insert(newConfim);
                     }
                  }
                  newConfim.setTotalAmount(lastNewTotal.subtract(newConfim.getReceivableAmount()));
                  paymentConfirmMapper.insert(newConfim);
                  //更新合同下个阶段
                  contract.setNextScheduleName(endSchedule.getStageName());
                  contractMapper.updateById(contract);
               }
               if(ArrayUtil.isEmpty(afterSchedule.toArray())){
               if (ArrayUtil.isNotEmpty(afterSchedule.toArray()) && afterSchedule.size() > 1) {
                  //更新合同下个阶段
                  contract.setNextScheduleName(afterSchedule.get(0).getStageName());
                  contractMapper.updateById(contract);
               }
               if (ArrayUtil.isEmpty(afterSchedule.toArray())) {
                  schedule.setEffectiveEndDate(contract.getExpirationDate());
                  scheduleMapper.updateById(schedule);
                  //更新合同下个阶段
                  contract.setNextScheduleName("无");
                  contractMapper.updateById(contract);
                  //当前为合同最后阶段
                  PaymentConfirm lastNewConfirm = paymentConfirmMapper.selectOne(Wrappers.<PaymentConfirm>lambdaQuery().eq(PaymentConfirm::getContractId,contract.getId())
                        .orderByDesc(PaymentConfirm::getCreateTime).last("limit 1"));
                  if (lastNewConfirm.getTotalAmount().compareTo(new BigDecimal("0"))>=0){
                     //有预付且超过应收 关闭合同状态
                     contract.setContractStatus(3);
                     contractMapper.updateById(contract);
                  }
               }
            }
         });
         }
      }
      return R.ok();
   }
   private void saveContractOutBound(Contract contract, OutBoundAddDTO addDTO, ContractSubjectMatter subjectMatter,Long outBoundId) {
      ContractOutBound contractOutBound = new ContractOutBound();
      contractOutBound.setContractId(contract.getId());
      contractOutBound.setContractName(contract.getContractName());
      contractOutBound.setBusGuestId(contract.getPartyAId());
      contractOutBound.setBusGuestName(contract.getPartyA());
      contractOutBound.setSubjectMatterCode(subjectMatter.getMaterialCode());
      contractOutBound.setSubjectMatterName(subjectMatter.getMaterialInternalName());
      contractOutBound.setOutBoundTime(addDTO.getOutBoundTime());
      contractOutBound.setOutBoundNum(subjectMatter.getLastDeliveredQuantity());
      contractOutBound.setOutBoundAttNames(addDTO.getOutBoundAttNames());
      contractOutBound.setOutBoundAttPaths(addDTO.getOutBoundAttPaths());
      contractOutBound.setOutBoundId(outBoundId);
      contractOutBoundMapper.insert(contractOutBound);
   }
}