有在线添加水印的工具,比如 下面这个
image.png
JAVA 端可以使用 itextpdf 来实现

demo 程序尝试

首先添加全家桶依赖

  1. // ~ pdf ==========================
  2. implementation 'com.itextpdf:itext7-core:7.1.16'

编写

  1. package cn.mrcode.tools.pdf_watermark;
  2. import com.itextpdf.io.font.PdfEncodings;
  3. import com.itextpdf.io.font.constants.StandardFonts;
  4. import com.itextpdf.io.image.ImageData;
  5. import com.itextpdf.io.image.ImageDataFactory;
  6. import com.itextpdf.kernel.font.PdfFont;
  7. import com.itextpdf.kernel.font.PdfFontFactory;
  8. import com.itextpdf.kernel.geom.Rectangle;
  9. import com.itextpdf.kernel.pdf.PdfDocument;
  10. import com.itextpdf.kernel.pdf.PdfPage;
  11. import com.itextpdf.kernel.pdf.PdfReader;
  12. import com.itextpdf.kernel.pdf.PdfWriter;
  13. import com.itextpdf.kernel.pdf.canvas.PdfCanvas;
  14. import com.itextpdf.kernel.pdf.extgstate.PdfExtGState;
  15. import com.itextpdf.layout.Document;
  16. import com.itextpdf.layout.element.Paragraph;
  17. import com.itextpdf.layout.property.TextAlignment;
  18. import com.itextpdf.layout.property.VerticalAlignment;
  19. import java.io.File;
  20. /**
  21. * @author mrcode
  22. * @date 2021/10/21 20:57
  23. */
  24. public class Demo {
  25. public static final String DEST = "C:\\temp\\xx2.pdf";
  26. public static final String IMG = "C:\\temp\\b1.png";
  27. public static final String SRC = "d:\\temp\xx.pdf";
  28. public static void main(String[] args) throws Exception {
  29. new Demo().manipulatePdf(DEST);
  30. }
  31. protected void manipulatePdf(String dest) throws Exception {
  32. PdfDocument pdfDoc = new PdfDocument(new PdfReader(SRC), new PdfWriter(dest));
  33. Document doc = new Document(pdfDoc);
  34. // PdfFont font = PdfFontFactory.createFont(StandardFonts.HELVETICA);
  35. // 这个实体是宋体中文字体,只有 4, m
  36. // 字体文件源码中只支持 afm、pfm、ttf、otf、woff、woff2,等你提供不支持的格式就会报错,点进去看源码就知道了
  37. // ttc 看到源码中也支持,但是提供 ttc 就报错,暂时没深入看源码如何报错
  38. final PdfFont font = PdfFontFactory.createFont("C:\\temp\\songti.ttf",
  39. // 水平书写
  40. PdfEncodings.IDENTITY_H,
  41. // 是否将字体嵌入到目标文档中
  42. // 该参数被 PdfFontFactory.EmbeddingStrategy.PREFER_EMBEDDED 代替,true 对应 PREFER_EMBEDDED,如果可能,嵌入字体
  43. true);
  44. // 多行文本使用 \n 换行
  45. Paragraph paragraph = new Paragraph("My watermark (中国强text) \n 中国")
  46. .setFont(font)
  47. .setOpacity(0.1F) // 字体透明度 0-1 完全透明~不透明
  48. .setFontSize(30); // 字体大小
  49. ImageData img = ImageDataFactory.create(IMG);
  50. float w = img.getWidth();
  51. float h = img.getHeight();
  52. // 用于添加图片水印,设置图片水印透明度
  53. PdfExtGState gs1 = new PdfExtGState().setFillOpacity(0.5f);
  54. // Implement transformation matrix usage in order to scale image
  55. for (int i = 1; i <= pdfDoc.getNumberOfPages(); i++) {
  56. PdfPage pdfPage = pdfDoc.getPage(i);
  57. // 获取页面大小,考虑页面旋转
  58. Rectangle pageSize = pdfPage.getPageSizeWithRotation();
  59. // 当页面有旋转时,内容自动旋转
  60. pdfPage.setIgnorePageRotationForContent(true);
  61. // 计算添加的位置坐标
  62. float x = (pageSize.getLeft() + pageSize.getRight()) / 2;
  63. float y = (pageSize.getTop() + pageSize.getBottom()) / 2;
  64. // 添加图片水印使用
  65. PdfCanvas over = new PdfCanvas(pdfDoc.getPage(i));
  66. over.saveState();
  67. over.setExtGState(gs1);
  68. if (i % 2 == 1) {
  69. // 添加文本水印
  70. // 参数分别为:文本、x 坐标、y 坐标、添加到第几页、文本水平对齐方式、文本垂直对齐方式、旋转弧度
  71. doc.showTextAligned(paragraph, x, y, i, TextAlignment.CENTER, VerticalAlignment.TOP, 0);
  72. } else {
  73. // 添加图片水印
  74. over.addImageWithTransformationMatrix(img, w, 0, 0, h, x - (w / 2), y - (h / 2), false);
  75. }
  76. over.restoreState();
  77. }
  78. doc.close();
  79. }
  80. }

