Java通过Html(freemarker模板)生成PDF实战, 可支持商用
技术架构
springboot + freemarker + [pdfbox] + flying-saucer-pdf
生成流程:
- freemarker: 根据数据填充ftl模板文件,得到包含有效数据的html文件(包含页眉页脚页码的处理,和解决中文渲染等问题)。
- flying-saucer-pdf: 将html转换成PDF文件。
- 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模板注意事项:
- 图片需要通过base64的方式加载,直接加载图片路径可能无法渲染
- 字体名称需要和Java代码中加载的字体名称保持一致,中文无法渲染可能是没有设置别名
- 替换的变量,如果有null值需要在模板中判断
- 如果填充的变量中存在特殊字符,通过<![CDATA[${变量名}]]> 方式设置
- 部分高级的CSS样式或者标签可能不支持
- 页眉页脚采用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 dependency
PDDocument 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 font
ClassTemplateLoader classTemplateLoader = new ClassTemplateLoader(FreeMarkerUtils.class, "/static/font");
String enFontPath = classTemplateLoader.getBasePackagePath() + "Poppins-Medium.ttf";
ITextFontResolver fontResolver = renderer.getFontResolver();
// the first font needn't set alias
fontResolver.addFont(enFontPath, BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
// add Chinese font, the second font must set alias
String 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:
- 关于中文字体不显示的问题: 一般是字体未正确加载,或者读取时字体名称不正确,完全不需要去更改什么源码class文件,高版本的flying-saucer-pdf 早已支持。
- 加载多个字体时,后面的字体需要设置别名,ftl模板中也需要使用设置的别名。
字体文件:(可在文章底部的github项目中获取)
- PingFang-Regular.ttf 中文字体
- Poppins-Medium.ttf 英文字体
3. PDF展示
Github源码下载:https://github.com/Janche/springboot-html2pdf-demo