package cn.iocoder.yudao.module.erp.service.finance; import cn.hutool.core.collection.CollUtil; import cn.hutool.core.lang.Assert; import cn.hutool.core.util.ObjectUtil; import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.util.collection.CollectionUtils; import cn.iocoder.yudao.framework.common.util.object.BeanUtils; import cn.iocoder.yudao.module.erp.controller.admin.finance.vo.receipt.ErpFinanceReceiptPageReqVO; import cn.iocoder.yudao.module.erp.controller.admin.finance.vo.receipt.ErpFinanceReceiptSaveReqVO; import cn.iocoder.yudao.module.erp.dal.dataobject.finance.ErpFinanceReceiptDO; import cn.iocoder.yudao.module.erp.dal.dataobject.finance.ErpFinanceReceiptItemDO; import cn.iocoder.yudao.module.erp.dal.dataobject.sale.ErpSaleOutDO; import cn.iocoder.yudao.module.erp.dal.dataobject.sale.ErpSaleReturnDO; import cn.iocoder.yudao.module.erp.dal.mysql.finance.ErpFinanceReceiptItemMapper; import cn.iocoder.yudao.module.erp.dal.mysql.finance.ErpFinanceReceiptMapper; import cn.iocoder.yudao.module.erp.dal.redis.no.ErpNoRedisDAO; import cn.iocoder.yudao.module.erp.enums.ErpAuditStatus; import cn.iocoder.yudao.module.erp.enums.common.ErpBizTypeEnum; import cn.iocoder.yudao.module.erp.service.sale.ErpCustomerService; import cn.iocoder.yudao.module.erp.service.sale.ErpSaleOutService; import cn.iocoder.yudao.module.erp.service.sale.ErpSaleReturnService; import cn.iocoder.yudao.module.system.api.user.AdminUserApi; import jakarta.annotation.Resource; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import org.springframework.validation.annotation.Validated; import java.math.BigDecimal; import java.util.Collection; import java.util.Collections; import java.util.List; import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception; import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.*; import static cn.iocoder.yudao.module.erp.enums.ErrorCodeConstants.*; // TODO 芋艿:记录操作日志 /** * ERP 收款单 Service 实现类 * * @author 芋道源码 */ @Service @Validated public class ErpFinanceReceiptServiceImpl implements ErpFinanceReceiptService { @Resource private ErpFinanceReceiptMapper financeReceiptMapper; @Resource private ErpFinanceReceiptItemMapper financeReceiptItemMapper; @Resource private ErpNoRedisDAO noRedisDAO; @Resource private ErpCustomerService customerService; @Resource private ErpAccountService accountService; @Resource private ErpSaleOutService saleOutService; @Resource private ErpSaleReturnService saleReturnService; @Resource private AdminUserApi adminUserApi; @Override @Transactional(rollbackFor = Exception.class) public Long createFinanceReceipt(ErpFinanceReceiptSaveReqVO createReqVO) { // 1.1 校验订单项的有效性 List receiptItems = validateFinanceReceiptItems( createReqVO.getCustomerId(), createReqVO.getItems()); // 1.2 校验客户 customerService.validateCustomer(createReqVO.getCustomerId()); // 1.3 校验结算账户 if (createReqVO.getAccountId() != null) { accountService.validateAccount(createReqVO.getAccountId()); } // 1.4 校验财务人员 if (createReqVO.getFinanceUserId() != null) { adminUserApi.validateUser(createReqVO.getFinanceUserId()); } // 1.5 生成收款单号,并校验唯一性 String no = noRedisDAO.generate(ErpNoRedisDAO.FINANCE_RECEIPT_NO_PREFIX); if (financeReceiptMapper.selectByNo(no) != null) { throw exception(FINANCE_RECEIPT_NO_EXISTS); } // 2.1 插入收款单 ErpFinanceReceiptDO receipt = BeanUtils.toBean(createReqVO, ErpFinanceReceiptDO.class, in -> in .setNo(no).setStatus(ErpAuditStatus.PROCESS.getStatus())); calculateTotalPrice(receipt, receiptItems); financeReceiptMapper.insert(receipt); // 2.2 插入收款单项 receiptItems.forEach(o -> o.setReceiptId(receipt.getId())); financeReceiptItemMapper.insertBatch(receiptItems); // 3. 更新销售出库、退货的收款金额情况 updateSalePrice(receiptItems); return receipt.getId(); } @Override @Transactional(rollbackFor = Exception.class) public void updateFinanceReceipt(ErpFinanceReceiptSaveReqVO updateReqVO) { // 1.1 校验存在 ErpFinanceReceiptDO receipt = validateFinanceReceiptExists(updateReqVO.getId()); if (ErpAuditStatus.APPROVE.getStatus().equals(receipt.getStatus())) { throw exception(FINANCE_RECEIPT_UPDATE_FAIL_APPROVE, receipt.getNo()); } // 1.2 校验客户 customerService.validateCustomer(updateReqVO.getCustomerId()); // 1.3 校验结算账户 if (updateReqVO.getAccountId() != null) { accountService.validateAccount(updateReqVO.getAccountId()); } // 1.4 校验财务人员 if (updateReqVO.getFinanceUserId() != null) { adminUserApi.validateUser(updateReqVO.getFinanceUserId()); } // 1.5 校验收款单项的有效性 List receiptItems = validateFinanceReceiptItems( updateReqVO.getCustomerId(), updateReqVO.getItems()); // 2.1 更新收款单 ErpFinanceReceiptDO updateObj = BeanUtils.toBean(updateReqVO, ErpFinanceReceiptDO.class); calculateTotalPrice(updateObj, receiptItems); financeReceiptMapper.updateById(updateObj); // 2.2 更新收款单项 updateFinanceReceiptItemList(updateReqVO.getId(), receiptItems); } private void calculateTotalPrice(ErpFinanceReceiptDO receipt, List receiptItems) { receipt.setTotalPrice(getSumValue(receiptItems, ErpFinanceReceiptItemDO::getReceiptPrice, BigDecimal::add, BigDecimal.ZERO)); receipt.setReceiptPrice(receipt.getTotalPrice().subtract(receipt.getDiscountPrice())); } @Override @Transactional(rollbackFor = Exception.class) public void updateFinanceReceiptStatus(Long id, Integer status) { boolean approve = ErpAuditStatus.APPROVE.getStatus().equals(status); // 1.1 校验存在 ErpFinanceReceiptDO receipt = validateFinanceReceiptExists(id); // 1.2 校验状态 if (receipt.getStatus().equals(status)) { throw exception(approve ? FINANCE_RECEIPT_APPROVE_FAIL : FINANCE_RECEIPT_PROCESS_FAIL); } // 2. 更新状态 int updateCount = financeReceiptMapper.updateByIdAndStatus(id, receipt.getStatus(), new ErpFinanceReceiptDO().setStatus(status)); if (updateCount == 0) { throw exception(approve ? FINANCE_RECEIPT_APPROVE_FAIL : FINANCE_RECEIPT_PROCESS_FAIL); } } private List validateFinanceReceiptItems( Long customerId, List list) { return convertList(list, o -> BeanUtils.toBean(o, ErpFinanceReceiptItemDO.class, item -> { if (ObjectUtil.equal(item.getBizType(), ErpBizTypeEnum.SALE_OUT.getType())) { ErpSaleOutDO saleOut = saleOutService.validateSaleOut(item.getBizId()); Assert.equals(saleOut.getCustomerId(), customerId, "客户必须相同"); item.setTotalPrice(saleOut.getTotalPrice()).setBizNo(saleOut.getNo()); } else if (ObjectUtil.equal(item.getBizType(), ErpBizTypeEnum.SALE_RETURN.getType())) { ErpSaleReturnDO saleReturn = saleReturnService.validateSaleReturn(item.getBizId()); Assert.equals(saleReturn.getCustomerId(), customerId, "客户必须相同"); item.setTotalPrice(saleReturn.getTotalPrice().negate()).setBizNo(saleReturn.getNo()); } else { throw new IllegalArgumentException("业务类型不正确:" + item.getBizType()); } })); } private void updateFinanceReceiptItemList(Long id, List newList) { // 第一步,对比新老数据,获得添加、修改、删除的列表 List oldList = financeReceiptItemMapper.selectListByReceiptId(id); List> diffList = diffList(oldList, newList, // id 不同,就认为是不同的记录 (oldVal, newVal) -> oldVal.getId().equals(newVal.getId())); // 第二步,批量添加、修改、删除 if (CollUtil.isNotEmpty(diffList.get(0))) { diffList.get(0).forEach(o -> o.setReceiptId(id)); financeReceiptItemMapper.insertBatch(diffList.get(0)); } if (CollUtil.isNotEmpty(diffList.get(1))) { financeReceiptItemMapper.updateBatch(diffList.get(1)); } if (CollUtil.isNotEmpty(diffList.get(2))) { financeReceiptItemMapper.deleteBatchIds(convertList(diffList.get(2), ErpFinanceReceiptItemDO::getId)); } // 第三步,更新销售出库、退货的收款金额情况 updateSalePrice(CollectionUtils.newArrayList(diffList)); } private void updateSalePrice(List receiptItems) { receiptItems.forEach(receiptItem -> { BigDecimal totalReceiptPrice = financeReceiptItemMapper.selectReceiptPriceSumByBizIdAndBizType( receiptItem.getBizId(), receiptItem.getBizType()); if (ErpBizTypeEnum.SALE_OUT.getType().equals(receiptItem.getBizType())) { saleOutService.updateSaleInReceiptPrice(receiptItem.getBizId(), totalReceiptPrice); } else if (ErpBizTypeEnum.SALE_RETURN.getType().equals(receiptItem.getBizType())) { saleReturnService.updateSaleReturnRefundPrice(receiptItem.getBizId(), totalReceiptPrice.negate()); } else { throw new IllegalArgumentException("业务类型不正确:" + receiptItem.getBizType()); } }); } @Override @Transactional(rollbackFor = Exception.class) public void deleteFinanceReceipt(List ids) { // 1. 校验不处于已审批 List receipts = financeReceiptMapper.selectBatchIds(ids); if (CollUtil.isEmpty(receipts)) { return; } receipts.forEach(receipt -> { if (ErpAuditStatus.APPROVE.getStatus().equals(receipt.getStatus())) { throw exception(FINANCE_RECEIPT_DELETE_FAIL_APPROVE, receipt.getNo()); } }); // 2. 遍历删除,并记录操作日志 receipts.forEach(receipt -> { // 2.1 删除收款单 financeReceiptMapper.deleteById(receipt.getId()); // 2.2 删除收款单项 List receiptItems = financeReceiptItemMapper.selectListByReceiptId(receipt.getId()); financeReceiptItemMapper.deleteBatchIds(convertSet(receiptItems, ErpFinanceReceiptItemDO::getId)); // 2.3 更新销售出库、退货的收款金额情况 updateSalePrice(receiptItems); }); } private ErpFinanceReceiptDO validateFinanceReceiptExists(Long id) { ErpFinanceReceiptDO receipt = financeReceiptMapper.selectById(id); if (receipt == null) { throw exception(FINANCE_RECEIPT_NOT_EXISTS); } return receipt; } @Override public ErpFinanceReceiptDO getFinanceReceipt(Long id) { return financeReceiptMapper.selectById(id); } @Override public PageResult getFinanceReceiptPage(ErpFinanceReceiptPageReqVO pageReqVO) { return financeReceiptMapper.selectPage(pageReqVO); } // ==================== 收款单项 ==================== @Override public List getFinanceReceiptItemListByReceiptId(Long receiptId) { return financeReceiptItemMapper.selectListByReceiptId(receiptId); } @Override public List getFinanceReceiptItemListByReceiptIds(Collection receiptIds) { if (CollUtil.isEmpty(receiptIds)) { return Collections.emptyList(); } return financeReceiptItemMapper.selectListByReceiptIds(receiptIds); } }