spring boot使用thymeleaf模板引擎实现html文件转pdf的实现过程

一、相关说明

选择thymeleaf+html的原因:个人觉得thymeleaf有这很好的官方文档做支撑,而且语法更好理解而是用ftl模板语法比较古老很少有人使用,因此本人选择thymeleaf作为模板渲染工具进行代码的开发。

本文主要解决:

  1. html与thymeleaf的集合实现内容的渲染最后由后端生成目标文件
  2. html内容汉字的样式渲染,以及相关坑的说明
  3. 模板页眉和页脚可使用html标签实现自定义样式功能
  4. 实现文字水印和图片水印的方法
  5. 传给前端可以是文件流也可以是上传到文件管理后的文件地址

二、引入相关依赖

 <!--生成pdf依赖 start-->
        <dependency>
            <groupId>org.xhtmlrenderer</groupId>
            <artifactId>flying-saucer-pdf-itext5</artifactId>
            <version>9.4.0</version>
        </dependency>

        <dependency>
            <groupId>org.xhtmlrenderer</groupId>
            <artifactId>flying-saucer-core</artifactId>
            <version>9.4.0</version>
            <scope>compile</scope>
        </dependency>

        <dependency>
            <groupId>net.sf.jtidy</groupId>
            <artifactId>jtidy</artifactId>
            <version>r938</version>
            <scope>compile</scope>
        </dependency>

        <dependency>
            <groupId>org.thymeleaf</groupId>
            <artifactId>thymeleaf</artifactId>
            <version>3.0.6.RELEASE</version>
            <scope>compile</scope>
        </dependency>

<!--生成pdf依赖  end-->

三、准备模板和相关字体

(1)、静态文件位置参考

注大坑:关于中文汉字的一些表达形式类似于:仿宋、方正小标宋简体下载字体包后文字包的名称也是中文但是linux服务器对汉字是无法识别的所以本地测试好好的一旦上生产就拉闸了,该包名导致字体包无法使用因此要怎么该呢?按照css能够识别的样式来该,具体汉字和英文名称如何对应请参考:

中文与css样式对照表

字体包下载请参考:

官方推荐字体下载地址

在这里引入使用即可,但有些字体可能需要官方授权才能使用(方正小标宋简体),因此后端可能会有问题因此需要准备多款类似的字体作以备用。

(2)、模板准备:

header.html 页眉内容

<div th:fragment="header">
    <table>
        <tr>
            <td><span class="barcode">*TEST BARCODE*</span></td>
            <td class="right"><img class="logo" src="logo.png" alt="logo"></td>
        </tr>
    </table>
</div>

footer.html 页脚

<div th:fragment="footer">
    <hr>
    <div class="page-count" style="font-family: 仿宋_GB2312">
        <span> 第 </span> <span id="pagenumber"></span><span> 页 共 </span><span id="pagecount"></span> <span>页</span>
    </div>
</div>

style.css 样式文件

body {
    font-family: sans-serif;
    font-size: 12px;
}

@page {

    @bottom-left {
        content: element(footer);
        vertical-align: top;
        padding-top: 10px;
    }

    @top-right {
        content: element(header);
        vertical-align: bottom;
        padding-bottom: 10px;
    }

    margin-top: 3.3cm;
    margin-left: 2cm;
    margin-right: 2cm;
    margin-bottom: 3.3cm;

    size: A4 portrait;
}

div.header {
    position: running(header);
}

div.footer {
    display: block;
    margin-top: 0.5cm;
    position: running(footer);
}

#pagenumber:before {
    content: counter(page);
}

#pagecount:before {
    content: counter(pages);
}

.logo-container {
    text-align: right;
}

.page-count {
    font-family: 仿宋_GB2312, Serif;
    text-align: right;
}

.logo {
    width: 275px;
}

h1 {
    font-size: 18px;
    margin-bottom: 40px;
    margin-top: 40px;
}

p {
    font-family: SimSun, Serif;
    font-size: 12px;
}

.footer {
    text-align: center;
}

span{
    font-family: SimSun, Serif;
    font-size: 12px;
}

@font-face {
    font-family: "Bar-Code 39";
    src: url(Code39.ttf) format('truetype');
}

@font-face {
    font-family: FangSong_GB2312;
    src: url(FangSong_GB2312.ttf) format("truetype");
}
@font-face {
    font-family: FZXiaoBiaoSong-B05S;
    src: url(FZXiaoBiaoSong-B05S.TTF) format("truetype");
}

