poi_tl处理图片

文章介绍了如何使用poi_tl模板引擎在Excel或Word中处理一个字段下的多张图片,实现按比例缩放并适应单元格大小。通过Java代码示例展示了动态计算图片宽高比,以确保每行最多放置3张图片,并保持图片的自适应显示。此外,还提供了一个重写策略类用于渲染图片。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一个字段下多张图片的按比例缩放显示

使用过poi_tl模板引擎的童鞋都知道,单张图片用{{@var}}标签即可实现,下面是多张图片的实现方式以及图片如何在单元格内实现按比例缩放显示。

实现代码

Map<String,Object> dynamicTableMap = new HashMap<>();
//图片数据
List<String> screenShots = param.getJSONArray("images").toJavaList(String.class);
//单元格的最大宽度
Integer maxWidth = 620;
//单元格最大高度
Integer maxHeight = 300;
Integer rowSize = 3; // 每行放3张图片
List<PictureRenderData> pictures = new ArrayList<>();
List<String> resizeList = new ArrayList<>();

//放3张图片的宽高比 默认
double defaultWidthByHeight = (maxWidth / rowSize) * 1.0 / maxHeight;
double singleWidth = maxWidth * 1.0 / rowSize; //一行放3个,单个宽度
double fixWidth = maxWidth * 1.0 / 2; //如果只有一张,且按照宽度调整的,图片就用这个宽度,可按实际情况调整
//要调整宽度或高度的图片数量
Integer resizePic = screenShots.size() % rowSize;
for (int i = 0;i<screenShots.size();i++){
    File picture = new File(screenShots.get(i));
    BufferedImage sourceImg = ImageIO.read(picture);
    double imageWidthByHeight = sourceImg.getWidth() * 1.0 / sourceImg.getHeight();
    //这个是要调整的图片
    if(screenShots.size() - i <= resizePic){
        //动态的宽高比
        double dynamicWidthByHeight = (maxWidth / resizePic) * 1.0 / maxHeight;
        if(imageWidthByHeight - dynamicWidthByHeight >0 ){
            pictures.add(new PictureRenderData((int)fixWidth,(int)((sourceImg.getHeight()*fixWidth)/sourceImg.getWidth()),screenShots.get(i)));
        }else if(imageWidthByHeight - dynamicWidthByHeight <0) {
            pictures.add(new PictureRenderData(((sourceImg.getWidth()*maxHeight)/sourceImg.getHeight()),maxHeight,screenShots.get(i)));
        }
    }
    else{
        if (imageWidthByHeight - defaultWidthByHeight > 0) {
            pictures.add(new PictureRenderData((int) singleWidth, (int) ((sourceImg.getHeight() * singleWidth) / sourceImg.getWidth()), screenShots.get(i)));
        } else if (imageWidthByHeight - defaultWidthByHeight < 0) {
            pictures.add(new PictureRenderData(((sourceImg.getWidth() * maxHeight) / sourceImg.getHeight()), maxHeight, screenShots.get(i)));
        }
    }

}
dynamicTableMap.put("images",pictures);
Configure config = Configure.builder()
                .bind("images",new MultiImageRenderPolicy())
    			.useSpringEL(false).build();
        XWPFTemplate template = XWPFTemplate.compile(resource, config).render(dynamicTableMap);
        template.writeToFile("d:/out_example_payment.docx");



    //重写策略 
    public class MultiImageRenderPolicy extends AbstractRenderPolicy<List<PictureRenderData>>{
        @Override
        public void doRender(RenderContext<List<PictureRenderData>> renderContext) throws Exception {
            WhereDelegate whereDelegate = renderContext.getWhereDelegate();
            Object obj = renderContext.getThing();
            if(obj instanceof List){
                List<PictureRenderData> pictureRenderDataList = renderContext.getThing();
                if(pictureRenderDataList != null && pictureRenderDataList.size() !=0){
                    for(PictureRenderData pic : pictureRenderDataList){
                        InputStream stream = null;;
                        try{
                            stream = new ByteArrayInputStream(pic.getPictureSupplier().get());
                            whereDelegate.addPicture(stream,pic.getPictureType().type(),pic.getPictureStyle().getWidth(),pic.getPictureStyle().getHeight());
                        }finally {
                            IOUtils.closeQuietly(stream);
                        }
                    }
                }
            }
        }
        @Override
        protected void afterRender(RenderContext<List<PictureRenderData>> context) {
            // 清空标签
            clearPlaceholder(context, true);
        }
    }

模板绑定格式

在这里插入图片描述

实现效果

3张图片
2张图片

总结说明

  1. 每行显示几张图片用户可以自己定义,设置rowSize即可
  2. 当图片数量大于rowSize时,会对多出来的图片重新计算图片宽度
  3. 文中的宽度高度值都是像素值
### 使用 poi-tl 生成透明图片 为了实现使用 `poi-tl` 库创建带有透明背景的图像文件,可以利用 Apache POI 的扩展库 `poi-ooxml-schemas` 和其他辅助工具来操作 Excel 文件中的图形对象。具体来说,在处理图片插入时,可以通过设置特定参数确保所添加的图片具有透明属性。 下面是一个简单的 Python 脚本示例,展示了如何通过 `poi-tl` 插入一张支持 alpha 通道(即允许部分像素完全不显示从而形成透明效果)PNG 图片到工作表中: ```python from PIL import Image, ImageDraw, ImageFont import os from poitl import Workbook def create_transparent_image(text="Sample Text", output_path='transparent.png'): img = Image.new('RGBA', (400, 200), color=(255, 255, 255, 0)) # 创建一个新的 RGBA 模式的空白图层 d = ImageDraw.Draw(img) try: font = ImageFont.truetype("arial.ttf", size=70) # 尝试加载字体 except IOError: font = ImageFont.load_default() # 如果失败则使用默认字体 text_width, text_height = d.textsize(text, font=font) position_x = int((img.width - text_width)/2) position_y = int((img.height - text_height)/2) d.text((position_x , position_y ), text=text, fill=(0, 0, 0, 180), font=font) img.save(output_path,"PNG") create_transparent_image() wb = Workbook() ws = wb.active image_path = 'transparent.png' with open(image_path,'rb') as f: image_data = f.read() ws.add_picture(image=image_data,left=10,top=10,width=200,height=100,alpha=True)[^1] os.remove(image_path) # 清理临时文件 ``` 此脚本首先定义了一个函数用于生成包含指定文本并保存为 PNG 格式的半透明图像;接着实例化了 `Workbook` 对象,并在其活动的工作表上指定了要放置该图像的位置以及尺寸大小,最后设置了 `alpha=True` 参数使得插入后的图片能够保持其原有的透明度特性。 需要注意的是上述代码片段假设读者已经安装好了必要的依赖包如 Pillow (`PIL`) 及 `poitl` 。如果尚未完成这一步骤,则需先执行相应的 pip 安装命令。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值