Editing Existing PDF Files in Java

Editing Existing PDF Files in Java

1. Overview

In this article, we'll see how to edit the content of an existing PDF file in Java. First, we'll just add new content. Then, we'll focus on removing or replacing some pre-existing content.

2. Adding the iText7 Dependency

We'll use the iText7 library to add content to the PDF file. Later on, we'll use the pdfSweep add-on to remove or replace content.

Note that iText is licensed under AGPL, which might limit the distribution of a commercial application: iText License Model.

First, let's add these dependencies to our pom.xml:

xml 复制代码
<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>itext7-core</artifactId>
    <version>7.2.3</version>
    <type>pom</type>
</dependency>
<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>cleanup</artifactId>
    <version>3.0.1</version>
</dependency>

3. File Handling

Let's understand the steps for handling our PDF with iText7:

  • First, we open a PdfReader to read the content of the source file. This throws an IOException if an error occurs at any time while reading the file.
  • Then, we open a PdfWriter to the destination file. If this file doesn't exist or can't be created, a FileNotFoundException is thrown.
  • After that, we'll open a PdfDocument which uses our PdfReader and PdfWriter.
  • Finally, closing the PdfDocument closes both the underlying PdfReader and PdfWriter.

Let's write a main() method that runs our whole treatment. For the sake of simplicity, we'll just rethrow any Exception that could occur:

java 复制代码
public static void main(String[] args) throws IOException {
    PdfReader reader = new PdfReader("src/main/resources/baeldung.pdf");
    PdfWriter writer = new PdfWriter("src/main/resources/baeldung-modified.pdf");
    PdfDocument pdfDocument = new PdfDocument(reader, writer);
    addContentToDocument(pdfDocument);
    pdfDocument.close();
}

In the following section, we'll complete step-by-step the addContentToDocument() method in order to fill our PDF with new content. The source document's a PDF file that only contains the text "Hello Baeldung*"* on the top left. The destination file will be created by the program.

4. Adding Content to the File

We'll now add various types of content to the file.

4.1. Adding a Form

We'll start by adding a form to the file. Our form will be very simple and contain a unique field called name.

Furthermore, we need to tell iText where to place the field. In this case, we'll put it at the following point: (35,400) . The coordinates (0,0) refer to the bottom left of the document. Lastly, we'll set the dimension of the field to 100×30:

java 复制代码
PdfFormField personal = PdfFormField.createEmptyField(pdfDocument);
personal.setFieldName("information");
PdfTextFormField name = PdfFormField.createText(pdfDocument, new Rectangle(35, 400, 100, 30), "name", "");
personal.addKid(name);
PdfAcroForm.getAcroForm(pdfDocument, true)
    .addField(personal, pdfDocument.getFirstPage());

Additionally, we've explicitly specified iText to add the form to the first page of the document.

4.2. Adding a New Page

Let's now have a look at how we can add a new page to the document. We'll use the addNewPage() method.

This method can accept the index of the added page if we want to specify it. For instance, we can add a new page at the beginning of the document:

java 复制代码
pdfDocument.addNewPage(1);

4.3. Adding an Annotation

We'll now want to add an annotation to the document. Concretely, an annotation looks like a squared comic bubble.

We'll add it on top of the form that's now located on the second page of the document. Consequently, we'll place it on the coordinates (40,435). Additionally, we'll give it a simple name and content. These will only show up when hovering over the annotation:

java 复制代码
PdfAnnotation ann = new PdfTextAnnotation(new Rectangle(40, 435, 0, 0)).setTitle(new PdfString("name"))
    .setContents("Your name");
pdfDocument.getPage(2)
    .addAnnotation(ann);

Here's how the middle of our second page now looks:

4.4. Adding an Image

From now on, we'll add layout elements to the page. In order to do this, we won't be able to manipulate the PdfDocument directly anymore. We'll rather create a Document from it and work with that. Moreover, we'll need to close the Document in the end. Closing a Document automatically closes the base PdfDocument. So we could remove the part where we closed the PdfDocument earlier:

java 复制代码
Document document = new Document(pdfDocument);
// add layout elements
document.close();

Now, to add the image, we'll need to load it from its location. We'll do this using the create() method of the ImageDataFactory class. This throws a MalformedURLException if the passed file URL can't be parsed. In this example, we'll use an image of Baeldung's logo placed in the resources directory:

ImageData imageData = ImageDataFactory.create("src/main/resources/baeldung.png");

The next step will be to set the image's properties in the file. We'll set its size to 550×100 . We'll put it on the first page of our PDF, at the (10,50) coordinates. Let's see the code to add the image:

java 复制代码
Image image = new Image(imageData).scaleAbsolute(550,100)
    .setFixedPosition(1, 10, 50);
document.add(image);

The image is automatically rescaled to the given size. So here's how it looks in the document:

4.5. Adding a Paragraph

The iText library brings some tools to add text to the file. The font can be parameterized on the pieces themselves, or directly on the Paragraph element.

For instance, let's add the following sentence on top of the first page: This is a demo from Baeldung tutorials . We'll set the font size of the beginning of this sentence to 16 and the global font size of Paragraph to 8:

java 复制代码
Text title = new Text("This is a demo").setFontSize(16);
Text author = new Text("Baeldung tutorials.");
Paragraph p = new Paragraph().setFontSize(8)
    .add(title)
    .add(" from ")
    .add(author);
document.add(p);

4.6. Adding a Table

Last but not least, we can also add a table to the file. For example, we'll define a double-entry table with two cells and two headers on top of them. We won't specify any position. So it'll be naturally added on top of the document, right after the Paragraph we just added:

java 复制代码
Table table = new Table(UnitValue.createPercentArray(2));
table.addHeaderCell("#");
table.addHeaderCell("company");
table.addCell("name");
table.addCell("baeldung");
document.add(table);

Let's see the beginning of the first page of the document now:

java 复制代码
package org.example.pdf;

import com.itextpdf.forms.PdfAcroForm;
import com.itextpdf.forms.fields.PdfFormField;
import com.itextpdf.forms.fields.PdfTextFormField;
import com.itextpdf.io.image.ImageData;
import com.itextpdf.io.image.ImageDataFactory;
import com.itextpdf.kernel.geom.Rectangle;
import com.itextpdf.kernel.pdf.PdfDocument;
import com.itextpdf.kernel.pdf.PdfReader;
import com.itextpdf.kernel.pdf.PdfString;
import com.itextpdf.kernel.pdf.PdfWriter;
import com.itextpdf.kernel.pdf.annot.PdfAnnotation;
import com.itextpdf.kernel.pdf.annot.PdfTextAnnotation;
import com.itextpdf.layout.Document;
import com.itextpdf.layout.element.Image;
import com.itextpdf.layout.element.Paragraph;
import com.itextpdf.layout.element.Table;
import com.itextpdf.layout.element.Text;
import com.itextpdf.layout.properties.UnitValue;

import java.io.File;
import java.io.IOException;
import java.net.MalformedURLException;

/**
 * @author: guanglai.zhou
 * @date: 2023/12/13 13:38
 */
public class EditPdfMain {

    public static void main(String[] args) throws IOException {
        File inFile = getFileFromResources("baeldung.pdf");
        PdfReader reader = new PdfReader(inFile);
        File outFile = getFileFromResources("baeldung-modified.pdf");
        PdfWriter writer = new PdfWriter(outFile);
        PdfDocument pdfDocument = new PdfDocument(reader, writer);
        addContentToDocument(pdfDocument);
        pdfDocument.close();
        System.out.println(inFile.getPath());
        System.out.println(outFile.getPath());
    }

    /**
     * 直接读取 src/main/resources目录下的文件
     */
    public static File getFileFromResources(String fileName) {
        String path = OpenPdfMain.class.getResource("/" + fileName).getPath();
        return new File(path);
    }