.barcode {
    font-family: "Bar-Code 39";
    font-size: 26px;
}

.right {
    text-align: right;
}

.left {
    text-align: left;
}

.bottom {
    vertical-align: bottom;
}

.address-block {
    height: 100px;
}

table {
    width: 100%;
}

hr {
    background-color: #000000;
    border: dashed #000000 0.5px;
    height: 1px;
}

.page-break {
    page-break-after: always;
}

.next-page {
    page-break-before: always
}

template.html 主体部分模板

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">

    <head>
        <meta charset="UTF-8">
        <link rel="stylesheet" type="text/css" media="all" href="style.css"/>
    </head>

    <body>
        <div class='header' th:include="header::header"></div>
<!--        页脚一定要放在正文上面不然第一页页脚加载不上-->
        <div class='footer' th:include="footer::footer"></div>
        <p >良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
        <p>良渚实验室教授委员会主要职责包括:审定良渚实验室人才引进、人才待遇体系、绩效评估等;审定良渚实验室学术相关的事项;审定良渚实验室师德师风相关的事项等。</p>
<!--        <div th:utext="${data.name}"></div>-->

    </body>

</html>

注意:这里的内容暂时写死用于测试效果如果想直接从后端捞数据则请参考thymeleaf语法

请参考:thymeleaf基本语法

四、后端生成pdf方法实现

package com.zjhc.lzlab.utils;

import com.zjhc.lzlab.constants.CommonConstants;
import org.springframework.stereotype.Component;
import org.thymeleaf.TemplateEngine;
import org.thymeleaf.templateresolver.ClassLoaderTemplateResolver;
import org.w3c.tidy.Tidy;
import org.xhtmlrenderer.pdf.ITextRenderer;
import org.thymeleaf.context.Context;

import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.HashMap;
import java.util.Map;

import static com.itextpdf.text.pdf.BaseFont.EMBEDDED;
import static com.itextpdf.text.pdf.BaseFont.IDENTITY_H;
import static org.thymeleaf.templatemode.TemplateMode.HTML;

/**
 * @ClassName: Template2PdfUtils
 * @Description: html模板转为pdf
 * @Author: TXW
 * @Date: 2024-01-12
 */
@Component
public class Template2PdfUtils {


