首页 前端知识 Java通过Html(ftl模板)生成PDF实战, 可支持商用

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

2024-06-26 23:06:55 前端知识 前端哥 47 330 我要收藏

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 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:
  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

转载请注明出处或者链接地址:https://www.qianduange.cn//article/13640.html
评论
发布的文章

在echarts中使用geojson地图

2024-07-29 00:07:05

大家推荐的文章
会员中心 联系我 留言建议 回顶部
复制成功!