水印生成效果:
文字水印
image.png
图片水印
image.png

宋体简体中文 ttf 字体文件(下载后将.txt 后缀去掉):songti.ttf.txt

封装成工具类

功能:

  • 支持垂直水平居中、页面平铺添加水印
  • 支持添加单行/多行文本水印:字体、字号、颜色、透明度、旋转度、平铺时:水平和垂直间隔
  • 支持添加图片水印 ```java package cn.mrcode.tools.pdf_watermark;

import com.itextpdf.io.font.PdfEncodings; import com.itextpdf.io.font.constants.StandardFonts; import com.itextpdf.io.image.ImageData; import com.itextpdf.io.image.ImageDataFactory; import com.itextpdf.kernel.colors.DeviceRgb; import com.itextpdf.kernel.font.PdfFont; import com.itextpdf.kernel.font.PdfFontFactory; import com.itextpdf.kernel.geom.Rectangle; import com.itextpdf.kernel.pdf.PdfDocument; import com.itextpdf.kernel.pdf.PdfPage; import com.itextpdf.kernel.pdf.PdfReader; import com.itextpdf.kernel.pdf.PdfWriter; import com.itextpdf.kernel.pdf.canvas.PdfCanvas; import com.itextpdf.kernel.pdf.extgstate.PdfExtGState; import com.itextpdf.layout.Document; import com.itextpdf.layout.element.Paragraph; import com.itextpdf.layout.property.TextAlignment; import com.itextpdf.layout.property.VerticalAlignment;

import java.io.IOException;

/**

  • PDF 水印添加 *
  • @author mrcode
  • @date 2021/10/22 21:27 / public class PdFWatermarkUtil { /*

    • 添加文字水印; 默认为居中添加 *
    • @param watermark
    • @param srcPath 原始 PDF 文件绝对路径
    • @param destPath 添加完水印后的 PDF 存放路径 */ public static void addWatermark(TextWatermark watermark, String srcPath, String destPath) throws IOException { PdfDocument pdfDoc = new PdfDocument(new PdfReader(srcPath), new PdfWriter(destPath)); Document doc = new Document(pdfDoc); PdfFont font = getPdfFont(watermark.getFontPath());

      // 设置文字水印样式 final String text = watermark.getText(); final int fontSize = watermark.getFontSize(); Paragraph paragraph = new Paragraph(text)

      1. .setFont(font)

      // .setFontColor(new DeviceRgb(0, 0, 0))

      1. .setOpacity(watermark.getOpacity()) // 字体透明度 0-1 完全透明~不透明
      2. .setFontSize(fontSize); // 字体大小

      final String color = watermark.getColor(); // 设置 RGB 颜色 if (color != null) {

      1. final String[] rgbs = color.split(",");
      2. final DeviceRgb deviceRgb = new DeviceRgb(
      3. Integer.parseInt(rgbs[0].trim()),
      4. Integer.parseInt(rgbs[1].trim()),
      5. Integer.parseInt(rgbs[2].trim()));
      6. paragraph.setFontColor(deviceRgb);

      }

      // 获取水印文字宽度 final float textWidth = font.getWidth(text, fontSize); // 文字高度则是字体大小 final float textHeight = fontSize; final int tileMode = watermark.getTileMode(); final int pageModeOfHorizontalInterval = watermark.getPageModeOfHorizontalInterval() == null ? 50 : watermark.getPageModeOfHorizontalInterval(); final int pageModeOfVerticalInterval = watermark.getPageModeOfVerticalInterval() == null ? (int) textHeight : watermark.getPageModeOfVerticalInterval();

      for (int i = 1; i <= pdfDoc.getNumberOfPages(); i++) {

      1. PdfPage pdfPage = pdfDoc.getPage(i);
      2. // 获取页面大小,考虑页面旋转
      3. Rectangle pageSize = pdfPage.getPageSizeWithRotation();
      4. // 当页面有旋转时,内容自动旋转
      5. pdfPage.setIgnorePageRotationForContent(true);
      6. // 水印水平垂直居中
      7. if (tileMode == 1) {
      8. // 计算添加的位置坐标:这里使用居中位置,水平垂直居中
      9. float x = (pageSize.getLeft() + pageSize.getRight()) / 2;
      10. float y = (pageSize.getTop() + pageSize.getBottom()) / 2;
      11. // 参数分别为:文本、x 坐标、y 坐标、添加到底几页、文本水平对齐方式、文本垂直对齐方式、旋转弧度
      12. doc.showTextAligned(paragraph,
      13. x, // 文本所在 x y 坐标,文字将围绕这个点进行对齐或则旋转
      14. y,
      15. i, // 添加到 PDF 第几页
      16. TextAlignment.CENTER, // 文本水平对齐方式
      17. VerticalAlignment.TOP, // 文本垂直对齐方式
      18. // 将 角度 转换为 弧度
      19. (float) Math.toRadians(watermark.getRadAngle()));
      20. } else {
      21. // 水印按照设置平铺页面
      22. // 注意这里的坐标点是 文字中心点,所以宽度需要增加文字自己的宽度,否则会重合在一起
      23. for (float posX = 0f; posX < pageSize.getWidth(); posX = posX + textWidth + pageModeOfHorizontalInterval) {
      24. for (float posY = pageModeOfVerticalInterval; posY < pageSize.getHeight(); posY = posY + pageModeOfVerticalInterval) {
      25. // 参数分别为:文本、x 坐标、y 坐标、添加到底几页、文本水平对齐方式、文本垂直对齐方式、旋转弧度
      26. doc.showTextAligned(paragraph,
      27. posX, // 文本所在 x y 坐标,文字将围绕这个点进行对齐或则旋转
      28. posY,
      29. i, // 添加到 PDF 第几页
      30. TextAlignment.CENTER, // 文本水平对齐方式
      31. VerticalAlignment.TOP, // 文本垂直对齐方式
      32. // 将 角度 转换为 弧度
      33. (float) Math.toRadians(watermark.getRadAngle()));
      34. }
      35. }
      36. }

      } doc.close(); }

      /**

    • 获取字体,支持 afm、pfm、ttf、otf、woff、woff2 字体,ttc 目前直接报错 *
    • @param fontPath 字体文件绝对路径,如果为空则返回默认的英文字体
    • @return */ private static PdfFont getPdfFont(String fontPath) throws IOException { if (fontPath == null) {

      1. return PdfFontFactory.createFont(StandardFonts.HELVETICA);

      } return PdfFontFactory.createFont(

      1. fontPath,
      2. // 水平书写
      3. PdfEncodings.IDENTITY_H,
      4. // 是否将字体嵌入到目标文档中: 如果可能,嵌入字体
      5. PdfFontFactory.EmbeddingStrategy.PREFER_EMBEDDED);

      }

      /**

    • 添加图片水印; 默认为居中添加 *
    • @param watermark
    • @param srcPath 原始 PDF 文件绝对路径
    • @param destPath 添加完水印后的 PDF 存放路径 */ public static void addWatermark(ImageWatermark watermark, String srcPath, String destPath) throws IOException { PdfDocument pdfDoc = new PdfDocument(new PdfReader(srcPath), new PdfWriter(destPath)); Document doc = new Document(pdfDoc);

      // 创建图片 ImageData img = ImageDataFactory.create(watermark.getPath()); float w = img.getWidth(); float h = img.getHeight();

      // 设置透明度 PdfExtGState gs1 = new PdfExtGState().setFillOpacity(watermark.getOpacity());

      // 循环添加到每一页的 PDF 中 for (int i = 1; i <= pdfDoc.getNumberOfPages(); i++) {

      1. PdfPage pdfPage = pdfDoc.getPage(i);
      2. // 获取页面大小,考虑页面旋转
      3. Rectangle pageSize = pdfPage.getPageSizeWithRotation();
      4. // 当页面有旋转时,内容自动旋转
      5. pdfPage.setIgnorePageRotationForContent(true);
      6. // 计算添加的位置坐标:这里使用居中位置,水平垂直居中
      7. float x = (pageSize.getLeft() + pageSize.getRight()) / 2;
      8. float y = (pageSize.getTop() + pageSize.getBottom()) / 2;
      9. // 添加图片水印使用
      10. PdfCanvas over = new PdfCanvas(pdfDoc.getPage(i));
      11. over.saveState();
      12. over.setExtGState(gs1);
      13. // 添加图片水印:位置水平垂直居中
      14. over.addImageWithTransformationMatrix(img, w, 0, 0, h, x - (w / 2), y - (h / 2), false);
      15. over.restoreState();

      } doc.close(); } }

  1. 文本和图片参数类
  2. ```java
  3. package cn.mrcode.tools.pdf_watermark;
  4. import com.itextpdf.io.font.constants.StandardFonts;
  5. import javax.validation.constraints.Max;
  6. import javax.validation.constraints.Size;
  7. import io.swagger.annotations.ApiModel;
  8. import io.swagger.annotations.ApiModelProperty;
  9. import lombok.Data;
  10. import lombok.ToString;
  11. /**
  12. * 文字水印
  13. *
  14. * @author mrcode
  15. */
  16. @Data
  17. @ToString
  18. @ApiModel
  19. public class TextWatermark {
  20. /**
  21. * 文字水印,多行可使用 \n 换行
  22. */
  23. private String text;
  24. /**
  25. * 透明度 0-1(完全透明-不透明)
  26. */
  27. private float opacity = 0.5F;
  28. /**
  29. * 颜色:只支持 RGB; 为空则默认为黑色;比如 0,0,0;
  30. * <pre>
  31. * 建议使用 rgba 提供用户选择,后面的 a 的数值用于透明度的设置,展示的颜色和水印效果类似
  32. * </pre>
  33. */
  34. private String color;
  35. /**
  36. * 旋转角度
  37. */
  38. private float radAngle = 0F;
  39. /**
  40. * 字体文件路径;如果为空,则使用标准的英文字体 StandardFonts.HELVETICA
  41. * <pre>
  42. * 支持: afm、pfm、ttf、otf、woff、woff2
  43. * </pre>
  44. *
  45. * @see StandardFonts#HELVETICA
  46. */
  47. private String fontPath;
  48. /**
  49. * 字号大小,
  50. */
  51. private int fontSize = 30;
  52. /**
  53. * 文本平铺方式: 1:文本水平垂直居中 2:页面平铺
  54. */
  55. private int tileMode = 1;
  56. /**
  57. * 页面平铺:文字水平间隔;默认为 50
  58. */
  59. private Integer pageModeOfHorizontalInterval;
  60. /**
  61. * 页面平铺:文字垂直间隔; 建议至少为字体大小(默认为字体大小),如果有旋转,则合理的高度是 (文字个数 * 文字高度)
  62. */
  63. private Integer pageModeOfVerticalInterval;
  64. }
  1. package cn.mrcode.tools.pdf_watermark;
  2. import lombok.Data;
  3. import lombok.ToString;
  4. /**
  5. * 文字水印
  6. *
  7. * @author mrcode
  8. */
  9. @Data
  10. @ToString
  11. public class ImageWatermark {
  12. /**
  13. * 图片所在绝对路径
  14. */
  15. private String path;
  16. /**
  17. * 透明度 0-1(完全透明-不透明)
  18. */
  19. private float opacity = 0.5F;
  20. }

