Java实现pdf中动态插入图片

今天接到一个需求,需要在pdf中的签名处,插入签名照片,但签名位置不固定,话不多说上代码:

1、首先引入itextpdf依赖包:

        <dependency><groupId>com.itextpdf</groupId><artifactId>itextpdf</artifactId><version>5.5.13</version></dependency>

2、具体实现逻辑如下:

	/*** description PDF插入图片(根据关键字位置动态插入)** @author yanzy* @version 1.0* @date 2025/3/27 17:26*/public static void imgToPdf2(String pdfPath, String imagePath, String outputPath, String keyword) {try {// 读取原始PDFPdfReader reader = new PdfReader(pdfPath);FileOutputStream fos = new FileOutputStream(outputPath);PdfStamper stamper = new PdfStamper(reader, fos);// 遍历PDF每一页,查找关键字并插入签名图片Boolean keyBool = true;for (int i = 1; i <= reader.getNumberOfPages(); i++) {// 获取当前页面的文本和坐标信息List<Position> positions = getPositions(reader, i);// 合并连续文本块StringBuilder combinedText = new StringBuilder();List<Position> textChunks = new ArrayList<>();for (Position pos : positions) {combinedText.append(pos.getText());textChunks.add(pos);}// 查找关键字在整个文本中的位置String fullText = combinedText.toString();int index = fullText.indexOf(keyword);if (index != -1) {// 获取关键字起始位置Position endPos = findPositionByIndex(textChunks, index + keyword.length() - 1);// 计算插入坐标(取结束位置右侧)float imgX = endPos.getX() + 20; // 横纵标,右侧偏移20单位float imgY = endPos.getY() - 20; // 纵坐标,向下移动20单位log.info("关键字位置 -> X:{}  Y: {} | 图片位置 -> X: {}  Y:{}", endPos.getX(), endPos.getY(), imgX, imgY);// 添加图片Image img = Image.getInstance(imagePath);// 图片尺寸img.scaleToFit(100, 50);// 设置图片的插入位置(X,Y坐标)img.setAbsolutePosition(imgX, imgY);// PdfStamper:直接修改原始PDF,避免图层顺序问题PdfContentByte canvas = stamper.getOverContent(i);canvas.addImage(img);keyBool = false;}}if (keyBool) {throw new Exception("未找到关键字位置!");}stamper.close();reader.close();log.info("图片插入成功!");} catch (Exception e) {e.printStackTrace();log.error("pdf插入图片失败! errMsg: {}", e.getMessage());}}/*** description 根据文本索引找到对应位置** @author yanzy* @version 1.0* @date 2025/3/27 17:26*/private static Position findPositionByIndex(List<Position> chunks, int targetIndex) {int currentIndex = 0;for (Position chunk : chunks) {int chunkLength = chunk.getText().length();if (currentIndex + chunkLength > targetIndex) {return chunk;}currentIndex += chunkLength;}return chunks.get(chunks.size() - 1);}/*** description 获取PDF页面的所有文本位置** @author yanzy* @version 1.0* @date 2025/3/27 17:26*/private static List<Position> getPositions(PdfReader reader, int pageNumber) throws Exception {List<Position> positions = new ArrayList<>();// 创建PdfContentByte和RenderListener来提取文本PdfReaderContentParser parser = new PdfReaderContentParser(reader);parser.processContent(pageNumber, new RenderListener() {@Overridepublic void beginTextBlock() {}@Overridepublic void endTextBlock() {}@Overridepublic void renderText(TextRenderInfo renderInfo) {String text = renderInfo.getText();float x = renderInfo.getBaseline().getStartPoint().get(0);float y = renderInfo.getBaseline().getStartPoint().get(1);positions.add(new Position(x, y, text));}@Overridepublic void renderImage(ImageRenderInfo renderInfo) {}});return positions;}public static void main(String[] args) {// 输入PDF路径String srcPdf = "E://test.pdf";// 输出PDF路径String destPdf = "E://output_image.pdf";// 图片路径String imagePath = "E://test.jpg";// 关键字String keyword = "本人签名:";//imgToPdf(srcPdf, imagePath, destPdf);imgToPdf2(srcPdf, imagePath, destPdf, keyword);}
/*** description pdf中的文本坐标** @author yanzy* @date 2025/3/28 15:36*/
public class Position {private final float x;private final float y;private final String text;public Position(float x, float y, String text) {this.x = x;this.y = y;this.text = text;}public float getX() {return x;}public float getY() {return y;}public String getText() {return text;}}

3、下面是pdf的内容:
在这里插入图片描述
4、运行程序后的结果:
在这里插入图片描述
5、可以看到签名照片已经成功加上了

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/75878.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OpenCV 图形API(2)为什么需要图形API?

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 G-API背后的动机 G-API模块为OpenCV带来了基于图的执行模型。本章简要描述了这种新模型如何在两个方面帮助软件开发者&#xff1a;优化和移植图像处理算法…

基于Spring AI开发本地Jenkins MCP Server服务

前言 首先介绍下MCP是什么&#xff1f; MCP是由开发了 Claude 模型的 Anthropic 公司2024年12月提出并开源的一项开放标准&#xff0c;全称&#xff1a;Model Context Protocol&#xff0c;它是一个开放协议&#xff0c;它使 LLM 应用与外部数据源和工具之间的无缝集成成为可能…

vcpkg安装指定版本的库

一.vcpkg安装 使用git将vcpkg源码克隆到本地制定目录&#xff08;D:\vcpkg&#xff09;&#xff0c;并初始化 git clone https://github.com/microsoft/vcpkg.git cd vcpkg ./bootstrap-vcpkg.sh # Linux/macOS .\bootstrap-vcpkg.bat # Windows 如下图&#xff1a; 二.安…

数据结构C语言练习(单双链表)

本篇练习题(单链表)&#xff1a; 1.力扣 203. 移除链表元素 2.力扣 206. 反转链表 3.力扣 876. 链表的中间结点 4.力扣 21. 合并两个有序链表 5. 牛客 链表分割算法详解 6.牛客 链表回文结构判断 7. 力扣 160. 相交链表 8. 力扣 141 环形链表 9. 力扣 142 环形链表 II…

nginx部署前端项目(linux、docker)

引言 在CentOS 7系统上使用docker安装nginx&#xff0c;使用nginx部署一个由Vue开发、打包的项目 docker安装nginx 这里不多赘述&#xff0c;直接上docker-compose.yml代码 nginx:container_name: nginximage: nginx:1.27.2ports:- "80:80"volumes:- /docker/ngin…

WPF ContentPresenter详解2

ContentPresenter与ContentControl的区别 ContentControl 和 ContentPresenter 是 WPF 中两个相关的控件&#xff0c;但它们在用途和功能上有一些关键的区别。理解这两者的区别和联系有助于更好地设计和开发用户界面。 1. 类层次结构 ContentControl&#xff1a;位于 WPF 控件…

【HTML5游戏开发教程】零基础入门合成大西瓜游戏实战 | JS物理引擎+Canvas动画+完整源码详解

《从咖啡杯到财务自由&#xff1a;一个程序员的合成之旅——当代码遇上物理引擎的匠心之作》 &#x1f31f; 这是小游戏开发系列的第四篇送福利文章&#xff0c;感谢一路以来支持和关注这个项目的每一位朋友&#xff01; &#x1f4a1; 文章力求严谨&#xff0c;但难免有疏漏之…

鸿蒙OS 5.0 服务能力框架深入剖析

鸿蒙OS 5.0 服务能力框架中关键类的作用分析 1\. 鸿蒙OS 5.0 服务能力框架导论 鸿蒙OS 5.0&#xff0c;亦称鸿蒙智联 5 1&#xff0c;标志着华为在分布式操作系统领域迈出的重要一步。与早期版本采用兼容安卓的AOSP层、Linux内核以及LiteOS内核不同&#xff0c;鸿蒙OS 5.0 专注…

RTMP推流+EasyDSS云服务+边缘AI分析的无人机监控系统设计

在现代科技不断发展的背景下&#xff0c;无人机技术已经广泛应用于各个领域&#xff0c;从航拍摄影到工业巡检&#xff0c;从农业监测到应急救援&#xff0c;无人机以其高效的工作能力&#xff0c;为人们的生活和工作带来了诸多便利与创新&#xff0c;而其视频传输与分析系统更…

HCIP(VLAN综合实验)

实验拓补图 实验分析 一、实验目的 掌握VLAN的创建和配置方法理解VLAN在局域网中的作用学习如何通过VLAN实现网络隔离和通信 二、实验环境 交换机&#xff08;SW1、SW2、SW3&#xff09;个人电脑&#xff08;PC1、PC2、PC3、PC4、PC5、PC6&#xff09;路由器&#xff08;R1…

Linux系统编程 | 线程的基本概念

&#x1f493;个人主页&#xff1a;mooridy &#x1f493;专栏地址&#xff1a;Linux 关注我&#x1f339;&#xff0c;和我一起学习更多计算机的知识! &#x1f51d;&#x1f51d;&#x1f51d; 什么是线程 程序中的一个执行路线就叫做线程 一个进程至少要有一个执行线程,单…

小林coding-12道Spring面试题

1.说一下你对 Spring 的理解?spring的核心思想说说你的理解&#xff1f; 2.Spring IoC和AOP 介绍一下?Spring的aop介绍一下?IOC和AOP是通过什么机制来实现的?怎么理解SpringIoc&#xff1f;依赖倒置&#xff0c;依赖注入&#xff0c;控制反转分别是什么&#xff1f;依赖注…

第十二章——位运算

按位的与& 若x的第i位和y的第i位都是1&#xff0c;那么&#xff08;x&y&#xff09;1&#xff0c;否则&#xff08;x&y&#xff09; 0 应用&#xff1a;希望让某一位或某些位为0 。取一个数中的一段。 按位的或| 若x的第i位1或y的第i位1&#xff0c;那么&…

计算机等级考试数据库三级(笔记3)

插入 修改 现要创建一个具有如下功能的触发器&#xff1a;每当在销售表中插入一条销售记录时&#xff0c;修改商品表中对应商品的销售总量&#xff0c;假设一次只插入一条销售记录。请补全下列代码。CREATE TRIGGER tri insert on xx FOR xx AS xx 商品表 xx 销售总量xx (SELEC…

【Leetcode 每日一题】2716. 最小化字符串长度

问题背景 给你一个下标从 0 0 0 开始的字符串 s s s&#xff0c;重复执行下述操作 任意 次&#xff1a; 在字符串中选出一个下标 i i i&#xff0c;并使 c c c 为字符串下标 i i i 处的字符。并在 i i i 左侧&#xff08;如果有&#xff09;和 右侧&#xff08;如果有&…

Flutter中实现拍照识题的功能

文章目录 **1. 功能拆解****2. 具体实现步骤****(1) 拍照或选择图片****(2) 图片预处理&#xff08;可选&#xff09;****(3) 文字识别&#xff08;OCR&#xff09;****(4) 数学公式识别 → LaTeX****方案1&#xff1a;Mathpix API&#xff08;高精度&#xff0c;付费&#xff…

【Mysql:内置函数】

日期函数&#xff1a; 查看当前日期&#xff1a; select current_date();查看当前时间&#xff1a; select current_time(); 查看当前时间戳&#xff1a; select current_timestamp(); 计算两个日期的差值&#xff1a; select datediff(date1,date2); 当前的日期时间&a…

71. 我的第一个Linux驱动实验

一、字符设备驱动框架 字符设备驱动的编写主要就是驱动对应的open、close、read。。。其实就是 file_operations结构体的成员变量的实现。 其中关于 C 库以及如何通过系统调用“陷入” 到内核空间这个我们不用去管&#xff0c;我们重点关注的是应用程序和具体的驱动&#xff0…

jdk21使用Vosk实现语音文字转换,免费的语音识别

1.下载vosk的model vosk官网&#xff1a;https://alphacephei.com/vosk/models 我这里使用较小的vosk-model-small-cn-0.22 2.添加相关pom文件 <!-- 获取音频信息 --><dependency><groupId>org</groupId><artifactId>jaudiotagger</artifac…

如何一键安装所有Python项目的依赖!

在开发项目时&#xff0c;常常需要在多个环境中安装各种依赖。对开发者来说&#xff0c;每次手动一个个安装这些依赖是不是很麻烦&#xff1f;&#x1f605; 其实有个超简单的办法&#xff01;只需要一个脚本&#xff0c;就能快速解决问题&#xff01;&#x1f4a1; 这就是我们…