    /**
     * @Param tempName:模板名称,contentData 模板数据
     * */
    public File generatePdf(String fondPath,String tempName, Map<String, String> contentData) throws Exception {
        ClassLoaderTemplateResolver templateResolver = new ClassLoaderTemplateResolver();
        templateResolver.setPrefix("/templates/");
        templateResolver.setSuffix(".html");
        templateResolver.setTemplateMode(HTML);
        templateResolver.setCharacterEncoding(CommonConstants.UTF_8);

        TemplateEngine templateEngine = new TemplateEngine();
        templateEngine.setTemplateResolver(templateResolver);

//        Map<String, String> data = new HashMap<>();
//        data.put("name", "<ul> <li class=\"ql-indent-1\"><strong><u>三生三世 三生三世</u></strong><img src=\"\"></img></li><li class=\"ql-indent-1\"><img src=\"\"></img></li><li class=\"ql-indent-1\">2222222测22222222222222222222222222222222222</li><li class=\"ql-indent-1\">22222222222222222222222222222222222222222222222222222222222222</li></ul>");
//
        Context context = new Context();
        context.setVariable("data", contentData);
//        File tempFile = new File("test.pdf"); //本地生成查看结果用(项目中结构中会生成)
        File tempFile = File.createTempFile("redFile", ".pdf");
        String renderedHtmlContent = templateEngine.process(tempName, context);
        String xHtml = convertToXhtml(renderedHtmlContent);

        String baseUrl = FileSystems.getDefault().getPath("src", "main", "resources")
                .toUri().toURL().toString();

        ITextRenderer renderer = new ITextRenderer();
//        String path =  "D:\\data\\font\\Code39.ttf";
//        String path1 = "D:\\data\\font\\ARIALUNI.TTF";
//        String path2 = "D:\\data\\font\\MSYH.ttc";
//        String path3 = "D:\\data\\font\\MSYHBD.ttc";
//        String path4 = "D:\\data\\font\\MSYHL.ttc";
//        String path5 = "D:\\data\\font\\SIMHEI.TTF";
//        String path6 = "D:\\data\\font\\simsun.ttc";
//        String path7 = "D:\\data\\font\\FangSong_GB2312.ttf";
//        String path8 = "D:\\data\\font\\FZXiaoBiaoSong-B05S.TTF";

        String path = fondPath + "/Code39.ttf";
        String path1 = fondPath+"/ARIALUNI.TTF";
        String path2 = fondPath+"/MSYH.ttc";
        String path3 = fondPath+"/MSYHBD.ttc";
        String path4 = fondPath+"/MSYHL.ttc";
        String path5 = fondPath+"/SIMHEI.TTF";
        String path6 = fondPath+"/simsun.ttc";
        String path7 = fondPath+"/FangSong_GB2312.ttf";
        String path8 = fondPath+"/FZXiaoBiaoSong-B05S.TTF";

        renderer.getFontResolver().addFont(path, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path1, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path2, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path3, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path4, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path5, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path6, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path7, IDENTITY_H, EMBEDDED);
        renderer.getFontResolver().addFont(path8, IDENTITY_H, EMBEDDED);
        renderer.setDocumentFromString(xHtml, baseUrl);
        renderer.layout();
        OutputStream outputStream = new FileOutputStream(tempFile);
        renderer.createPDF(outputStream);

        return tempFile;


        /**
         *
         * // 1. 读取临时文件内容都到 response
         *             try (FileInputStream fis = new FileInputStream(tempFile);
         *                  ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
         *                 byte[] buffer = new byte[1024];
         *                 int len;
         *                 while ((len = fis.read(buffer)) > 0) {
         *                     baos.write(buffer, 0, len);
         *                 }
         *                 baos.flush(); // ensure all data is written to the byte array
         *                 // 2. 将字节数组作为响应的主体返回给前端
         *                 byte[] fileBytes = baos.toByteArray();
         *                 response.setContentType("application/octet-stream"); // 设置响应内容类型为二进制流
         *                 response.setContentLength(fileBytes.length); // 设置响应内容长度
         *                 try (OutputStream os = response.getOutputStream()) {
         *                     os.write(fileBytes); // write bytes to output stream
         *                     os.flush(); // flush the stream to ensure all bytes are written out
         *                 } catch (IOException e) {
         *                     e.printStackTrace();
         *                 }
         *                 outputStream.close();
         *             } catch (IOException e) {
         *                 e.printStackTrace();
         *             }
         * */
    }


    private static String convertToXhtml(String html) throws UnsupportedEncodingException {
        Tidy tidy = new Tidy();
        tidy.setInputEncoding(CommonConstants.UTF_8);
        tidy.setOutputEncoding(CommonConstants.UTF_8);
        tidy.setXHTML(true);
        ByteArrayInputStream inputStream = new ByteArrayInputStream(html.getBytes(CommonConstants.UTF_8));
        ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
        tidy.parseDOM(inputStream, outputStream);
        return outputStream.toString(CommonConstants.UTF_8);
    }
}

 注意:在这里引入服务器 的字体和本地字体位置不一样在上车生产和测试的时候注意切换(我用的是nacos所以配置文件使用的是生产的所以可以在yml文件中注意切换)

tempFile:文件打开生成结果会在项目目录中可点开查看,在模板中使用的所有字体需要再这里引入否则文件中的字体还是无法显示。

如果是给前端传流的话可以将文件写入流中,否则将文件返回出去后使用文件上传的方法将文件上传到文件管理系统再将文件地址返回到前端可以仿照代码:

/**这是一个impl方法*/
public AjaxResult getRedFilePdfShow(HttpServletResponse response, CompositeManageRedFileVo redFileVo){
        Map<String,String> fileMsg = new HashMap<>();
        //设置pdf文件内容
        Map<String,String> data = new HashMap<>();
        data.put("title",redFileVo.getFileTitle());
        data.put("fileNo",redFileVo.getFileNo());
        data.put("signDept",redFileVo.getSignDept());
        try {
            data.put("content", URLDecoder.decode(redFileVo.getContent(),"UTF-8"));
            Template2PdfUtils template2PdfUtils = new Template2PdfUtils();
            File file = template2PdfUtils.generatePdf(fontPath, TemplatesFileEnum.RED_FILE.getTempName(), data);
            String filePath = FileUploadUtils.uploadFileByFile(basePath, file);

            fileMsg.put("fileName",file.getName());
            fileMsg.put("filePath",ip + filePath);
        } catch (Exception e) {
          e.printStackTrace();
        }
        return AjaxResult.success(fileMsg);
    }

