(最新)itext7 freemarker动态模板转pdf

1.引入依赖

<!--PDF导出POM-->
<dependency><groupId>com.itextpdf</groupId><artifactId>itext7-core</artifactId><version>8.0.3</version><type>pom</type>
</dependency>
<dependency><groupId>com.itextpdf</groupId><artifactId>html2pdf</artifactId><version>5.0.3</version>
</dependency>
<dependency><groupId>org.freemarker</groupId><artifactId>freemarker</artifactId><version>2.3.31</version>
</dependency>

2.编写FTL文件

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title></title>
</head><style>.pic {width: 100%;height: 100%;}
</style><body>
<div id="app"><#--封面图--><#if show><img class="pic" src="${coverImageFile!''}"/></#if>
</div>
</body>
</html>

3.后台代码

@Resource
Configuration configuration;
/*** 导出*/@Overridepublic void export(HttpServletResponse response){//获取文件byte[]byte[] pdfBytes=getPdfBytes();try{//输出pdf文件String chineseName="中文文件名";response.setCharacterEncoding("utf-8");String fileName=URLEncoder.encode(chineseName, StandardCharsets.UTF_8).replaceAll("\\+", "%20");response.setHeader("Content-disposition", "attachment;filename*=" + fileName + ".pdf");//设置响应头(导出为pdf文件)response.setContentType("application/pdf");//设置响应头(返回流给前端预览使用)//response.setContentType("application/octet-stream;charset=UTF-8");// 获取HttpServletResponse的OutputStreamOutputStream os=response.getOutputStream();// 写入字节流数组os.write(pdfBytes);os.flush();os.close();}catch(IOException e){throw new InvalidCmdException(e.getMessage());}}
private byte[] getPdfBytes(){//获取模板替换数据Map<String, Object> map=new HashMap<>();map.put("show", true);map.put("coverImageFile", "base64图片");try{//读取freemarker模板Template template=configuration.getTemplate("FTL文件路径");StringWriter out=new StringWriter();// 将数据模型合并到模板中生成HTMLtemplate.process(map, out);String htmlContent=out.toString();// 初始化字节输出流用于保存PDFByteArrayOutputStream pdfOutput=new ByteArrayOutputStream();// 设置中文字体ConverterProperties converterProperties=new ConverterProperties();FontProvider fontProvider=new DefaultFontProvider(true, true, false);PdfDocument pdfDoc;FontProgram fontProgram=FontProgramFactory.createFont("static/simhei.ttf");// 添加中文字体到字体提供者fontProvider.addFont(fontProgram);converterProperties.setFontProvider(fontProvider);// 转换HTML到PDFtry(ByteArrayOutputStream outputStream=pdfOutput){// 初始化文档PdfWriter writer=new PdfWriter(outputStream);pdfDoc=new PdfDocument(writer);//设置pdf纸张大小pdfDoc.setDefaultPageSize(PageSize.A4);// 将HTML转换为PDFHtmlConverter.convertToPdf(htmlContent, pdfDoc, converterProperties);}// 从字节输出流中获取PDF的字节数组byte[] pdfBytes=pdfOutput.toByteArray();// 关闭流pdfOutput.close();pdfDoc.close();out.close();return pdfBytes;}catch(IOException e){throw new InvalidCmdException("读取候选人freemarker模板错误");}catch(TemplateException e){throw new InvalidCmdException("替换模板值错误");}}

4.注意事项

itext7本身不支持中文 所以需要在项目中引入字体文件。

freemarker图片需要使用base64格式图片

byte[] 数组转base64

 /*** 图片转base64*/private String convertBase64(String ossFileId){if(String2Utils.isNotBlank(ossFileId)){//这里使用的是阿里的oss查询OssUrlDto ossUrlDto=ossFileService.findDfsById(ossFileId).orElse(null);try{byte[] adviserBytes=ossClientService.getContentByteByKey(ossUrlDto.getBucketName(), ossUrlDto.getPath());return "data:image/jpeg;base64," + Base64.getEncoder().encodeToString(adviserBytes);}catch(Exception e){throw new InvalidCmdException("图片转base64失败");}}return "";}

给导出的PDF设置页面边距,网上搜索了很多中办法,只有一种成功了

再FTL页面中添加

<style>@page {size: A4;margin: 10mm;}
</style>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/817071.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

BackTrader 中文文档(七)

原文&#xff1a;www.backtrader.com/ TA-Lib 原文&#xff1a;www.backtrader.com/docu/talib/talib/ 即使backtrader提供了大量内置指标&#xff0c;而且开发指标主要是定义输入、输出并以自然方式编写公式&#xff0c;一些人还是想要使用TA-LIB。一些原因包括&#xff1a; 指…

基于SignalR视频聊天 一

环境 VS2022 WIN10 .NET8 VSCode VUE SignalR 1.安装SignalR客户端库 需要在Vue.js项目中安装SignalR客户端库。可以使用npm或者yarn来安装 npm install microsoft/signalr2.创建SignalR服务 创建SignalR服务&#xff0c;以便客户端&#xff08;Vue.js应用&#xff09;能…

抄袭可耻 - 2023面试高手抄袭对比图

原创博客(伏城之外)抄袭博客(2023面试高手)对比图华为OD机试 - 跳马(Java & JS & Python & C & C++)_华为od岗c卷机试马走日-CSDN博客2024年华为OD机试真题-跳马-Python-OD统一考试(C卷)-CSDN博客

集合体系java

Collection:单列集合&#xff1a;每个元素只包含一个值 Collection集合存储的是地址 Collection的三种遍历方法如下 //迭代器是用来遍历集合的专用方式&#xff08;数组没有迭代器&#xff09;&#xff0c;在java中迭代器的代表是Iterator //boolean hasNext():询问当前位置…

Java中队列

队列是一种常见的数据结构&#xff0c;它按照先进先出&#xff08;FIFO&#xff09;的原则管理元素。在 Java 中&#xff0c;队列通常是通过链表或数组实现的&#xff0c;不同的实现类在内部数据结构和操作上可能有所不同。 1.原理 1.数据结构&#xff1a;队列的基本数据结构…

【python图形界面问题解决】wxPython创建图形界面程序,在代码编译器中正常运行,但是打包后却不能运行解决办法

一、问题 使用wxPython创建一个图形界面&#xff0c;在VSCODE中正常运行&#xff0c;但是打包后&#xff0c;却不能运行&#xff0c;只出现一个一闪而过的窗口&#xff0c;这时最需要看看这窗口到底显示了什么内容。这里可以使用录屏软件录制屏幕&#xff0c;这里使用LICEcap小…

Android多线程编程

前言 本文由于介绍Android多线程编程的学习。 线程基本用法 定义线程有两种方式&#xff0c;分别是继承Thread类、实现Runnable接口&#xff1a; 继承Thread类&#xff1a;只需新建一个类继承自Thread&#xff0c;然后重写父类的run()方法&#xff0c;在这个方法里面写耗时…

美国卖家需知!儿童玩具CPC认证ASTMF-23标准更新

2023年10月13日&#xff0c;美国材料与试验协会&#xff08;ASTM&#xff09;发布了最新版本的玩具安全标准ASTM F963-23。这一标准的修订涵盖了声学、电池可及性、充气材料、弹射玩具等技术方面的要求。同时&#xff0c;它还为邻苯二甲酸盐和重金属在玩具基材中的使用提供了豁…

Towards IP Geolocation Using Delay and TopologyMeasurements(TBG)(2006年)

下载地址:Towards IP geolocation using delay and topology measurements | Proceedings of the 6th ACM SIGCOMM conference on Internet measurement 被引次数:492 Katz-Bassett E, John J P, Krishnamurthy A, et al. Towards IP geolocation using delay and topology …

通讯录的实现(单链表版本)

我们首先要知道通讯录的实现是基于单链表的基础上的&#xff0c;所以我们首先要搞懂单链表。&#xff08;注意&#xff1a;今天的代码量较多&#xff09;&#xff0c;但这不是阻挡我们前进的脚步&#xff0c;冲冲冲&#xff01;&#xff01;&#xff01; 单链表的简要概述 我们…

Scala---集合(数组,Map,元组(Tuple),Zip拉链)详解

scala的集合分为了两类&#xff0c;一类是可变的集合&#xff08;集合可以执行增删改查操作&#xff09;&#xff0c;另一类是不可变集合&#xff08;集合元素在初始化的时候确定&#xff0c;后续只能进行查&#xff0c;有的可以进行修改&#xff0c;有的不可以&#xff09;。二…

Transformers 微调

Transformers 微调 基于 Transformers 实现模型微调训练的主要流程数据字段数据拆分&#xff08;分成训练跟测试&#xff09;下载数据集数据集抽样预处理数据数据抽样微调训练配置加载 BERT 模型训练超参数&#xff08;TrainingArguments&#xff09;模型权重保存路径(output_d…

2024.4.19 Python爬虫复习day07 可视化3

综合案例 需求: 已知2020年疫情数据,都是json数据,需要从文件中读出,进行处理和分析,最终实现数据可视化折线图 相关知识点: json json简介: 本质是一个特定格式的字符串 举例: [{},{},{}] 或者 {}python中json包: import jsonpython数据转为json数据: 变量接收json…

微服务架构使用和docker部署方法(若依)

这里以若依官方网站开源的微服务框架为例子记录使用方法过程。 开源地址&#xff1a;RuoYi-Cloud: &#x1f389; 基于Spring Boot、Spring Cloud & Alibaba的分布式微服务架构权限管理系统&#xff0c;同时提供了 Vue3 的版本 下载后&#xff0c;用IDEA社区版开发工具打…

【量化交易】顶底分型策略

在众多的量化策略中&#xff0c;顶底分型策略因其独特的市场趋势捕捉能力和简洁的实现方式而受到许多投资者的青睐。本文将详细介绍顶底分型策略的原理&#xff0c;并展示如何使用Python在聚宽平台上实现这一策略。 感兴趣的朋友&#xff0c;可以在下方公号内回复&#xff1a;0…

GNU Radio Radar Toolbox编译及安装

文章目录 前言一、GNU Radio Radar Toolbox 介绍二、gr-radar 安装三、具体使用四、OFDM 雷达仿真 前言 GNU Radio Radar Toolbox&#xff08;gr-radar&#xff09;是一个开放源码的工具箱&#xff0c;用于 GNU Radio 生态系统&#xff0c;主要目的是为雷达信号处理提供必要的…

vue源码解析——diff算法/双端比对/patchFlag/最长递增子序列

虚拟dom——virtual dom&#xff0c;提供一种简单js对象去代替复杂的 dom 对象&#xff0c;从而优化 dom 操作。virtual dom 是“解决过多的操作 dom 影响性能”的一种解决方案。virtual dom 很多时候都不是最优的操作&#xff0c;但它具有普适性&#xff0c;在效率、可维护性之…

Leetcode 3111. Minimum Rectangles to Cover Points

Leetcode 3111. Minimum Rectangles to Cover Points 1. 解题思路2. 代码实现 题目链接&#xff1a;3111. Minimum Rectangles to Cover Points 1. 解题思路 这一题在这次比赛的4道题当中算是比较简单的&#xff0c;基本就只需要将所有的点排序之后然后使用贪婪算法来cover住…

【C++造神计划】运算符

1 赋值运算符 赋值运算符的功能是将一个值赋给一个变量 int a 5; // 将整数 5 赋给变量 a 运算符左边的部分叫作 lvalue&#xff08;left value&#xff09;&#xff0c;右边的部分叫作 rvalue&#xff08;right value&#xff09; 左边 lvalue 必须是一个变量 右边 rval…

木马免杀代码之python反序列化分离免杀

本篇文章主要用到python来对CobaltStrike生成的Shellcode进行分离免杀处理, 因此要求读者要有一定的python基础, 下面我会介绍pyhon反序列化免杀所需用到的相关函数和库 exec函数 exec函数是python的内置函数, 其功能与eval()函数相同, 但不同的是exec函数支持多行python代码…