测试用例

  1. package cn.mrcode.tools.pdf_watermark;
  2. import org.junit.jupiter.api.Test;
  3. import java.io.IOException;
  4. /**
  5. * @author mrcode
  6. */
  7. class PdFWatermarkUtilTest {
  8. public static final String DEST = "C:\\temp\\xx2.pdf";
  9. public static final String IMG = "C:\\temp\\b1.png";
  10. public static final String SRC = "d:\\temp\\xx1.pdf";
  11. public static final String FONT = "C:\\temp\\songti.ttf";
  12. @Test
  13. void addTextWatermark() throws IOException {
  14. final TextWatermark watermark = new TextWatermark();
  15. watermark.setText("中国强、则世界强");
  16. watermark.setFontPath(FONT);
  17. watermark.setRadAngle(45);
  18. watermark.setFontSize(40);
  19. watermark.setOpacity(0.1F);
  20. PdFWatermarkUtil.addWatermark(watermark, SRC, DEST);
  21. }
  22. @Test
  23. void addImageWatermark() throws IOException {
  24. final ImageWatermark watermark = new ImageWatermark();
  25. watermark.setOpacity(0.5F);
  26. watermark.setPath(IMG);
  27. PdFWatermarkUtil.addWatermark(watermark, SRC, DEST);
  28. }
  29. }

前端实现参考、生成效果参考

前端参数收集,则可以参考如下所示
image.png
文字垂直间隔可以结合 文字大小与文字个数进行建议(文字垂直间隔; 建议至少为字体大小,如果有旋转,则合理的高度简单计算是 (文字个数 * 文字高度))

水印效果参考(下图效果以上图参数作为入参):
image.png
水平居中时
image.png