你还可能会用到:File to MultipartFile

    public static String uploadFileByFile (String path,File file) throws IOException {
        DiskFileItem item = new DiskFileItem("file"
                , MediaType.MULTIPART_FORM_DATA_VALUE
                , true
                , file.getName()
                , (int)file.length()
                , file.getParentFile());
        try {
            OutputStream os = item.getOutputStream();
            os.write(FileUtils.readFileToByteArray(file));
        } catch (IOException e) {
            e.printStackTrace();
        }
        MultipartFile commonsMultipartFile = new CommonsMultipartFile(item);
        return upload(path, commonsMultipartFile);
    }

五、给目标文件pdf文件添加水印

下列都是实例方法可以直接将方法复制到上面的Utils方法里面进行使用

1、添加文字水印方法

/**
     * 添加文字水印
     * @param pdfPath 需要添加水印的pdf文件  例如:XXXX.pdf
     * @param outFilePath 最终生成含有水印的文件地址 例如:XXXX_waterMark.pdf
     * @param fontPath 文字的字体设置  例如:XXX/XXX/XXX.TTF
     * @param text 水印内容
     * */
    public void addTextWatermark(String pdfPath,String outFilePath, String fontPath,String text) throws IOException, DocumentException {
        // 打开现有的PDF
        PdfReader pdfReader = new PdfReader(pdfPath);
        // 创建一个新的PDF文件名用于输出带水印的PDF,初始化PdfStamper用于添加水印
        PdfStamper pdfStamper = new PdfStamper(pdfReader, new FileOutputStream(outFilePath));
        //获取pdf的总页数
        int n = pdfReader.getNumberOfPages();
        // 设置水印字体
        //BaseFont baseFont = BaseFont.createFont(BaseFont.HELVETICA, BaseFont.WINANSI, BaseFont.EMBEDDED);
        BaseFont baseFont = BaseFont.createFont(fontPath, BaseFont.IDENTITY_H, BaseFont.EMBEDDED);

        // 在PDF中添加水印
        for (int i = 1; i <= n; i++) {
            PdfContentByte overContent = pdfStamper.getOverContent(i); // 获取每一页的内容
            // 设置水印透明度
            PdfGState gs = new PdfGState();
            gs.setFillOpacity(0.6f); // 设置透明度
            overContent.setGState(gs);
            // 计算水印位置和间隔
            float xInterval = 150;
            float yInterval = 100;
            float xStart = 40;
            float yStart = 50;
            float x, y;
            int rotation = 45; // 水印旋转角度
            // 在PDF中添加重复水印
            for (x = xStart; x < PageSize.A4.getWidth(); x += xInterval) {
                for (y = yStart; y < PageSize.A4.getHeight(); y += yInterval) {
                    overContent.beginText();
                    overContent.setFontAndSize(baseFont, 50);
                    overContent.setColorFill(BaseColor.PINK);
                    overContent.showTextAligned(Element.ALIGN_CENTER, text, x, y, rotation);
                    overContent.endText();
                }
            }
        }
        // 关闭PdfStamper
        pdfStamper.close();
        pdfReader.close();
    }

效果展示:

2、添加图片水印的方法

/**
     * 添加图片水印
     * @param pdfPath 需要添加水印的pdf文件  例如:XXXX.pdf
     * @param outFilePath 最终生成含有水印的文件地址 例如:XXXX_waterMark.pdf
     * @param imagePath 图片路径 XXX.png
     * */
    public void addImageWatermark(String pdfPath,String outFilePath, String imagePath) throws IOException, DocumentException {
        // 打开现有的PDF
        PdfReader pdfReader = new PdfReader(pdfPath);
        // 初始化PdfStamper用于添加水印
        PdfStamper pdfStamper = new PdfStamper(pdfReader, new FileOutputStream(outFilePath));

        int n = pdfReader.getNumberOfPages(); // 获取PDF的总页数

// 加载水印图片
        Image image = Image.getInstance(imagePath); // 替换为您的图片路径
        image.setAbsolutePosition(200, 400); // 设置图片的位置
        image.scaleToFit(250, 250); // 设置图片的大小
        image.setRotationDegrees(-45); // 设置图片的旋转角度

// 循环每一页,添加水印
        for (int i = 1; i <= n; i++) {
            PdfContentByte content = pdfStamper.getOverContent(i); // 获取每一页的内容

            // 设置水印透明度
            PdfGState gs = new PdfGState();
            gs.setFillOpacity(0.2f); // 设置透明度
            content.setGState(gs);

            // 计算水印位置和间隔
            float xInterval = 200; // 水印的水平间隔
            float yInterval = 100; // 水印的垂直间隔
            float x, y;

            // 在PDF中添加重复水印
            for (x = 0; x < PageSize.A4.getWidth(); x += xInterval) {
                for (y = 0; y < PageSize.A4.getHeight(); y += yInterval) {
                    Image watermarkImage = Image.getInstance(image);
                    watermarkImage.setAbsolutePosition(x, y);
                    content.addImage(watermarkImage);
                }
            }
        }

        // 关闭PdfStamper
        pdfStamper.close();
        pdfReader.close();
    }

