当前位置: 首页 > news >正文

Java通过Html(ftl模板)生成PDF实战, 可支持商用

Java通过Html(freemarker模板)生成PDF实战, 可支持商用

技术架构

springboot + freemarker + [pdfbox] + flying-saucer-pdf

生成流程:

  1. freemarker: 根据数据填充ftl模板文件,得到包含有效数据的html文件(包含页眉页脚页码的处理,和解决中文渲染等问题)。
  2. flying-saucer-pdf: 将html转换成PDF文件。
  3. pdfbox: 操作PDF文件,完成加解密等操作。

依赖包

<!-- springboot版本, 2.2.x也是支持的 -->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><version>3.3.0</version>
</dependency>
<!-- load freemarker template file --><dependency><groupId>org.freemarker</groupId><artifactId>freemarker</artifactId><version>2.3.30</version></dependency>
<!-- convert html to pdf --><dependency><groupId>org.xhtmlrenderer</groupId><artifactId>flying-saucer-pdf</artifactId><version>9.4.1</version></dependency><dependency><groupId>org.xhtmlrenderer</groupId><artifactId>flying-saucer-core</artifactId><version>9.4.1</version></dependency><!-- operate pdf, such as encypt/decypt,不做加解密可不引用 --><dependency><groupId>org.apache.pdfbox</groupId><artifactId>pdfbox</artifactId><version>2.0.24</version></dependency><!-- encrypt/decrypt zip --><dependency><groupId>net.lingala.zip4j</groupId><artifactId>zip4j</artifactId><version>2.11.5</version></dependency>
PS:网上很多文章使用 itext5/7来生成PDF的,用于个人学习或者开源项目确实没问题,但是不方便用于公司商业项目,因为itext是基于AGPL 的开源协议,商用是需要收费的,当然贵司愿意付费也是okay的。

