【EasyExcel等比例缩小导出图片】

EasyExcel等比例缩小导出图片

一、背景

  • 使用EasyExcel导出excel文件,但是需要同时导出图片信息,且图片信息不能影响行高和单元格宽度,图片本身被导出时,不能因为压缩导致图片变形

二、思路

  • 使用EasyExcel最主要是要实现各种hadler,通过实现CellWriteHandler,获取单元格图片,进行图片压缩;
  • 由于excel的行高和单元格宽度换算方式完全不一样,所以要通过计算得出最合适的压缩比例
  • (excel宽度 / 字符宽度转换为标准字符宽度) * 估算的字符到像素的转换因子 = 宽度像素;
  • (excel高度 / 点转换为英寸) * 英寸转换为像素 = 行高像素;
  • 最终宽度像素或者行高像素 根据行高与像素的转换因子 ,计算出需要设置的单元格间距值,即可正常缩小图片

三、代码

java 复制代码
public static class QuotationCustomCellWriteHandler implements CellWriteHandler {
        // 256分之一的字符宽度转换为标准字符宽度。
        public static final int STANDARD_CHARACTER_WIDTH = 256;
        // 7.5是一个估算的字符到像素的转换因子
        public static final float CHARACTER_2_PIXEL_FACTOR = 7.5f;
        // 将点转换为英寸,因为1点 = 1/72英寸。
        public static final int PIXEL_2_INCH_FACTOR = 72;
        // 英寸转换为像素,其中96是常用的DPI(每英寸像素数)值。
        public static final int DPI = 96;
        // 行高与像素的转换因子
        public static final float ROW_HEIGHT_2_PIXEL_FACTOR = 1.3333f;

        /**
         * 外界传入的行高,内部查不到
         */
        private final int rowHeight;

        public QuotationCustomCellWriteHandler(int rowHeight) {
            this.rowHeight = rowHeight;
        }

        @Override
        public void afterCellDataConverted(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder,
                                           WriteCellData<?> cellData, Cell cell, Head head, Integer relativeRowIndex,
                                           Boolean isHead) {
            boolean noImageValue = Objects.isNull(cellData) || cellData.getImageDataList() == null || cellData.getImageDataList().isEmpty();
            if (Objects.equals(Boolean.TRUE, isHead) || noImageValue) {
                return;
            }
            Sheet sheet = cell.getSheet();
            ImageData imageData = cellData.getImageDataList().get(0);
            imageData.setRelativeLastRowIndex(0);
            imageData.setRelativeLastColumnIndex(0);
            // 处理图像缩放 - 等比例缩放
            try (ByteArrayInputStream bis = new ByteArrayInputStream(imageData.getImage())) {
                BufferedImage image = ImageIO.read(bis);
                int targetWidth = (int) ((double) sheet.getColumnWidth(cell.getColumnIndex()) / STANDARD_CHARACTER_WIDTH * CHARACTER_2_PIXEL_FACTOR);
                int targetHeight = (int) (rowHeight * 1.0 / PIXEL_2_INCH_FACTOR * DPI);

                // 计算图像的缩放比例
                double scaleX = (double) targetWidth / image.getWidth();
                double scaleY = (double) targetHeight / image.getHeight();
                double scale = Math.min(scaleX, scaleY);

                // 计算缩放后的图像大小
                int scaledWidth = (int) (image.getWidth() * scale);
                int scaledHeight = (int) (image.getHeight() * scale);

                // 计算上下左右四个角的空白
                int topPadding = (targetHeight - scaledHeight) / 2;
                int bottomPadding = targetHeight - scaledHeight - topPadding;
                int leftPadding = (targetWidth - scaledWidth) / 2;
                int rightPadding = targetWidth - scaledWidth - leftPadding;

                // 行高(点)= 像素高度 / 1.3333
                imageData.setTop((int) (topPadding / ROW_HEIGHT_2_PIXEL_FACTOR));
                imageData.setBottom((int) (bottomPadding / ROW_HEIGHT_2_PIXEL_FACTOR));
                imageData.setLeft((int) (leftPadding / ROW_HEIGHT_2_PIXEL_FACTOR));
                imageData.setRight((int) (rightPadding / ROW_HEIGHT_2_PIXEL_FACTOR));
            } catch (Exception e) {
                log.error("QuotationCustomCellWriteHandler afterCellDataConverted err", e);
            }
            CellWriteHandler.super.afterCellDataConverted(writeSheetHolder, writeTableHolder, cellData, cell, head, relativeRowIndex, isHead);
        }
    }
相关推荐
张先shen17 分钟前
Elasticsearch RESTful API入门:全文搜索实战(Java版)
java·大数据·elasticsearch·搜索引擎·全文检索·restful
天河归来1 小时前
springboot框架redis开启管道批量写入数据
java·spring boot·redis
张先shen1 小时前
Elasticsearch RESTful API入门:全文搜索实战
java·大数据·elasticsearch·搜索引擎·全文检索·restful
codervibe1 小时前
如何用 Spring Security 构建无状态权限控制系统(含角色菜单控制)
java·后端
codervibe1 小时前
项目中如何用策略模式实现多角色登录解耦?(附实战代码)
java·后端
TCChzp1 小时前
synchronized全链路解析:从字节码到JVM内核的锁实现与升级策略
java·jvm
大葱白菜1 小时前
🧩 Java 枚举详解:从基础到实战,掌握类型安全与优雅设计
java·程序员
笑衬人心。1 小时前
在 Mac 上安装 Java 和 IntelliJ IDEA(完整笔记)
java·macos·intellij-idea
SimonKing2 小时前
颠覆传统IO:零拷贝技术如何重塑Java高性能编程?
java·后端·程序员
sniper_fandc2 小时前
SpringBoot系列—MyBatis(xml使用)
java·spring boot·mybatis