Java使用EasyExcel导出图片(原比例大小)到Excel中

EasyExcel导出图片

又开始写Excel导出的需求了,哈哈哈……

目前的需求是将图表分析对的饼图和折线图,也就是一张完整的图片单独导出到Excel中

为了方便客户在业务报告时,可以使用数据分析图片,从而更清晰准确地展示数据趋势

因此关键点是将图片原比例尺寸大小导出,不能进行压缩

原数据是由图表📈+表格数据组成,下图所示:

现在需要将上述数据,分两个Sheet导出到Excel:图表📈Sheet +表格数据Sheet

关于表格数据导出,此处就不做展示了,往期的Excel导出有文章说明,此文只导出图表📈Sheet

步骤1:将图片保存成url

将图片转为url的形式传参

{
  "imagePath": "http://www.echola.com/8pJTiC_1724644117200.png",
   …… //其他参数
}

步骤2:自定义图片导出工具类

@Slf4j
public class CustomImageStrategy implements CellWriteHandler {

    private final HashMap<String, List<ImageData>> imageDataMap = new HashMap<>(16);

    @Override
    public void afterCellDataConverted(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, WriteCellData<?> cellData, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {
        if (isHead) {
            return;
        }
        // 将单元格图片数据复制出来,清空单元格图片数据
        if (!CollectionUtils.isEmpty(cellData.getImageDataList())) {
            imageDataMap.put(cell.getRowIndex() + "_" + cell.getColumnIndex(), cellData.getImageDataList());
            cellData.setType(CellDataTypeEnum.EMPTY);
            cellData.setImageDataList(new ArrayList<>());
        }
    }

    @Override
    public void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {
        if (isHead || CollectionUtils.isEmpty(cellDataList)) {
            return;
        }
        String key = cell.getRowIndex() + "_" + cell.getColumnIndex();
        List<ImageData> imageDataList = imageDataMap.get(key);
        if (CollectionUtils.isEmpty(imageDataList)) {
            return;
        }

        // 插入图片
        for (int i = 0; i < imageDataList.size(); i++) {
            ImageData imageData = imageDataList.get(i);
            if (ObjectUtils.isEmpty(imageData)) {
                continue;
            }
            byte[] image = imageData.getImage();
            this.insertImage(writeSheetHolder.getSheet(), cell, image, i);
        }
        imageDataMap.remove(key);
    }

    private void insertImage(Sheet sheet, Cell cell, byte[] pictureData, int i) {
        // 读取图片的宽度和高度
        int pictureWidth = 0;
        int pictureHeight = 0;
        try (ByteArrayInputStream bis = new ByteArrayInputStream(pictureData)) {
            BufferedImage bufferedImage = ImageIO.read(bis);
            pictureWidth = bufferedImage.getWidth();
            pictureHeight = bufferedImage.getHeight();
        } catch (Exception e) {
            log.error("Error reading image dimensions", e);
        }

        // 将像素转换为 EMU 单位
        int emuWidth = Units.pixelToEMU(pictureWidth);
        int emuHeight = Units.pixelToEMU(pictureHeight);
        int index = sheet.getWorkbook().addPicture(pictureData, Workbook.PICTURE_TYPE_PNG);
        Drawing<?> drawing = sheet.getDrawingPatriarch();
        if (drawing == null) {
            drawing = sheet.createDrawingPatriarch();
        }
        CreationHelper helper = sheet.getWorkbook().getCreationHelper();
        ClientAnchor anchor = helper.createClientAnchor();
        // 设置图片在哪个单元格中
        anchor.setCol1(cell.getColumnIndex());
        anchor.setCol2(cell.getColumnIndex() + 1);
        anchor.setRow1(cell.getRowIndex());
        anchor.setRow2(cell.getRowIndex() + 1);

        // 设置图片在单元格中的位置
        // 横向偏移量
        anchor.setDx1(0);
        anchor.setDx2(emuWidth);
        // 纵向偏移量
        anchor.setDy1(0);
        anchor.setDy2(emuHeight);

        // 设置图片可以随着单元格移动
        anchor.setAnchorType(ClientAnchor.AnchorType.MOVE_AND_RESIZE);
        drawing.createPicture(anchor, index);
        // 设置行高和列宽为图片的实际大小
        Row row = sheet.getRow(cell.getRowIndex());
        if (row == null) {
            row = sheet.createRow(cell.getRowIndex());
        }
        // 设置行高,1像素大约等于0.75点
        float maxRowHeight = 300;
        float imageHeight = (float) (pictureHeight * 0.75);
        row.setHeightInPoints(Math.min(imageHeight, maxRowHeight));

        // 设置列宽,列宽单位是1/256个字符,1个字符大约为7像素
        int maxColumnWidth = 255 * 256;
        int imageWidth = pictureWidth * 37;
        sheet.setColumnWidth(cell.getColumnIndex(), Math.min(imageWidth, maxColumnWidth));
    }

调试过程中,出了一点小问题,出了这个异常:

The maximum column width for an individual cell is 255 characters

由于图片宽度过大,导致超过了Excel单元格的最大宽度255,因此需要兼容最大行高和列宽

原代码段是:

 // 设置行高,1像素大约等于0.75点
row.setHeightInPoints((float) (pictureHeight * 0.75));

// 设置列宽,列宽单位是1/256个字符,1个字符大约为7像素
sheet.setColumnWidth(cell.getColumnIndex(), pictureWidth * 37);

修改后代码段:

// 设置行高,1像素大约等于0.75点
float maxRowHeight = 300;
float imageHeight = (float) (pictureHeight * 0.75);
row.setHeightInPoints(Math.min(imageHeight, maxRowHeight));

// 设置列宽,列宽单位是1/256个字符,1个字符大约为7像素
int maxColumnWidth = 255 * 256;
int imageWidth = pictureWidth * 37;
sheet.setColumnWidth(cell.getColumnIndex(), Math.min(imageWidth, maxColumnWidth));

步骤3:导出图片

public void exportAnalysis(EnergyStatisticDTO param, HttpServletResponse response) {
        String fileName = "数据分析" + DateUtil.format(new Date(), DatePattern.PURE_DATE_FORMAT);
        try {
            EasyExcelUtils.setResponse(response, fileName);
            ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream()).build();
            //各用电用途趋势
            String sheetName = "数据分析图示";
            this.meterImageSheet(sheetName, param.getImagePath(), excelWriter);
            String imagePath = param.getImagePath();
            if (StringUtils.isEmpty(imagePath)) return;
            //支持多图片导出,此处只是单图片
            List<ImageDTO> imageList = new ArrayList<>();
            ImageDTO imageDTO = new ImageDTO();
            imageDTO.setImageUrl(Collections.singletonList(imagePath));
            imageList.add(imageDTO);
            WriteSheet imageSheet = EasyExcel.writerSheet(1, sheetName)
                    .registerWriteHandler(new CustomImageStrategy())
                    .build();
            excelWriter.write(imageList, imageSheet);
            excelWriter.finish();
        } catch (Exception e) {
            log.error("数据分析:{}", e.getMessage());
        }
    }

Excel导出图片效果图

可以看出和Web的图表的大小是基本一致的

撒花完结……

EasyExcel 是一个基于Java的开源Excel操作工具,可以用来读取、写入Excel文件。如果你想要将图片导出Excel,可以按照以下步骤操作: 1. 首先,需要将图片保存到本地,可以使用Java的ImageIO类将图片从网络上下载并保存到本地。 2. 然后,使用EasyExcel的WriteSheet类创建一个新的Sheet,并设置好Sheet的名称、表头等信息。 3. 接着,使用EasyExcel的WriteSheet类的write方法将数据写入到Excel。在写入数据的过程,可以使用EasyExcel的FillConfig类的setImage方法将图片插入到Excel。 4. 最后,使用EasyExcelExcelWriter类将所有的Sheet写入到Excel文件,并关闭ExcelWriter。 下面是一个简单的示例代码: ```java // 创建ExcelWriter ExcelWriter excelWriter = EasyExcel.write("test.xlsx").build(); // 创建Sheet WriteSheet writeSheet = EasyExcel.writerSheet("Sheet1").build(); // 将图片插入到Excel FillConfig fillConfig = FillConfig.builder().setImage(ImageIO.read(new File("image.jpg"))).build(); List<Object> list = new ArrayList<Object>(); list.add(fillConfig); excelWriter.fill(list, fillConfig, writeSheet); // 写入数据到Excel List<List<Object>> data = new ArrayList<List<Object>>(); data.add(Arrays.asList("Name", "Age", "Image")); data.add(Arrays.asList("Tom", 20, fillConfig)); writeSheet.setHead(data.subList(0, 1)); writeSheet.setClazz(Student.class); excelWriter.write(data.subList(1, data.size()), writeSheet); // 关闭ExcelWriter excelWriter.finish(); ``` 在这个示例代码,我们将一张名为 "image.jpg" 的图片插入到了Excel,并将它作为数据的一部分写入到了Excel。注意,在将图片插入到Excel时,需要使用FillConfig类指定图片的位置和大小等信息。同时,在写入数据的时候,需要设置数据的类型为Student类,并且将图片插入到数据
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值