package com.ruoyi.sales.controller; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.toolkit.ObjectUtils; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.ruoyi.common.utils.poi.ExcelUtil; import com.ruoyi.framework.aspectj.lang.annotation.Log; import com.ruoyi.framework.aspectj.lang.enums.BusinessType; import com.ruoyi.framework.web.controller.BaseController; import com.ruoyi.framework.web.domain.AjaxResult; import com.ruoyi.framework.web.domain.R; import com.ruoyi.framework.web.page.TableDataInfo; import com.ruoyi.sales.dto.InvoiceLedgerDto; import com.ruoyi.sales.dto.SalesLedgerDto; import com.ruoyi.sales.mapper.InvoiceLedgerMapper; import com.ruoyi.sales.mapper.ReceiptPaymentMapper; import com.ruoyi.sales.pojo.ReceiptPayment; import com.ruoyi.sales.pojo.SalesLedger; import com.ruoyi.sales.service.ICommonFileService; import com.ruoyi.sales.service.ISalesLedgerService; import io.swagger.annotations.Api; import io.swagger.annotations.ApiOperation; import io.swagger.annotations.ApiParam; import lombok.AllArgsConstructor; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.util.CollectionUtils; import org.springframework.web.bind.annotation.*; import org.springframework.web.multipart.MultipartFile; import javax.servlet.http.HttpServletResponse; import java.io.FileNotFoundException; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.math.BigDecimal; import java.net.URLEncoder; import java.util.*; import java.util.stream.Collectors; /** * 销售台账Controller * * @author ruoyi * @date 2025-05-08 */ @RestController @RequestMapping("/sales/ledger") @AllArgsConstructor @Api(tags = "销售台账") @Slf4j public class SalesLedgerController extends BaseController { private ISalesLedgerService salesLedgerService; private ICommonFileService commonFileService; @Autowired private InvoiceLedgerMapper invoiceLedgerMapper; @Autowired private ReceiptPaymentMapper receiptPaymentMapper; /** * 导入销售台账 */ @Log(title = "导入销售台账", businessType = BusinessType.INSERT) @PostMapping("/import") @ApiOperation("导入销售台账") public AjaxResult importData(@RequestParam("file") @ApiParam(value = "Excel文件", required = true) MultipartFile file) { return salesLedgerService.importData(file); } @ApiOperation("导出销售台账模板") @PostMapping("/exportTemplate") public void exportTemplate(HttpServletResponse response) { // 1. 模板文件在resources/static下的路径 String templatePath = "static/销售台账导入模板.xlsx"; // 2. 获取模板文件的输入流 try (InputStream inputStream = this.getClass().getClassLoader().getResourceAsStream(templatePath)) { if (inputStream == null) { throw new FileNotFoundException("模板文件不存在:" + templatePath); } // 3. 设置响应头,触发浏览器下载 response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"); response.setCharacterEncoding("utf-8"); String fileName = URLEncoder.encode("销售台账导入模板.xlsx", "utf-8").replaceAll("\\+", "%20"); response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName); // 4. 将模板文件写入响应输出流 try (OutputStream outputStream = response.getOutputStream()) { byte[] buffer = new byte[1024]; int len; while ((len = inputStream.read(buffer)) > 0) { outputStream.write(buffer, 0, len); } outputStream.flush(); } } catch (IOException e) { log.error("导出销售台账模板失败", e); // 若模板文件读取失败,返回错误提示 try { response.getWriter().write("模板导出失败:" + e.getMessage()); } catch (IOException ex) { log.error("响应输出错误", ex); } } } /** * 查询销售台账列表 */ @GetMapping("/list") public TableDataInfo list(Page page, SalesLedgerDto salesLedgerDto) { startPage(); List list = salesLedgerService.selectSalesLedgerList(salesLedgerDto); // 计算已开票金额/未开票金额(已填写发票金额为准) if(CollectionUtils.isEmpty(list)){ return getDataTable(list); } List salesLedgerIds = list.stream().map(SalesLedger::getId).collect(Collectors.toList()); List invoiceLedgerDtoList = invoiceLedgerMapper.invoicedTotal(salesLedgerIds); if(CollectionUtils.isEmpty(invoiceLedgerDtoList)){ return getDataTable(list); } for (SalesLedger salesLedger : list) { for (InvoiceLedgerDto invoiceLedgerDto : invoiceLedgerDtoList) { if (salesLedger.getId().intValue() == invoiceLedgerDto.getSalesLedgerId()) { BigDecimal noInvoiceAmountTotal = salesLedger.getContractAmount().subtract(invoiceLedgerDto.getInvoiceTotal()); salesLedger.setNoInvoiceAmountTotal(noInvoiceAmountTotal); } } } return getDataTable(list); } /** * 查询销售台账和产品父子列表 */ @GetMapping("/getSalesLedgerWithProducts") public SalesLedgerDto getSalesLedgerWithProducts(SalesLedgerDto salesLedgerDto) { return salesLedgerService.getSalesLedgerWithProducts(salesLedgerDto); } /** * 导出销售台账列表 */ @Log(title = "销售台账", businessType = BusinessType.EXPORT) @PostMapping("/export") public void export(HttpServletResponse response, SalesLedgerDto salesLedgerDto) { Page page = new Page(-1,-1); IPage salesLedgerIPage = listPage(page, salesLedgerDto); ExcelUtil util = new ExcelUtil(SalesLedger.class); if(salesLedgerIPage == null){ util.exportExcel(response, new ArrayList<>(), "销售台账数据"); return; } List list = salesLedgerIPage.getRecords(); util.exportExcel(response, list, "销售台账数据"); } /** * 导出开票登记列表 */ @Log(title = "导出开票登记列表", businessType = BusinessType.EXPORT) @PostMapping("/exportOne") public void exportOne(HttpServletResponse response, SalesLedgerDto salesLedgerDto) { Page page = new Page(); page.setCurrent(-1); page.setSize(-1); IPage salesLedgerIPage = listPage(page, salesLedgerDto); ExcelUtil util = new ExcelUtil(SalesLedger.class); util.exportExcel(response, salesLedgerIPage == null ? new ArrayList<>() : salesLedgerIPage.getRecords(), "导出开票登记列表"); } /** * 新增修改销售台账 */ @Log(title = "销售台账", businessType = BusinessType.INSERT) @PostMapping("/addOrUpdateSalesLedger") public AjaxResult add(@RequestBody SalesLedgerDto salesLedgerDto) { return toAjax(salesLedgerService.addOrUpdateSalesLedger(salesLedgerDto)); } /** * 删除销售台账 */ @Log(title = "销售台账", businessType = BusinessType.DELETE) @DeleteMapping("/delLedger") public AjaxResult remove(@RequestBody Long[] ids) { if (ids == null || ids.length == 0) { return AjaxResult.error("请传入要删除的ID"); } return toAjax(salesLedgerService.deleteSalesLedgerByIds(ids)); } /** * 查询销售台账不分页 * * @param salesLedgerDto * @return */ @GetMapping("/listNoPage") public AjaxResult listNoPage(SalesLedgerDto salesLedgerDto) { List list = salesLedgerService.selectSalesLedgerList(salesLedgerDto); return AjaxResult.success(list); } /** * 销售台账附件删除 */ @Log(title = "销售台账附件删除", businessType = BusinessType.DELETE) @DeleteMapping("/delLedgerFile") public AjaxResult delLedgerFile(@RequestBody Long[] ids) { if (ids == null || ids.length == 0) { return AjaxResult.error("请传入要删除的ID"); } return toAjax(commonFileService.deleteSalesLedgerByIds(ids)); } /** * 本月销售合同金额 */ @GetMapping("/getContractAmount") public AjaxResult getContractAmount() { try { BigDecimal contractAmount = salesLedgerService.getContractAmount(); return AjaxResult.success(contractAmount != null ? contractAmount : BigDecimal.ZERO); } catch (Exception e) { return AjaxResult.error("获取合同金额失败:" + e.getMessage()); } } /** * 客户合同金额TOP5统计 */ @GetMapping("/getTopFiveList") public AjaxResult getTopFiveList() { return AjaxResult.success(salesLedgerService.getTopFiveList()); } /** * 近半年开票,回款金额 */ @GetMapping("/getAmountHalfYear") public AjaxResult getAmountHalfYear(@RequestParam(value = "type",defaultValue = "1") Integer type) { return AjaxResult.success(salesLedgerService.getAmountHalfYear(type)); } /** * 查询销售台账列表 */ @GetMapping("/listPage") public IPage listPage(Page page, SalesLedgerDto salesLedgerDto) { IPage iPage = salesLedgerService.selectSalesLedgerListPage(page, salesLedgerDto); // 查询结果为空,直接返回 if (CollectionUtils.isEmpty(iPage.getRecords())) { return iPage; } // 获取当前页所有台账记录的 ID 集合 List salesLedgerIds = iPage.getRecords().stream().map(SalesLedger::getId).collect(Collectors.toList()); // 查询发票信息的已开票金额 List invoiceLedgerDtoList = invoiceLedgerMapper.invoicedTotal(salesLedgerIds); if (CollectionUtils.isEmpty(invoiceLedgerDtoList)) { invoiceLedgerDtoList = Collections.emptyList(); } // 转换发票数据, key 为台账ID, value 为该台账的总开票金额 Map invoiceTotals = invoiceLedgerDtoList.stream() .filter(dto -> dto.getSalesLedgerId() != null && dto.getInvoiceTotal() != null) .collect(Collectors.toMap( dto -> dto.getSalesLedgerId().longValue(), InvoiceLedgerDto::getInvoiceTotal, BigDecimal::add // 存在重复ID执行累加 )); // 查询回款/付款记录 List receiptPayments = Collections.emptyList(); if (!CollectionUtils.isEmpty(salesLedgerIds)) { receiptPayments = receiptPaymentMapper.selectList(new LambdaQueryWrapper() .in(ReceiptPayment::getSalesLedgerId, salesLedgerIds)); } // 转换回款数据, key 为台账ID, value 为该台账的总回款金额 Map receiptTotals = new HashMap<>(); if (!CollectionUtils.isEmpty(receiptPayments)) { for (ReceiptPayment receiptPayment : receiptPayments) { if (receiptPayment.getSalesLedgerId() != null && receiptPayment.getReceiptPaymentAmount() != null) { // 如果 key 存在则相加,不存在则放入 receiptTotals.merge(receiptPayment.getSalesLedgerId(), receiptPayment.getReceiptPaymentAmount(), BigDecimal::add); } } } for (SalesLedger salesLedger : iPage.getRecords()) { Long ledgerId = salesLedger.getId(); // 合同总金额 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 (ObjectUtils.isNotEmpty(salesLedgerDto.getStatus())) { if (salesLedgerDto.getStatus()) { // 清除所有“未开票金额”为 0 的记录 iPage.getRecords().removeIf(salesLedger -> Objects.equals(salesLedger.getNoInvoiceAmountTotal(), new BigDecimal("0.00"))); iPage.setTotal(iPage.getRecords().size()); } } return iPage; } @ApiOperation("查询销售台账消耗物料信息") @GetMapping("/getSalesLedgerWithProductsLoss") public R getSalesLedgerWithProductsLoss(Long salesLedgerId) { return R.ok(salesLedgerService.getSalesLedgerWithProductsLoss(salesLedgerId)); } }