    private static void addContentToDocument(PdfDocument pdfDocument) throws MalformedURLException {
        // 4.1. add form 添加一个表格到PDF文档当中
        PdfFormField personal = PdfFormField.createEmptyField(pdfDocument);
        personal.setFieldName("information");
        /**
         * 坐标0,0指的是左边底部
         */
        PdfTextFormField name = PdfFormField.createText(pdfDocument,
                new Rectangle(35, 400, 100, 30), "name", "");
        personal.addKid(name);
        /**
         * 明确指定将表格添加到第一页
         */
        PdfAcroForm.getAcroForm(pdfDocument, true)
                .addField(personal, pdfDocument.getFirstPage());

        // 4.2. add new page 通过index指定添加到文档的开头
        pdfDocument.addNewPage(1);

        // 4.3. add annotation 在第一步新增表格的上面添加注释
        PdfAnnotation ann = new PdfTextAnnotation(new Rectangle(40, 435, 0, 0))
                .setTitle(new PdfString("name"))
                .setContents("Your name");
        pdfDocument.getPage(2).addAnnotation(ann);

        /**
         * From now on, we'll add layout elements to the page.
         * In order to do this, we won't be able to manipulate the *PdfDocument* directly anymore.
         * We'll rather create a *Document* from it and work with that.
         * Moreover, we'll need to close the *Document* in the end.
         * **Closing a Document automatically closes the base PdfDocument.**
         * So we could remove the part where we closed the *PdfDocument* earlier:
         */
        // create document form pdf document
        Document document = new Document(pdfDocument);

        // 4.4. add an image 设置图片大小为 550*100
        final File file = getFileFromResources("baeldung.png");
        ImageData imageData = ImageDataFactory.create(file.toURI().toURL());
        Image image = new Image(imageData).scaleAbsolute(550, 100)
                // 存放在第一页 坐标为 10*50
                .setFixedPosition(1, 10, 50);
        document.add(image);

        // 4.5. add a paragraph 添加文件片段
        Text title = new Text("This is a demo").setFontSize(16);
        Text author = new Text("Baeldung tutorials.");
        Paragraph p = new Paragraph().setFontSize(8)
                .add(title)
                .add(" from ")
                .add(author);
        document.add(p);

        // 4.6. add a table 添加表格
        Table table = new Table(UnitValue.createPercentArray(2));
        table.addHeaderCell("#");
        table.addHeaderCell("company");
        table.addCell("name");
        table.addCell("baeldung");
        document.add(table);

        // close the document
        // this automatically closes the pdfDocument, which then closes automatically the pdfReader and pdfWriter
        document.close();
    }

}

5. Removing Content From the File

Let's now see how we can remove content from the PDF file. To keep things simple, we'll write another main() method.

Our source PDF file will be the baeldung-modified.pdf file and the destination will be a new baeldung-cleaned.pdf file. We'll work directly on the PdfDocument object. From now on, we'll use iText's pdfSweep add-on.

5.1. Removing Text From the File

To remove a given text from the file, we'll need to define a cleanup strategy. In this example, the strategy will simply be to find all text matching Baeldung . The last step is to call the autoSweepCleanUp() static method of PdfCleaner . This method will create a custom PdfCleanUpTool which will throw an IOException if any error happens during file handling:

java 复制代码
CompositeCleanupStrategy strategy = new CompositeCleanupStrategy();
strategy.add(new RegexBasedCleanupStrategy("Baeldung"));
PdfCleaner.autoSweepCleanUp(pdfDocument, strategy);

As we can see, the occurrences of the Baeldung word in the source file are overlayed with a black rectangle in the result file. This behavior is suitable, for instance, for data anonymization:

5.2. Removing Other Content From the File

Unfortunately, it's very difficult to detect any non-text content in the file. However, pdfSweep offers the possibility to erase the content of a portion of the file. Thus, if we know where the content we want to remove is located, we'll be able to take advantage of this possibility.

