From fca21683e1b5b906d2514082ddfbae8eb820c9ea Mon Sep 17 00:00:00 2001
From: shiyunteng <shiyunteng@example.com>
Date: 星期五, 15 五月 2026 17:12:55 +0800
Subject: [PATCH] feat:合同模版、出库标的物、合同管理拆分

---
 platformx-business-finance-biz/src/main/java/com/by4cloud/platformx/business/service/impl/OutBoundServiceImpl.java |  235 +++++++++++++++++++++++++++++++++-------------------------
 1 files changed, 132 insertions(+), 103 deletions(-)

diff --git a/platformx-business-finance-biz/src/main/java/com/by4cloud/platformx/business/service/impl/OutBoundServiceImpl.java b/platformx-business-finance-biz/src/main/java/com/by4cloud/platformx/business/service/impl/OutBoundServiceImpl.java
index 98b4924..252dee9 100644
--- a/platformx-business-finance-biz/src/main/java/com/by4cloud/platformx/business/service/impl/OutBoundServiceImpl.java
+++ b/platformx-business-finance-biz/src/main/java/com/by4cloud/platformx/business/service/impl/OutBoundServiceImpl.java
@@ -19,10 +19,7 @@
 
 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;
 
 /**
@@ -50,7 +47,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()))
+				.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)
@@ -78,64 +75,67 @@
 			return R.failed("鍑哄簱鏁伴噺瓒呭嚭鍚堝悓璁㈠崟鏁伴噺");
 		}
 		//褰撳墠鍏ュ簱鏍囩殑鐗�
-		List<ContractSubjectMatter> currentOutList = new ArrayList<>();
+		Map<Long, Object> currentOutMap = new HashMap<>();
+		//鍑哄簱
 		addDTO.getSubjectMatterList().stream().forEach(outSubjectMatterAddDTO -> {
 			BigDecimal remainNum = outSubjectMatterAddDTO.getOutBoundNum();
 			for (ContractSubjectMatter subjectMatter : subjectMatterList) {
-				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);
-						currentOutList.add(subjectMatter);
-						//鏂板鍚堝悓鍑哄簱鍘嗗彶
-						saveContractOutBound(contract,addDTO,subjectMatter);
-					} 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);
-						currentOutList.add(subjectMatter);
-						//鏂板鍚堝悓鍑哄簱鍘嗗彶
-						saveContractOutBound(contract,addDTO,subjectMatter);
-						break;
-					}
+				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);
+						} 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);
+							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);
-						currentOutList.add(subjectMatter);
-						//鏂板鍚堝悓鍑哄簱鍘嗗彶
-						saveContractOutBound(contract,addDTO,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);
-						//鏂板鍚堝悓鍑哄簱鍘嗗彶
-						saveContractOutBound(contract,addDTO,subjectMatter);
-						break;
+						//鏈氦浠樼姸鎬佷笅
+						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);
+						} 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);
+							break;
+						}
 					}
 				}
 			}
@@ -147,32 +147,32 @@
 		});
 
 		//鍑哄簱搴旀敹娆炬槑缁�
-		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.getDeliveryStatus()==1?item.getDeliveredQuantity():item.getQuantity()).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());
@@ -183,53 +183,74 @@
 					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()));
+							paymentConfirmMapper.insert(paymentConfirm);
+						} else {
+							//涓嶅瓨鍦ㄥ彂璐у簲鏀� 鍑哄簱鎬婚
+							paymentConfirm.setTotalAmount(lastConfirm.getTotalAmount().subtract(paymentConfirm.getTransationAmount()));
+							paymentConfirmMapper.insert(paymentConfirm);
+						}
 					} else {
+						//鏃犱箣鍓嶉樁娈靛簲鏀�
+						paymentConfirm.setTotalAmount(paymentConfirm.getTransationAmount().multiply(new BigDecimal("-1")));
 						paymentConfirmMapper.insert(paymentConfirm);
 					}
+
 					//鏇存柊鍙戣揣鍓嶆敹娆剧敓鏁堟椂闂�
-					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())
 							.orderByAsc(ContractPaymentSchedule::getCreateTime));
-					if (ArrayUtil.isNotEmpty(afterSchedule.toArray())&&afterSchedule.size()==1){
+					if (ArrayUtil.isNotEmpty(afterSchedule.toArray()) && afterSchedule.size() == 1) {
 						//鏈�鍚庨樁娈电敓鏁堟椂闂�
 						ContractPaymentSchedule endSchedule = afterSchedule.get(0);
-						if (StrUtil.equals(endSchedule.getStageName(),"璐ㄤ繚閲�")) {
+						if (StrUtil.equals(endSchedule.getStageName(), "璐ㄤ繚閲�")) {
 							endSchedule.setEffectiveDate(DateUtil.offsetDay(addDTO.getOutBoundTime(), endSchedule.getAgreedDays()));
 							endSchedule.setEffectiveEndDate(contract.getExpirationDate());
 							scheduleMapper.updateById(endSchedule);
@@ -263,20 +284,28 @@
 						contract.setNextScheduleName(endSchedule.getStageName());
 						contractMapper.updateById(contract);
 					}
-					if(ArrayUtil.isNotEmpty(afterSchedule.toArray())&&afterSchedule.size()>1){
+					if (ArrayUtil.isNotEmpty(afterSchedule.toArray()) && afterSchedule.size() > 1) {
 						//鏇存柊鍚堝悓涓嬩釜闃舵
 						contract.setNextScheduleName(afterSchedule.get(0).getStageName());
 						contractMapper.updateById(contract);
 					}
-					if(ArrayUtil.isEmpty(afterSchedule.toArray())){
+					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();

--
Gitblit v1.9.1