|  | @@ -29,6 +29,7 @@ import com.rtrh.common.util.StringUtil;
 | 
	
		
			
				|  |  |  import com.rtrh.core.vo.ListMessage;
 | 
	
		
			
				|  |  |  import com.rtrh.projects.modules.projects.dao.SubSourceDao;
 | 
	
		
			
				|  |  |  import com.rtrh.projects.modules.projects.enums.ProjectStatusEnum;
 | 
	
		
			
				|  |  | +import com.rtrh.projects.modules.projects.enums.SubInfoStatusEnum;
 | 
	
		
			
				|  |  |  import com.rtrh.projects.modules.projects.po.SubInfoGxj;
 | 
	
		
			
				|  |  |  import com.rtrh.projects.modules.projects.vo.SubInfoTotalExcel;
 | 
	
		
			
				|  |  |  import com.rtrh.projects.modules.projects.vo.export.*;
 | 
	
	
		
			
				|  | @@ -791,6 +792,245 @@ public class SubInfoExportController extends BaseController {
 | 
	
		
			
				|  |  |              }
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | +    /**
 | 
	
		
			
				|  |  | +     * 导出明细
 | 
	
		
			
				|  |  | +     *
 | 
	
		
			
				|  |  | +     * @param response
 | 
	
		
			
				|  |  | +     */
 | 
	
		
			
				|  |  | +    @PostMapping("/exportDetailExcel")
 | 
	
		
			
				|  |  | +    public void exportDetailExcel(HttpServletResponse response, @RequestBody SubInfoQueryTzVO queryVO) {
 | 
	
		
			
				|  |  | +        List<SubInfoGxj> list = subInfoQueryService.queryAll(getCurUser().getLoginUser(), queryVO);
 | 
	
		
			
				|  |  | +        ExportParams exportParams = new ExportParams();
 | 
	
		
			
				|  |  | +        exportParams.setType(ExcelType.XSSF);
 | 
	
		
			
				|  |  | +        Workbook workbook;
 | 
	
		
			
				|  |  | +        if (SubInfoStatusEnum.CB.getCode().equals(queryVO.getStatus())){
 | 
	
		
			
				|  |  | +            List<SubInfoCbVO> cbVOList = list.stream().map(e -> {
 | 
	
		
			
				|  |  | +                SubInfoCbVO subInfoCbVO = new SubInfoCbVO();
 | 
	
		
			
				|  |  | +                BeanUtil.copyProperties(e, subInfoCbVO);
 | 
	
		
			
				|  |  | +                return subInfoCbVO;
 | 
	
		
			
				|  |  | +            }).collect(Collectors.toList());
 | 
	
		
			
				|  |  | +            exportParams.setTitle("“四个一批”工业项目表(储备项目库)");
 | 
	
		
			
				|  |  | +            exportParams.setSheetName("储备项目库");
 | 
	
		
			
				|  |  | +            workbook = ExcelExportUtil.exportExcel(exportParams, SubInfoCbVO.class, cbVOList);
 | 
	
		
			
				|  |  | +        }else if (SubInfoStatusEnum.XJ.getCode().equals(queryVO.getStatus())){
 | 
	
		
			
				|  |  | +            List<SubInfoXjVO> xjVOList = list.stream().map(e -> {
 | 
	
		
			
				|  |  | +                SubInfoXjVO subInfoTzVO = new SubInfoXjVO();
 | 
	
		
			
				|  |  | +                BeanUtil.copyProperties(e, subInfoTzVO);
 | 
	
		
			
				|  |  | +                return subInfoTzVO;
 | 
	
		
			
				|  |  | +            }).collect(Collectors.toList());
 | 
	
		
			
				|  |  | +            exportParams.setTitle("“四个一批”工业项目表(新建项目库)");
 | 
	
		
			
				|  |  | +            exportParams.setSheetName("新建项目库");
 | 
	
		
			
				|  |  | +            workbook = ExcelExportUtil.exportExcel(exportParams, SubInfoXjVO.class, xjVOList);
 | 
	
		
			
				|  |  | +        }else if (SubInfoStatusEnum.ZJ.getCode().equals(queryVO.getStatus())){
 | 
	
		
			
				|  |  | +            List<SubInfoZjVO> zjVoList = list.stream().map(e -> {
 | 
	
		
			
				|  |  | +                SubInfoZjVO subInfoTzVO = new SubInfoZjVO();
 | 
	
		
			
				|  |  | +                BeanUtil.copyProperties(e, subInfoTzVO);
 | 
	
		
			
				|  |  | +                return subInfoTzVO;
 | 
	
		
			
				|  |  | +            }).collect(Collectors.toList());
 | 
	
		
			
				|  |  | +            exportParams.setTitle("“四个一批”工业项目表(在建项目库)");
 | 
	
		
			
				|  |  | +            exportParams.setSheetName("在建项目库");
 | 
	
		
			
				|  |  | +            workbook = ExcelExportUtil.exportExcel(exportParams, SubInfoZjVO.class, zjVoList);
 | 
	
		
			
				|  |  | +        }else if (SubInfoStatusEnum.TC.getCode().equals(queryVO.getStatus())){
 | 
	
		
			
				|  |  | +            List<SubInfoTcVO> tcList = list.stream().map(e -> {
 | 
	
		
			
				|  |  | +                SubInfoTcVO subInfoTzVO = new SubInfoTcVO();
 | 
	
		
			
				|  |  | +                BeanUtil.copyProperties(e, subInfoTzVO);
 | 
	
		
			
				|  |  | +                return subInfoTzVO;
 | 
	
		
			
				|  |  | +            }).collect(Collectors.toList());
 | 
	
		
			
				|  |  | +            exportParams.setTitle("“四个一批”工业项目表(投产项目库)");
 | 
	
		
			
				|  |  | +            exportParams.setSheetName("投产项目库");
 | 
	
		
			
				|  |  | +            workbook = ExcelExportUtil.exportExcel(exportParams, SubInfoTcVO.class, tcList);
 | 
	
		
			
				|  |  | +        }else {
 | 
	
		
			
				|  |  | +            List<SubInfoZkVO> zkVOList = list.stream().map(e -> {
 | 
	
		
			
				|  |  | +                SubInfoZkVO subInfoTzVO = new SubInfoZkVO();
 | 
	
		
			
				|  |  | +                BeanUtil.copyProperties(e, subInfoTzVO);
 | 
	
		
			
				|  |  | +                return subInfoTzVO;
 | 
	
		
			
				|  |  | +            }).collect(Collectors.toList());
 | 
	
		
			
				|  |  | +            exportParams.setTitle("“四个一批”工业项目表(项目总库)");
 | 
	
		
			
				|  |  | +            exportParams.setSheetName("项目总库");
 | 
	
		
			
				|  |  | +            workbook = ExcelExportUtil.exportExcel(exportParams, SubInfoZkVO.class, zkVOList);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        response.setContentType("application/xlsx;charset=utf-8");
 | 
	
		
			
				|  |  | +        response.setHeader("Content-Disposition", "attachment; filename=" + URLEncodeUtil.encode("项目明细表") + System.currentTimeMillis() + ".xlsx");
 | 
	
		
			
				|  |  | +        try (OutputStream outputStream = response.getOutputStream()) {
 | 
	
		
			
				|  |  | +            workbook.write(outputStream);
 | 
	
		
			
				|  |  | +        } catch (IOException e) {
 | 
	
		
			
				|  |  | +            e.printStackTrace();
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    /**
 | 
	
		
			
				|  |  | +     * 4库导出
 | 
	
		
			
				|  |  | +     *
 | 
	
		
			
				|  |  | +     * @param response
 | 
	
		
			
				|  |  | +     * @param vo
 | 
	
		
			
				|  |  | +     */
 | 
	
		
			
				|  |  | +    @PostMapping("/exportTotalExcelByStatus")
 | 
	
		
			
				|  |  | +    public void exportTotalExcelByStatus(HttpServletResponse response, @RequestBody SubInfoQueryTzVO vo) {
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        try {
 | 
	
		
			
				|  |  | +            List<SubInfoTotalExcel> subInfoTotalExcel = subInfoService.exportTotalExcelByStatus(vo);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            //获取属地字典
 | 
	
		
			
				|  |  | +            List<TSystable> jsddList = tSysTableService.getByKind(SysTableKind.JSDD);
 | 
	
		
			
				|  |  | +            //获取行业字典
 | 
	
		
			
				|  |  | +            List<TSystable> hyflList = tSysTableService.getByKind(SysTableKind.HYFL);
 | 
	
		
			
				|  |  | +            //遍历subInfoTotalExcel,将subjectId替换为jsddList中code=subjectId的title
 | 
	
		
			
				|  |  | +            for (SubInfoTotalExcel data : subInfoTotalExcel) {
 | 
	
		
			
				|  |  | +                if (data == null) {
 | 
	
		
			
				|  |  | +                    continue;
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +                String subjectId = data.getSubjectId();
 | 
	
		
			
				|  |  | +                String induskind = data.getInduskind();
 | 
	
		
			
				|  |  | +                if (StringUtil.isNotEmpty(induskind)) {
 | 
	
		
			
				|  |  | +                    Optional<TSystable> first = hyflList.stream().filter(item -> item.getCode().equals(induskind)).findFirst();
 | 
	
		
			
				|  |  | +                    first.ifPresent(tSystable -> data.setInduskind(tSystable.getTitle()));
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +                if (StringUtil.isNotEmpty(subjectId)) {
 | 
	
		
			
				|  |  | +                    Optional<TSystable> first = jsddList.stream().filter(item -> item.getCode().equals(subjectId)).findFirst();
 | 
	
		
			
				|  |  | +                    first.ifPresent(tSystable -> data.setSubjectId(tSystable.getTitle()));
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            //通过ExcelUtil.getBigWriter()创建Writer对象,BigExcelWriter用于大数据量的导出,不会引起溢出;
 | 
	
		
			
				|  |  | +            cn.hutool.poi.excel.ExcelWriter writer = ExcelUtil.getBigWriter();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            //写入标题
 | 
	
		
			
				|  |  | +            // 写入标题行
 | 
	
		
			
				|  |  | +            List<String> titleRow = CollUtil.newArrayList("储备项目库基本情况");
 | 
	
		
			
				|  |  | +            writer.writeHeadRow(titleRow);
 | 
	
		
			
				|  |  | +            // 合并标题行的所有单元格
 | 
	
		
			
				|  |  | +            writer.merge(0, 0, 0, 3, null, true);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            List<String> rowHead2 = CollUtil.newArrayList("项目分类", "项目分类", "项目个数", "计划总投资(亿元)");
 | 
	
		
			
				|  |  | +            writer.writeHeadRow(rowHead2);
 | 
	
		
			
				|  |  | +            writer.merge(1, 1, 0, 1, null, true);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            // 获取 SXSSFSheet 对象(确保在写入数据之后)
 | 
	
		
			
				|  |  | +            SXSSFSheet sheet = ((SXSSFWorkbook) writer.getWorkbook()).getSheetAt(0);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            // 设置每列的默认宽度
 | 
	
		
			
				|  |  | +            int defaultColumnWidth = 40; // 默认宽度为20个字符
 | 
	
		
			
				|  |  | +            for (int i = 0; i < rowHead2.size(); i++) {
 | 
	
		
			
				|  |  | +                sheet.setColumnWidth(i, defaultColumnWidth * 256); // 其他列保持默认宽度
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            //4、对数据按照属地字段进行分组
 | 
	
		
			
				|  |  | +            LinkedHashMap<String, List<SubInfoTotalExcel>> subjectIdList = subInfoTotalExcel.stream().collect(Collectors.groupingBy(SubInfoTotalExcel::getSubjectId, LinkedHashMap::new, Collectors.toList()));
 | 
	
		
			
				|  |  | +            //5、对数据按照金额1000-3000,3000-5000,5000-10000进行分组
 | 
	
		
			
				|  |  | +            LinkedHashMap<String, List<SubInfoTotalExcel>> moneyList = subInfoTotalExcel.stream().collect(Collectors.groupingBy(data -> {
 | 
	
		
			
				|  |  | +                if (data.getAmtTotal().compareTo(new BigDecimal(1000000)) >= 0) {
 | 
	
		
			
				|  |  | +                    return "100亿元以上";
 | 
	
		
			
				|  |  | +                } else if (data.getAmtTotal().compareTo(new BigDecimal(500000)) >= 0 && data.getAmtTotal().compareTo(new BigDecimal(1000000)) < 0) {
 | 
	
		
			
				|  |  | +                    return "50-100亿元";
 | 
	
		
			
				|  |  | +                } else if (data.getAmtTotal().compareTo(new BigDecimal(100000)) >= 0 && data.getAmtTotal().compareTo(new BigDecimal(500000)) < 0) {
 | 
	
		
			
				|  |  | +                    return "10-50亿元";
 | 
	
		
			
				|  |  | +                } else {
 | 
	
		
			
				|  |  | +                    return "10亿元以下";
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }, LinkedHashMap::new, Collectors.toList()));
 | 
	
		
			
				|  |  | +            //6、对数据按照行业进行分组
 | 
	
		
			
				|  |  | +            LinkedHashMap<String, List<SubInfoTotalExcel>> industryList = subInfoTotalExcel.stream().collect(Collectors.groupingBy(SubInfoTotalExcel::getInduskind, LinkedHashMap::new, Collectors.toList()));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            //遍历subInfoTotalExcel数据计算总金额
 | 
	
		
			
				|  |  | +            BigDecimal totalAmt = subInfoTotalExcel.stream().map(SubInfoTotalExcel::getAmtTotal).reduce(BigDecimal.ZERO, BigDecimal::add);
 | 
	
		
			
				|  |  | +            Integer totalNum = subInfoTotalExcel.size();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            LinkedHashMap<String, List<List<String>>> finalDataMap = new LinkedHashMap<>();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            //7、需要保存的数据
 | 
	
		
			
				|  |  | +            List<List<String>> areaData = CollUtil.newArrayList();
 | 
	
		
			
				|  |  | +            for (Map.Entry<String, List<SubInfoTotalExcel>> subInfoListEntry : subjectIdList.entrySet()) {
 | 
	
		
			
				|  |  | +                List<String> data = CollUtil.newArrayList();
 | 
	
		
			
				|  |  | +                data.add("按属地分");
 | 
	
		
			
				|  |  | +                data.add(subInfoListEntry.getKey());
 | 
	
		
			
				|  |  | +                data.add(String.valueOf(subInfoListEntry.getValue().size()));
 | 
	
		
			
				|  |  | +                data.add(String.valueOf(subInfoListEntry.getValue().stream().map(SubInfoTotalExcel::getAmtTotal).reduce(BigDecimal.ZERO, BigDecimal::add)));
 | 
	
		
			
				|  |  | +                areaData.add(data);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            finalDataMap.put("按属地划分", areaData);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            List<List<String>> moneyData = CollUtil.newArrayList();
 | 
	
		
			
				|  |  | +            for (Map.Entry<String, List<SubInfoTotalExcel>> subInfoListEntry : moneyList.entrySet()) {
 | 
	
		
			
				|  |  | +                List<String> data = CollUtil.newArrayList();
 | 
	
		
			
				|  |  | +                data.add("按金额分");
 | 
	
		
			
				|  |  | +                data.add(subInfoListEntry.getKey());
 | 
	
		
			
				|  |  | +                data.add(String.valueOf(subInfoListEntry.getValue().size()));
 | 
	
		
			
				|  |  | +                data.add(String.valueOf(subInfoListEntry.getValue().stream().map(SubInfoTotalExcel::getAmtTotal).reduce(BigDecimal.ZERO, BigDecimal::add)));
 | 
	
		
			
				|  |  | +                moneyData.add(data);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            finalDataMap.put("按金额划分", moneyData);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            List<List<String>> industryData = CollUtil.newArrayList();
 | 
	
		
			
				|  |  | +            for (Map.Entry<String, List<SubInfoTotalExcel>> subInfoListEntry : industryList.entrySet()) {
 | 
	
		
			
				|  |  | +                List<String> data = CollUtil.newArrayList();
 | 
	
		
			
				|  |  | +                data.add("按行业分");
 | 
	
		
			
				|  |  | +                data.add(subInfoListEntry.getKey());
 | 
	
		
			
				|  |  | +                data.add(String.valueOf(subInfoListEntry.getValue().size()));
 | 
	
		
			
				|  |  | +                data.add(String.valueOf(subInfoListEntry.getValue().stream().map(SubInfoTotalExcel::getAmtTotal).reduce(BigDecimal.ZERO, BigDecimal::add)));
 | 
	
		
			
				|  |  | +                industryData.add(data);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            finalDataMap.put("按行业划分", industryData);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            ServletOutputStream out = null;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            try {
 | 
	
		
			
				|  |  | +                //6.定义容器保存人物数据
 | 
	
		
			
				|  |  | +                List<List<String>> finalDataRows = new LinkedList<>();
 | 
	
		
			
				|  |  | +                int indexStart = 2;   //行业分类起始行
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                for (Map.Entry<String, List<List<String>>> dataListEntry : finalDataMap.entrySet()) {
 | 
	
		
			
				|  |  | +                    List<List<String>> dataList = dataListEntry.getValue();
 | 
	
		
			
				|  |  | +                    int dataSize = dataList.size();
 | 
	
		
			
				|  |  | +                    if (dataSize == 1) {
 | 
	
		
			
				|  |  | +                        indexStart += dataSize;
 | 
	
		
			
				|  |  | +                    } else {
 | 
	
		
			
				|  |  | +                        writer.merge(indexStart, indexStart + dataSize - 1, 0, 0, null, true);
 | 
	
		
			
				|  |  | +                        indexStart += dataSize;
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    //18.保存数据
 | 
	
		
			
				|  |  | +                    finalDataRows.addAll(dataList);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +                writer.write(finalDataRows, true);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                // 创建合计行
 | 
	
		
			
				|  |  | +                List<String> totalRow = CollUtil.newArrayList("合计", "合计", totalNum.toString(), totalAmt.toString());
 | 
	
		
			
				|  |  | +                writer.writeHeadRow(totalRow);
 | 
	
		
			
				|  |  | +                writer.merge(indexStart, indexStart, 0, 1, null, true);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                response.setContentType("application/vnd.ms-excel;charset=utf-8");
 | 
	
		
			
				|  |  | +                Date currentDate = new Date();
 | 
	
		
			
				|  |  | +                SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd");
 | 
	
		
			
				|  |  | +                String date = sdf.format(currentDate);
 | 
	
		
			
				|  |  | +                response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("重点项目调度-项目数据.xlsx", "UTF-8"));
 | 
	
		
			
				|  |  | +                out = response.getOutputStream();
 | 
	
		
			
				|  |  | +                writer.flush(out, true);
 | 
	
		
			
				|  |  | +            } finally {
 | 
	
		
			
				|  |  | +                //关闭输出Servlet流
 | 
	
		
			
				|  |  | +                IoUtil.close(out);
 | 
	
		
			
				|  |  | +                //关闭writer,释放内存
 | 
	
		
			
				|  |  | +                writer.close();
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        } catch (IOException e) {
 | 
	
		
			
				|  |  | +            // 设置错误状态码和消息
 | 
	
		
			
				|  |  | +            response.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
 | 
	
		
			
				|  |  | +            try {
 | 
	
		
			
				|  |  | +                e.printStackTrace();
 | 
	
		
			
				|  |  | +                response.getWriter().write("导出 Excel 文件时出现错误:" + e.getMessage());
 | 
	
		
			
				|  |  | +            } catch (IOException ex) {
 | 
	
		
			
				|  |  | +                ex.printStackTrace();
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        } catch (Exception e) {
 | 
	
		
			
				|  |  | +            // 处理其他可能的异常
 | 
	
		
			
				|  |  | +            response.setStatus(HttpServletResponse.SC_BAD_REQUEST);
 | 
	
		
			
				|  |  | +            e.printStackTrace();
 | 
	
		
			
				|  |  | +            try {
 | 
	
		
			
				|  |  | +                response.getWriter().write("请求参数错误或发生其他错误:" + e.getMessage());
 | 
	
		
			
				|  |  | +            } catch (IOException ex) {
 | 
	
		
			
				|  |  | +                ex.printStackTrace();
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      /**
 | 
	
		
			
				|  |  |       * 导出重点项目明细表
 |