1. 准备PDF的模板文件(freemarker文件)

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><title>PDF Demo Title</title><style>@page {size: A4;margin: 35mm 10mm 23mm 10mm;@top-center {content: element(headerTop);}@bottom-center {content: element(footerBottom);}}#pagenumber:before {content: counter(page);}#pagecount:before {content: counter(pages);}.headerTop {position: running(headerTop);color: white;}.footerBottom {color: #777E90;position: running(footerBottom);margin-top: 10mm;}* {padding: 0;margin: 0;}html,body {/*优先加载 Poppins英文字体,无法渲染则使用PingFang中文字体*/font-family: Poppins-Medium, PingFang, sans-serif;margin: 0 auto;}.content {color: #23262F;font-size: 8px;padding: 0 0;margin-top: 0;}.size16 {font-size: 16px;}.size12 {font-size: 12px;}.size10 {font-size: 10px;}.size8 {font-size: 8px;}.lineHeight18 {line-height: 18px;}.weight600 {font-weight: 600;}.weight500 {font-weight: 500;}.padding18 {padding: 18px;}.marginBottom8 {margin-bottom: 8px;}.marginLeft6 {margin-left: 6px;}.marginLeft24 {margin-left: 24px;}.marginLeft16 {margin-left: 16px;}.marginLeft13 {margin-left: 13px;}.marginLeft4 {margin-left: 4px;}.marginTop4 {margin-top: 4px;}.marginTop8 {margin-top: 8px;}.width140 {width: 140px;}.widthFull {width: 100%;}.heightFull {height: 100%;}.textAlignCenter {text-align: center;}.inlineCenter {display: inline-block;vertical-align: top;}.backGray {background-color: #F7F7F7;}.backBlue {background-color: #0E59F0;}.border {border: 1px solid #E5E5E5;}.colorBlue {color: #6E9BF6;}.colorDark {color: #777E90;}.colorBlack {color: #23262F;}.colorBottomLine {background-color: #6E9BF6;}.colorGray {color: #838CA4;}.table {border-spacing: 0;/*跨页表格标题*/-fs-table-paginate: paginate;}.table>thead>tr>th,.table>tbody>tr>td {padding: 16px 4px;text-align: left;word-break: break-all;word-wrap: break-word;white-space: normal;page-break-inside: avoid;page-break-after: auto;}.table>thead>tr>th {color: #838CA4;font-weight: 400;size: 12px;}.positionAbsolute {position: absolute;}.rightBottom8 {top: 80px;right: 30px;}.height8 {height: 8px;}.height6 {height: 6px;}.height16 {height: 16px;}.right0 {right: 0;}.lineGary {height: 1px;background-color: #E5E5E5;}</style>
</head><body>
<!-- 页眉 -->
<div class="headerTop"><img class="widthFull" style="margin-left: -1px" width="716px"src="" /><div class="positionAbsolute rightBottom8 size12 weight500">${headerDate}</div>
</div><!-- 页脚 -->
<div class="footerBottom"><div class="height6 colorBottomLine"></div><div class="marginTop8 size10"><div class="inlineCenter">footer name<span class="marginLeft4 colorBlack"><![CDATA[${footerName}]]></span></div><!-- 页码 --><div class="inlineCenter colorBlack positionAbsolute right0"><span id="pagenumber"></span>/<span id="pagecount"></span></div></div>
</div><div class="content"><!-- 用户信息 --><div class="border padding18"><div class="colorBlue marginBottom8 weight500 size16"><![CDATA[${userName}]]></div><div class=""><span>Account Name</span><span class="colorBlue marginLeft6 weight500"><![CDATA[${accountName}]]></span><span class="marginLeft24">Account ID</span><span class="colorBlue marginLeft6 weight500"><![CDATA[${accountId}]]></span></div></div><div class="height8"></div><!-- 表格 --><section style="display: ${display}"><div class="height16"></div><div class=""><span class="colorBlack size16 weight600 marginLeft4 inlineCenter">Deposit</span></div><div class="marginTop8 weight500 colorBlack size10">Deposit History</div><div class="height16"></div><div class="lineGary"></div><div class="marginTop8"><table class="widthFull table"><thead><tr class="backGray"><th>Currency</th><th>Request ID</th><th>Bank Name</th><th>Bank Number</th><th>Amount</th><th>Time</th></tr></thead><tbody><#if list?? && (list?size> 0)><#list list as detail><tr><td>${detail.tokenCode}</td><td>${detail.orderNo}</td><td><#if detail.bankName??><![CDATA[${detail.bankName}]]><#else>-</#if></td><td><#if detail.bankNumber??><#-- 处理特殊字符的渲染 --><![CDATA[${detail.bankNumber}]]><#else>-</#if></td><td><#if detail.qty??>${detail.qty}<#else>-</#if></td><td><#if detail.updateTime??>${detail.updateTime}<#else>-</#if></td></tr></#list></#if></tbody></table></div><div class="height16"></div><div class="lineGary"></div></section>
</div>
</body>
</html>
freemarker模板注意事项:
  1. 图片需要通过base64的方式加载,直接加载图片路径可能无法渲染
  2. 字体名称需要和Java代码中加载的字体名称保持一致,中文无法渲染可能是没有设置别名
  3. 替换的变量,如果有null值需要在模板中判断
  4. 如果填充的变量中存在特殊字符,通过<![CDATA[${变量名}]]> 方式设置
  5. 部分高级的CSS样式或者标签可能不支持
  6. 页眉页脚采用running的方式处理

2. Java Code

2.1 FreeMarkerUtils