效果展示:

以上水印的方法中可能会存在文字或者透明度以及相关参数需要进行调整因此按照客户需求修改相关的值即可 

六、项目效果展示:

模板代码:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">

<head>
    <meta charset="UTF-8">
</head>
<style>
    @page {
        @bottom-center {
            content: counter(page);
            font-family: SimSun;
            vertical-align: middle;
            font-size: 21px;
            padding-bottom: 10px;
        };

/*        @bottom-right {
            content: "第 " counter(page) " 页 共 " counter(pages) " 页 ";
            font-family: SimSun;
            font-size: 21px;
            color: red;

            //自定义页脚
           @bottom-left {
            content: element(footer);
            font-family: SimSun;
            vertical-align: middle;
            font-size: 21px;
            color: red;
            padding-bottom: 10px;
        };
        }*/
        margin-top: 2cm;
        margin-left: 2cm;
        margin-right: 2cm;
        margin-bottom: 3.3cm;

        size: A4 portrait;
    }

    @font-face {
        font-family: FangSong_GB2312;
        src: url(FangSong_GB2312.ttf) format("truetype");
    }
    @font-face {
        font-family: FZXiaoBiaoSong-B05S;
        src: url(FZXiaoBiaoSong-B05S.TTF) format("truetype");
    }

    #pagenumber:before {
        content: counter(page);
    }

    #pagecount:before {
        content: counter(pages);
    }

    .title {
        text-align: center;
        color: red;
        font-family: FZXiaoBiaoSong-B05S,serif;
        font-size: 41px;
    }

    .fileNo {
        font-family: FangSong_GB2312,serif;
        font-size: 21px;
        text-align: center;
    }

    span {
        font-family: 'Times New Roman', serif;
    }

    img {
        display: block;
        margin-left: auto;
        margin-right: auto;
    }

    div.footer {
        position: running(footer);
    }
    .footer {
        position: running(footer);
    }

    .bottom {
        vertical-align: bottom;
    }

    /*判断是否结束*/
    .page-break {
        page-break-after: always;
    }

    /*下一页*/
    .next-page {
        page-break-before: always
    }

</style>
<body>

<!--标题-->
<h1 class="title" style="font-family: FZXiaoBiaoSong-B05S, serif" th:text="${data.title}"></h1>
<br/>
<!--文号-->
<p class="fileNo" style="font-family: FangSong_GB2312, serif" th:text="${data.fileNo}"></p>
<hr style="background-color: red;border-style: none;height: 2px;"/>
<!--页脚一定要放在正文的前面,先加载页脚在加载正文不然页脚无法显示(TMD大坑)-->
<div class='footer' th:include="footer::footer"></div>
<!--正文-->
<div style="font-family: FangSong_GB2312, serif" th:utext="${data.content}"></div>

<div style="position: relative;bottom: 0;" >
    <hr>
    <table style="width: 100%;">
        <tr>
            <td style="text-align: left;font-family: 'NSimSun', serif;" th:text="${data.signDept}"></td>
            <td style="text-align: right;font-family: 'NSimSun', serif;" th:text="${#dates.format(#dates.createNow(), 'yyyy年MM月dd日')} + '印发'"></td>
        </tr>
    </table>
    <hr>
</div>
<!--        <div th:utext="${data.name}"></div>-->

</body>

</html>

生成效果:

 

希望能对看到这篇文章的你有帮助!

------------------------------------------------------------------------------

后续问题请看评论区,本人有回复部分问题的解决方案!

后续问题二:

在加载css文件的时,测试和生产可能不互通,因此css文件可能无法加载成功导致接口超时!

解决方案:

<link rel="stylesheet" type="text/css" th:href="@{http://localhost/data/files/edit_css1.css}"/>

 解决测试和生产环境的css文件加载切换的问题!

前提条件在测试和生产端必须有该文件,否则还是会加载失败

  • 21
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 7
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值