|
|
|
@ -0,0 +1,320 @@
|
|
|
|
|
package com.op.common.core.utils.poi;
|
|
|
|
|
|
|
|
|
|
import com.alibaba.fastjson2.JSONObject;
|
|
|
|
|
import com.op.common.core.domain.ExcelCol;
|
|
|
|
|
import org.apache.commons.compress.utils.IOUtils;
|
|
|
|
|
import org.apache.poi.ss.usermodel.*;
|
|
|
|
|
import org.apache.poi.ss.util.CellRangeAddress;
|
|
|
|
|
import org.apache.poi.xssf.streaming.SXSSFWorkbook;
|
|
|
|
|
import org.springframework.util.CollectionUtils;
|
|
|
|
|
import org.springframework.util.StringUtils;
|
|
|
|
|
|
|
|
|
|
import java.io.ByteArrayOutputStream;
|
|
|
|
|
import java.io.IOException;
|
|
|
|
|
import java.io.InputStream;
|
|
|
|
|
import java.nio.file.Files;
|
|
|
|
|
import java.nio.file.Paths;
|
|
|
|
|
import java.util.Iterator;
|
|
|
|
|
import java.util.List;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Excel导出报告相关处理
|
|
|
|
|
*
|
|
|
|
|
* @author OP
|
|
|
|
|
*/
|
|
|
|
|
public class ExcelReportMapUtil {
|
|
|
|
|
//下载
|
|
|
|
|
public static <T> SXSSFWorkbook initWorkbook(String sheetName , String title , List<ExcelCol> excelCol , List<T> data) throws IOException {
|
|
|
|
|
SXSSFWorkbook workbook = new SXSSFWorkbook();
|
|
|
|
|
int colSize = excelCol.size();
|
|
|
|
|
|
|
|
|
|
//创建Sheet(工作簿)
|
|
|
|
|
Sheet sheet = null;
|
|
|
|
|
if (!StringUtils.hasText(sheetName)){
|
|
|
|
|
sheet = workbook.createSheet();
|
|
|
|
|
}else{
|
|
|
|
|
sheet = workbook.createSheet(sheetName);
|
|
|
|
|
}
|
|
|
|
|
if(title != null){
|
|
|
|
|
|
|
|
|
|
// 从resources目录获取图片字节数组
|
|
|
|
|
byte[] imageBytes = getImageBytesFromResources("image/logo.png");
|
|
|
|
|
/**logo**/
|
|
|
|
|
int pictureIdx = workbook.addPicture(imageBytes, Workbook.PICTURE_TYPE_PNG);
|
|
|
|
|
CreationHelper helper = workbook.getCreationHelper();
|
|
|
|
|
Drawing<?> drawing = sheet.createDrawingPatriarch();
|
|
|
|
|
ClientAnchor anchor = helper.createClientAnchor();
|
|
|
|
|
anchor.setCol1(0); // B列
|
|
|
|
|
anchor.setRow1(0); // 第2行(Excel的行和列都是从0开始计数的)
|
|
|
|
|
anchor.setCol2(1); // C列(图片宽度跨越的列数,这里设置为1列宽)
|
|
|
|
|
anchor.setRow2(2); // 第6行(图片高度跨越的行数,这里设置为4行高,可以根据图片大小调整)
|
|
|
|
|
Picture pict = drawing.createPicture(anchor, pictureIdx);
|
|
|
|
|
pict.resize(); // 根据图片的实际大小调整图片在Excel中的显示大小
|
|
|
|
|
/**报告标题**/
|
|
|
|
|
sheet.addMergedRegion(new CellRangeAddress(0, 1, 2, 11));
|
|
|
|
|
// 获取合并后的单元格的第一个单元格(即C1),并设置值
|
|
|
|
|
Row row = sheet.getRow(0); // 获取第1行(索引为0)
|
|
|
|
|
if (row == null) {row = sheet.createRow(0); // 如果第1行不存在,则创建它
|
|
|
|
|
}
|
|
|
|
|
Cell cell = row.getCell(2); // 获取C列(索引为2)的单元格
|
|
|
|
|
if (cell == null) {
|
|
|
|
|
cell = row.createCell(2); // 如果C列的单元格不存在,则创建它
|
|
|
|
|
}
|
|
|
|
|
cell.setCellValue("中山榄菊日化实业有限公司"); // 设置单元格的值
|
|
|
|
|
cell.setCellStyle(getTitelStyle(workbook)); // 应用样式到单元格
|
|
|
|
|
/**报告二级标题、检验标准**/
|
|
|
|
|
// 合并C3到I4的单元格,并设置样式
|
|
|
|
|
mergeAndStyleCells(sheet, new CellRangeAddress(2, 3, 2, 8), "包材检验报告");
|
|
|
|
|
// 合并J3到L4的单元格,并设置样式
|
|
|
|
|
mergeAndStyleCells(sheet, new CellRangeAddress(2, 3, 9, 12), "编码:07GL06C003-001A");
|
|
|
|
|
/**左右表格1**/
|
|
|
|
|
// 合并A5到B6的单元格,并设置样式和内容
|
|
|
|
|
mergeAndStyleCells2(sheet, new CellRangeAddress(4, 5, 0, 1), "包材检验报告", true, true, IndexedColors.GREY_25_PERCENT);
|
|
|
|
|
// 合并C5到D6的单元格,并设置样式(无背景色,只有边框和内容居中)
|
|
|
|
|
mergeAndStyleCells2(sheet, new CellRangeAddress(4, 5, 2, 3), "", true, false, null);
|
|
|
|
|
// 合并E5到F6的单元格,并设置样式和内容
|
|
|
|
|
mergeAndStyleCells2(sheet, new CellRangeAddress(4, 5, 4, 5), "生产批号", true, true, IndexedColors.GREY_25_PERCENT);
|
|
|
|
|
// 合并G5到I6的单元格,并设置样式(无背景色,只有边框和内容居中)
|
|
|
|
|
mergeAndStyleCells2(sheet, new CellRangeAddress(4, 5, 6, 8), "", true, false, null);
|
|
|
|
|
// 合并J5到K6的单元格,并设置样式和内容
|
|
|
|
|
mergeAndStyleCells2(sheet, new CellRangeAddress(4, 5, 9, 10), "报告编号", true, true, IndexedColors.GREY_25_PERCENT);
|
|
|
|
|
// 合并L5到N6的单元格,并设置样式(无背景色,只有边框和内容居中)
|
|
|
|
|
mergeAndStyleCells2(sheet, new CellRangeAddress(4, 5, 11, 13), "", true, false, null);
|
|
|
|
|
|
|
|
|
|
// // 将工作簿写入文件(或输出流)
|
|
|
|
|
// try (ByteArrayOutputStream fileOut = new ByteArrayOutputStream()) {
|
|
|
|
|
// workbook.write(fileOut);
|
|
|
|
|
// // 这里可以将fileOut.toByteArray()保存到文件或发送到其他地方
|
|
|
|
|
// // 例如:Files.write(Paths.get("output.xlsx"), fileOut.toByteArray());
|
|
|
|
|
// } catch (IOException e) {
|
|
|
|
|
// e.printStackTrace();
|
|
|
|
|
// } finally {
|
|
|
|
|
// try {
|
|
|
|
|
// workbook.close();
|
|
|
|
|
// } catch (IOException e) {
|
|
|
|
|
// e.printStackTrace();
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
// String[] title1s = title.split(",");
|
|
|
|
|
// int rangeVal = colSize/title1s.length;
|
|
|
|
|
// //创建主标题行(第一行)
|
|
|
|
|
// Row sheetTitleRow = sheet.createRow(0);
|
|
|
|
|
//
|
|
|
|
|
// Cell titleCell0 = sheetTitleRow.createCell(0);//创建第一行第一个单元格
|
|
|
|
|
// titleCell0.setCellValue("");//传值
|
|
|
|
|
// titleCell0.setCellStyle(getHeaderFont(sheet.getWorkbook()));//设置样式
|
|
|
|
|
//
|
|
|
|
|
// //遍历表头名称,创建表头单元格
|
|
|
|
|
// for (int i = 0; i < title1s.length; i++) {
|
|
|
|
|
// Cell titleCell = sheetTitleRow.createCell(i*rangeVal+1);//创建第一行第一个单元格
|
|
|
|
|
// titleCell.setCellValue(title1s[i]);//传值
|
|
|
|
|
// titleCell.setCellStyle(getHeaderFont(sheet.getWorkbook()));//设置样式
|
|
|
|
|
// if((i*rangeVal+1)!=(i+1)*rangeVal){
|
|
|
|
|
// //主标题行合并单元格
|
|
|
|
|
// CellRangeAddress cellAddresses = new CellRangeAddress(0, 0, i*rangeVal+1, (i+1)*rangeVal);
|
|
|
|
|
// sheet.addMergedRegion(cellAddresses);
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// //创建表头行(第二行)
|
|
|
|
|
// Row sheetHeadRow = sheet.createRow(1);//1
|
|
|
|
|
// //遍历表头名称,创建表头单元格
|
|
|
|
|
// for (int i = 0; i < colSize; i++) {
|
|
|
|
|
// sheet.setColumnWidth(i, (excelCol.get(i).getWidth()) * 256);//宽度单位是字符的256分之一
|
|
|
|
|
// Cell headCell = sheetHeadRow.createCell(i);
|
|
|
|
|
// headCell.setCellValue(excelCol.get(i).getTitle());//传值
|
|
|
|
|
// headCell.setCellStyle(getHeaderFont(sheet.getWorkbook()));//设置样式
|
|
|
|
|
// }
|
|
|
|
|
// }else {
|
|
|
|
|
// //创建表头行(第二行)
|
|
|
|
|
// Row sheetHeadRow = sheet.createRow(0);//1
|
|
|
|
|
// //遍历表头名称,创建表头单元格
|
|
|
|
|
// for (int i = 0; i < colSize; i++) {
|
|
|
|
|
// sheet.setColumnWidth(i, (excelCol.get(i).getWidth()) * 256);//宽度单位是字符的256分之一
|
|
|
|
|
// Cell headCell = sheetHeadRow.createCell(i);
|
|
|
|
|
// headCell.setCellValue(excelCol.get(i).getTitle());//传值
|
|
|
|
|
// headCell.setCellStyle(getHeaderFont(sheet.getWorkbook()));//设置样式
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// //将data中的值填充到excel
|
|
|
|
|
// int rowNum = sheet.getLastRowNum()+1;
|
|
|
|
|
// if(!CollectionUtils.isEmpty(data)){
|
|
|
|
|
// Iterator<T> iterator = data.iterator();
|
|
|
|
|
// //遍历数据
|
|
|
|
|
// for (;iterator.hasNext();){
|
|
|
|
|
// Row dataRow = sheet.createRow(rowNum);//创建行
|
|
|
|
|
// T obj = iterator.next();//获取当前行对应的数据
|
|
|
|
|
// JSONObject jsonObject = JSONObject.parseObject(JSONObject.toJSONString(obj));
|
|
|
|
|
// for (int i = 0 ; i < colSize ; i++ ){
|
|
|
|
|
// Cell dataCell = dataRow.createCell(i);
|
|
|
|
|
// dataCell.setCellStyle(getDataFont(workbook));
|
|
|
|
|
// if(title!=null){//定量分析
|
|
|
|
|
// dataCell.setCellValue(getValue(jsonObject.get(excelCol.get(i).getField())));
|
|
|
|
|
// }else{
|
|
|
|
|
// if(i>=2){
|
|
|
|
|
// dataCell.setCellValue(getValueNum(jsonObject.get(excelCol.get(i).getField())));
|
|
|
|
|
// }else{
|
|
|
|
|
// dataCell.setCellValue(getValue(jsonObject.get(excelCol.get(i).getField())));
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
// iterator.remove();
|
|
|
|
|
// rowNum++;
|
|
|
|
|
// }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return workbook;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// //标题样式
|
|
|
|
|
// public static CellStyle getHeaderFont(Workbook workbook){
|
|
|
|
|
// Font font = workbook.createFont();
|
|
|
|
|
// font.setFontHeightInPoints((short) 15);//字体大小
|
|
|
|
|
// font.setBold(true);//加粗
|
|
|
|
|
// CellStyle cellStyle = workbook.createCellStyle();
|
|
|
|
|
// cellStyle.setFont(font);
|
|
|
|
|
// cellStyle.setAlignment(HorizontalAlignment.CENTER_SELECTION);//设置水平居中
|
|
|
|
|
// cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);//设置垂直居中
|
|
|
|
|
// // 设置上边框
|
|
|
|
|
// cellStyle.setBorderTop(BorderStyle.THIN);
|
|
|
|
|
// // 设置下边框
|
|
|
|
|
// cellStyle.setBorderBottom(BorderStyle.THIN);
|
|
|
|
|
// // 设置左边框
|
|
|
|
|
// cellStyle.setBorderLeft(BorderStyle.THIN);
|
|
|
|
|
// // 设置右边框
|
|
|
|
|
// cellStyle.setBorderRight(BorderStyle.THIN);
|
|
|
|
|
// cellStyle.setFillBackgroundColor(IndexedColors.AQUA.getIndex());
|
|
|
|
|
//
|
|
|
|
|
// return cellStyle;
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
// //内容样式
|
|
|
|
|
// public static CellStyle getDataFont(Workbook workbook){
|
|
|
|
|
// Font font = workbook.createFont();
|
|
|
|
|
// font.setFontHeightInPoints((short) 12);//字体大小
|
|
|
|
|
// font.setBold(false);//不加粗
|
|
|
|
|
// CellStyle cellStyle = workbook.createCellStyle();
|
|
|
|
|
// cellStyle.setFont(font);
|
|
|
|
|
// cellStyle.setAlignment(HorizontalAlignment.CENTER_SELECTION);//设置水平居中
|
|
|
|
|
// cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);//设置垂直居中
|
|
|
|
|
// cellStyle.setWrapText(true);//设置单元格内容自动换行
|
|
|
|
|
// return cellStyle;
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// //处理数据
|
|
|
|
|
// public static String getValue(Object object){
|
|
|
|
|
// if (object==null){
|
|
|
|
|
// return "";
|
|
|
|
|
// }else {
|
|
|
|
|
// return object.toString();
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
// //处理数据
|
|
|
|
|
// public static Integer getValueNum(Object object){
|
|
|
|
|
// if (object==null){
|
|
|
|
|
// return 0;
|
|
|
|
|
// }else {
|
|
|
|
|
// return Integer.parseInt(object.toString());
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
// 从resources目录获取图片的字节数组
|
|
|
|
|
private static byte[] getImageBytesFromResources(String resourceName) {
|
|
|
|
|
try (InputStream inputStream = ExcelReportMapUtil.class.getClassLoader().getResourceAsStream(resourceName);
|
|
|
|
|
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream()) {
|
|
|
|
|
|
|
|
|
|
if (inputStream == null) {
|
|
|
|
|
throw new RuntimeException("找不到资源: " + resourceName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IOUtils.copy(inputStream, byteArrayOutputStream);
|
|
|
|
|
return byteArrayOutputStream.toByteArray();
|
|
|
|
|
|
|
|
|
|
} catch (IOException e) {
|
|
|
|
|
throw new RuntimeException("读取资源时出错: " + resourceName, e);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
//报告大标题样式-1
|
|
|
|
|
public static CellStyle getTitelStyle(Workbook workbook){
|
|
|
|
|
Font font = workbook.createFont();
|
|
|
|
|
// 设置字体为加粗
|
|
|
|
|
font.setBold(true);
|
|
|
|
|
// 设置字体大小(例如,设置为16)
|
|
|
|
|
font.setFontHeightInPoints((short) 24);
|
|
|
|
|
CellStyle cellStyle = workbook.createCellStyle();
|
|
|
|
|
cellStyle.setFont(font);
|
|
|
|
|
cellStyle.setAlignment(HorizontalAlignment.CENTER_SELECTION);//设置水平居中
|
|
|
|
|
cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);//设置垂直居中
|
|
|
|
|
cellStyle.setWrapText(true);//设置单元格内容自动换行
|
|
|
|
|
return cellStyle;
|
|
|
|
|
}
|
|
|
|
|
private static void mergeAndStyleCells(Sheet sheet, CellRangeAddress cellRangeAddress, String cellValue) {
|
|
|
|
|
// 合并单元格
|
|
|
|
|
sheet.addMergedRegion(cellRangeAddress);
|
|
|
|
|
|
|
|
|
|
// 创建一个单元格样式
|
|
|
|
|
CellStyle style = sheet.getWorkbook().createCellStyle();
|
|
|
|
|
|
|
|
|
|
// 设置字体为加粗
|
|
|
|
|
Font font = sheet.getWorkbook().createFont();
|
|
|
|
|
font.setBold(true);
|
|
|
|
|
// 设置字体大小(例如,设置为14)
|
|
|
|
|
font.setFontHeightInPoints((short) 14);
|
|
|
|
|
style.setFont(font);
|
|
|
|
|
|
|
|
|
|
// 设置水平居中
|
|
|
|
|
style.setAlignment(HorizontalAlignment.CENTER);
|
|
|
|
|
// 设置垂直居中
|
|
|
|
|
style.setVerticalAlignment(VerticalAlignment.CENTER);
|
|
|
|
|
style.setWrapText(true);//设置单元格内容自动换行
|
|
|
|
|
// 获取合并后的单元格的第一个单元格,并设置值
|
|
|
|
|
Row row = sheet.getRow(cellRangeAddress.getFirstRow());
|
|
|
|
|
if (row == null) {
|
|
|
|
|
row = sheet.createRow(cellRangeAddress.getFirstRow());
|
|
|
|
|
}
|
|
|
|
|
Cell cell = row.getCell(cellRangeAddress.getFirstColumn());
|
|
|
|
|
if (cell == null) {
|
|
|
|
|
cell = row.createCell(cellRangeAddress.getFirstColumn());
|
|
|
|
|
}
|
|
|
|
|
cell.setCellValue(cellValue); // 设置单元格的值
|
|
|
|
|
cell.setCellStyle(style); // 应用样式到单元格
|
|
|
|
|
}
|
|
|
|
|
private static void mergeAndStyleCells2(Sheet sheet, CellRangeAddress cellRangeAddress, String cellValue, boolean centered, boolean hasBackground, IndexedColors backgroundColor) {
|
|
|
|
|
// 合并单元格
|
|
|
|
|
sheet.addMergedRegion(cellRangeAddress);
|
|
|
|
|
|
|
|
|
|
// 创建一个单元格样式
|
|
|
|
|
CellStyle style = sheet.getWorkbook().createCellStyle();
|
|
|
|
|
|
|
|
|
|
// 设置字体居中
|
|
|
|
|
if (centered) {
|
|
|
|
|
style.setAlignment(HorizontalAlignment.CENTER);
|
|
|
|
|
style.setVerticalAlignment(VerticalAlignment.CENTER);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置背景颜色(如果有)
|
|
|
|
|
if (hasBackground && backgroundColor != null) {
|
|
|
|
|
style.setFillForegroundColor(backgroundColor.getIndex());
|
|
|
|
|
style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置边框线
|
|
|
|
|
style.setBorderBottom(BorderStyle.THIN);
|
|
|
|
|
style.setBorderLeft(BorderStyle.THIN);
|
|
|
|
|
style.setBorderRight(BorderStyle.THIN);
|
|
|
|
|
style.setBorderTop(BorderStyle.THIN);
|
|
|
|
|
|
|
|
|
|
// 获取合并后的单元格的第一个单元格,并设置值
|
|
|
|
|
Row row = sheet.getRow(cellRangeAddress.getFirstRow());
|
|
|
|
|
if (row == null) {
|
|
|
|
|
row = sheet.createRow(cellRangeAddress.getFirstRow());
|
|
|
|
|
}
|
|
|
|
|
Cell cell = row.getCell(cellRangeAddress.getFirstColumn());
|
|
|
|
|
if (cell == null) {
|
|
|
|
|
cell = row.createCell(cellRangeAddress.getFirstColumn());
|
|
|
|
|
}
|
|
|
|
|
cell.setCellValue(cellValue); // 设置单元格的值
|
|
|
|
|
cell.setCellStyle(style); // 应用样式到单元格
|
|
|
|
|
}
|
|
|
|
|
}
|