As an example, we'll erase the content of the rectangle of size 100×35 located at (35,400) on the second page. This means we'll get rid of all the content of the form and the annotation. Furthermore, we'll erase the rectangle of size 90×70 located at (10,50) of the first page. This basically removes the B from Baeldung's logo. Using the PdfCleanUpTool class, here's the code to do all that:

java 复制代码
List<PdfCleanUpLocation> cleanUpLocations = Arrays.asList(new PdfCleanUpLocation(1, new Rectangle(10, 50, 90,70)), new PdfCleanUpLocation(2, new Rectangle(35, 400, 100, 35)));
PdfCleanUpTool cleaner = new PdfCleanUpTool(pdfDocument, cleanUpLocations, new CleanUpProperties());
cleaner.cleanUp();

We can now see the following image in baeldung-cleaned.pdf:

6. Replacing Content in the File

In this section, we'll do the same work as earlier, except that we'll replace the former text with a new text instead of only erasing it.

For more clarity, we'll use a new main() method again. Our source file will be the baeldung-modified.pdf file. Our destination file will be a new baeldung-fixed.pdf file.

Earlier we saw that the removed text was overlayed with a black background. However, this color is configurable. As we know the background of the text is white in our file, we'll force the overlay to be white. The beginning of the treatment will be similar to what we did earlier, except that we'll search for the text Baeldung tutorials.

However, after calling autoSweepCleanUp(), we'll query the strategy to get the location of the removed code. We'll then instantiate a PdfCanvas which will contain the replacement text HIDDEN. Additionally, we'll remove the top margin to have it a bit better aligned with the original text. The default alignment is indeed not so good. Let's look at the resulting code:

java 复制代码
CompositeCleanupStrategy strategy = new CompositeCleanupStrategy();
strategy.add(new RegexBasedCleanupStrategy("Baeldung").setRedactionColor(ColorConstants.WHITE));
PdfCleaner.autoSweepCleanUp(pdfDocument, strategy);

for (IPdfTextLocation location : strategy.getResultantLocations()) {
    PdfPage page = pdfDocument.getPage(location.getPageNumber() + 1);
    PdfCanvas pdfCanvas = new PdfCanvas(page.newContentStreamAfter(), page.getResources(), page.getDocument());
    Canvas canvas = new Canvas(pdfCanvas, location.getRectangle());
    canvas.add(new Paragraph("HIDDEN").setFontSize(8).setMarginTop(0f));
}

And we can have a look at the file:

7. Conclusion

In this tutorial, we've seen how to edit the content of a PDF file. We've seen that we can add new content, remove existing content, and even replace text in the original file with a new one.

As always, the code for this article can be found over on GitHub.

相关推荐
飞翔沫沫情12 分钟前
《快速部署Mysql-slave 容器,实现高效主从同步》
数据库·docker·mysql主从同步
SelectDB技术团队14 分钟前
一文了解多云原生的现代化实时数仓 SelectDB Cloud
大数据·数据库·数据仓库·云原生·云计算
hvinsion16 分钟前
Python PDF批量加密工具
android·python·pdf
爱补鱼的猫猫16 分钟前
混合合并两个pdf文件
pdf
Q_192849990628 分钟前
基于Spring Boot的工商局商家管理系统
java·spring boot·后端
m0_7482326439 分钟前
[MySQL报错]关于发生net start mysql 服务无法启动,服务没有报告任何错误的五种解决方案。
java
小学鸡!1 小时前
idea报错:There is not enough memory to perform the requested operation.
java·intellij-idea
木卫二号Coding1 小时前
docker-开源nocodb,使用已有数据库
数据库·docker·开源
StarRocks_labs1 小时前
StarRocks 存算分离在得物的降本增效实践
数据库·数据仓库·湖仓
L.S.V.2 小时前
Java 溯本求源之基础(三十)——封装,继承与多态
java·开发语言