import freemarker.template.Configuration;
import freemarker.template.DefaultObjectWrapper;
import freemarker.template.Template;
import freemarker.template.TemplateException;
import lombok.extern.slf4j.Slf4j;import java.io.*;
import java.nio.charset.StandardCharsets;
import java.util.Map;@Slf4j
public class FreeMarkerUtils {private static Template getTemplate(String templateFileName) {Configuration configuration = new Configuration(Configuration.VERSION_2_3_29);Template template = null;try {configuration.setObjectWrapper(new DefaultObjectWrapper());//设置编码格式configuration.setDefaultEncoding("UTF-8");//模板文件configuration.setClassForTemplateLoading(FreeMarkerUtils.class, "/templates");template = configuration.getTemplate(templateFileName + ".ftl", StandardCharsets.UTF_8.toString());} catch (IOException e) {e.printStackTrace();log.error("get template file failed, fileName:{}", templateFileName);}return template;}public static String generateHtmlStr(Map<String, Object> variables, String templateFileName) {Template template = getTemplate(templateFileName);StringWriter stringWriter = new StringWriter();template.setEncoding("UTF-8");try (BufferedWriter writer = new BufferedWriter(stringWriter)) {template.process(variables, writer);String htmlStr = stringWriter.toString();writer.flush();return htmlStr;} catch (TemplateException e) {throw new RuntimeException(e);} catch (IOException e) {throw new RuntimeException(e);}}/*** 删除xml无法识别的非法字符* @param content* @return*/public static String removeIllegalChar(String content) {return content.replaceAll("[\\x00-\\x08\\x0b-\\x0c\\x0e-\\x1f]", "");}
}

2.2 PdfTest Code


import com.janche.pdf.utils.FileUtil;
import com.janche.pdf.utils.FreeMarkerUtils;
import com.janche.pdf.vo.PdfVo;
import com.lowagie.text.pdf.BaseFont;
import freemarker.cache.ClassTemplateLoader;
import lombok.extern.slf4j.Slf4j;
import org.apache.pdfbox.pdmodel.PDDocument;
import org.apache.pdfbox.pdmodel.encryption.AccessPermission;
import org.apache.pdfbox.pdmodel.encryption.StandardProtectionPolicy;
import org.xhtmlrenderer.pdf.ITextFontResolver;
import org.xhtmlrenderer.pdf.ITextRenderer;import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.math.BigDecimal;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.IntStream;/*** @Description:* @Auther: lirong* @Date: 2024/04/16*/
@Slf4j
public class PdfTest {public static void main(String[] args) {ITextRenderer renderer = new ITextRenderer();try {addPdfFont(renderer);String htmlStr = FreeMarkerUtils.generateHtmlStr(loadPdfData(), "pdfTemplate");renderer.setDocumentFromString(FreeMarkerUtils.removeIllegalChar(htmlStr));renderer.layout();} catch (IOException e) {throw new RuntimeException(e);}File pdfFile = new File("output/demo.pdf");File encryptPdfFile = new File("output/demo_encrypt.pdf");File zipFile = new File("output/demo.zip");try (OutputStream os = new FileOutputStream(pdfFile)) {renderer.createPDF(os);// encrypt pdf, if needn't encrypt pdf file, can remove pdfBox dependencyPDDocument document = PDDocument.load(pdfFile);StandardProtectionPolicy policy = new StandardProtectionPolicy("123456", "1234", new AccessPermission());policy.setEncryptionKeyLength(128);policy.setPermissions(new AccessPermission());document.protect(policy);document.save(encryptPdfFile);document.close();} catch (Exception e) {throw new RuntimeException(e);}log.info("PDF file generated successfully!");}private static void addPdfFont(ITextRenderer renderer) throws IOException {// add English fontClassTemplateLoader classTemplateLoader = new ClassTemplateLoader(FreeMarkerUtils.class, "/static/font");String enFontPath = classTemplateLoader.getBasePackagePath() + "Poppins-Medium.ttf";ITextFontResolver fontResolver = renderer.getFontResolver();// the first font needn't set aliasfontResolver.addFont(enFontPath, BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);// add Chinese font, the second font must set aliasString chFontPath = classTemplateLoader.getBasePackagePath() + "PingFang-Regular.ttf";fontResolver.addFont(chFontPath, "PingFang", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED, null);}public static Map<String, Object> loadPdfData() {Map<String, Object> data = new HashMap<>();// 启用data.put("display", "block");// 隐藏
//        data.put("display", "none");data.put("footerName", "footer-龍");data.put("userName", "龍年發财");data.put("accountName", "Jackson-龍");data.put("accountId", "234324");LocalDateTime now = LocalDateTime.now();DateTimeFormatter dateFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");data.put("headerDate", now.format(dateFormatter));ArrayList<PdfVo> tableList = new ArrayList<>();IntStream.range(1, 20).forEach(i -> {LocalDateTime dateTime = now.plusDays(i);PdfVo pdfVo = PdfVo.builder().tokenCode("USD" + i).bankName("HK Bank" + i).bankNumber("&23**94&" + i).qty(BigDecimal.TEN.add(new BigDecimal(i))).orderNo("ab123445" + i).updateTime(dateTime.format(dateFormatter)).build();tableList.add(pdfVo);});data.put("list", tableList);return data;}
}
PS:
  1. 关于中文字体不显示的问题: 一般是字体未正确加载,或者读取时字体名称不正确,完全不需要去更改什么源码class文件,高版本的flying-saucer-pdf 早已支持。
  2. 加载多个字体时,后面的字体需要设置别名,ftl模板中也需要使用设置的别名。
字体文件:(可在文章底部的github项目中获取)
  1. PingFang-Regular.ttf 中文字体
  2. Poppins-Medium.ttf 英文字体

3. PDF展示

在这里插入图片描述

Github源码下载:https://github.com/Janche/springboot-html2pdf-demo

相关文章:

  • 从零学会【分镜头意识】拍摄思维
  • 一个与 WSL2 建立远程的简单方法
  • 基于朴素贝叶斯算法的新闻类型预测,django框架开发,前端bootstrap,有爬虫有数据库
  • 数字认证携手华为鸿蒙生态,升级智慧办公新体验
  • 【Qt】探索Qt绘图世界:自定义控件与视觉效果的全面指南
  • ggplot2绘图如何根据一个变量即区分颜色又区分深浅?
  • 超强算力 Orange Pi Kunpeng Pro 开发板基础测评与体验
  • pyqt绘制各种直线
  • LLM基础知识
  • LLM主要类别架构
  • PTA字符串删除
  • 19、matlab信号预处理中的中值滤波(medfilt1()函数)和萨维茨基-戈雷滤波滤(sgolayfilt()函数)
  • 【Python Cookbook】S01E14 从字典中提取子集
  • 【Linux】进程(4):优先级
  • 论文笔记:Image Anaimation经典论文-运动关键点模型(Monkey-Net)
  • php的引用
  •  D - 粉碎叛乱F - 其他起义
  • iOS筛选菜单、分段选择器、导航栏、悬浮窗、转场动画、启动视频等源码
  • mac修复ab及siege安装
  • Mac转Windows的拯救指南
  • MySQL主从复制读写分离及奇怪的问题
  • node.js
  • nodejs调试方法
  • React系列之 Redux 架构模式
  • 安卓应用性能调试和优化经验分享
  • 创建一个Struts2项目maven 方式
  • 技术发展面试
  • 开源SQL-on-Hadoop系统一览
  • 七牛云 DV OV EV SSL 证书上线,限时折扣低至 6.75 折!
  • 七牛云假注销小指南
  • 十年未变!安全,谁之责?(下)
  • 一个普通的 5 年iOS开发者的自我总结,以及5年开发经历和感想!
  • zabbix3.2监控linux磁盘IO
  • 回归生活:清理微信公众号
  • # .NET Framework中使用命名管道进行进程间通信
  • # 消息中间件 RocketMQ 高级功能和源码分析(七)
  • #ifdef 的技巧用法
  • $ git push -u origin master 推送到远程库出错
  • $.extend({},旧的,新的);合并对象,后面的覆盖前面的
  • (C#)获取字符编码的类
  • (C语言)输入自定义个数的整数,打印出最大值和最小值
  • (delphi11最新学习资料) Object Pascal 学习笔记---第5章第5节(delphi中的指针)
  • (pojstep1.1.1)poj 1298(直叙式模拟)
  • (pojstep1.3.1)1017(构造法模拟)
  • (附源码)计算机毕业设计SSM疫情居家隔离服务系统
  • (回溯) LeetCode 46. 全排列
  • (一)SpringBoot3---尚硅谷总结
  • (幽默漫画)有个程序员老公,是怎样的体验?
  • (转)人的集合论——移山之道
  • (转)重识new
  • ***汇编语言 实验16 编写包含多个功能子程序的中断例程
  • .aanva
  • .NET CORE 第一节 创建基本的 asp.net core
  • .net core docker部署教程和细节问题
  • .NET Core使用NPOI导出复杂,美观的Excel详解