Blog post

PDF Annotations with JavaScript — A Complete Overview

Hussain Arif Hussain Arif
Illustration: PDF Annotations with JavaScript — A Complete Overview

In this post, we’ll cover different types of PDF annotations and their potential use cases, and we’ll also show you how to implement PDF annotations in your apps by interacting with PSPDFKit’s annotation library via JavaScript code.

What Are PDF Annotations?

PDF annotations are objects — like text, graphics, highlights, text boxes, etc. — that you can add to a PDF document without changing the content. They’re useful for reviewing documents, collaborating with others, and implementing interactive features such as forms.

Normally, to annotate a document, you’d open a PDF in a third-party PDF editor and select an annotation tool:

Drawing on a PDF

But in place of an app, you can annotate PDF documents using JavaScript code.

This means everything is run on the client side and not on an external server. This is useful in situations where user privacy is important, since the server isn’t processing any sensitive data.

Types of PDF Annotations

The PDF specification supports many types of annotations. They come in two categories:

  • Markup annotations — These are used to mark up the content of a PDF file. They’re typically used during a review process to allow your users to collaborate, emphasize parts of the text, and take notes.

  • Non-markup annotations — These are used for non-markup purposes such as adding multimedia or fillable form fields to an existing PDF.

Next, we’ll cover a few popular types of annotations.

Text Markup

Text markup is used to annotate text in a PDF document. It can include highlighters, underlines, or strikeout annotations.

Text with highlighting, underlining, and strikeout annotations

Drawing Annotations

Drawing annotations are used to draw and write on a PDF page. You can customize the color, thickness, and transparency of the annotation tool.

Drawing on a PDF, with the toolbar showing the annotation tool options

Widget Annotations

Widgets are non-markup annotations. They appear in interactive forms, which include buttons, radio buttons, text fields, signature form fields, list boxes, and combo boxes. They can be used to create forms from flat documents.

A PDF form being filled in with typed text

Shape Annotations

These annotation types allow you to add a shape like a circle, box, or arrow annotation into a document. Cloudy annotations are commonly used in construction software to indicate a change in a plan.

Measurement Annotations

With measurement tools, users can calculate dimensions, measure distances between lines, or trace the perimeter of drawings in a PDF document. To ensure accuracy, the scale of measurement can be changed to match the scale of the document.

Text Annotations

This annotation type lets a user add text to a document. Users can do this by inserting a sticky note, adding a comment, or clicking and dragging a freeform text box.

Stamp Annotation

These annotation types allow users to quickly indicate the status of a document by — for example — adding an Approved or Declined stamp. They can also be used as a call to action in a document, where a user could add a stamp for a next step, such as Needs Signature.

Image Annotation

This type of annotation allows a user to upload and resize an image in their PDF. This annotation type can be used to quickly replace images within a document.

You can learn more about annotation types in this blog post.

Annotating in the UI vs. Programmatically

There are two ways to annotate PDF files: UI-based annotation and code-based annotation.

UI Annotation

With this method, the user selects an annotation tool in a PDF viewer and modifies a file manually.

Adding annotations in a PDF viewer

Programmatic Annotation

With programmatic annotation, a document is annotated automatically based on predefined rules written in the code. Programmatic annotation can be performed as part of an automation workflow or during runtime when a PDF is opened, or it can be manually triggered by the user.

Adding annotations programmatically

This way of handling annotations is necessary for the following situations:

  • Automation workflow — Governments and banks use automation software to stamp and insert timestamps on digital documents, which makes the process quicker and more efficient.

  • User-triggered events — If a user inputs text into a text field, the PDF runs JavaScript code to verify whether the data matches a predefined format. This ensures validation is done during runtime with zero human error.

  • Customization — Some PDF programs apply personal logos and convert text into clickable links.

Common Use Cases for PDF Annotations

There are many ways to use PDF annotations, but these are some of the more common use cases:

  • Proofreading — When writing a draft of an article, editors usually mark content to let the writer know that they need to fix a typo or an error.

  • Reviewing — A document review workflow could include approving or declining a PDF with a stamp.

  • Notetaking — This allows for collaboration or personal notes. Some examples of this include teachers adding notes to student assignments, adding notes to corporate documents during a board meeting, or construction professionals adding measurements and notes to project plans.

  • Signatures — Projects like HelloSign use a custom PDF annotator that lets professionals sign documents and contracts digitally. You can also add contract finalization to your workflow by programmatically adding e-signatures.

Adding JavaScript PDF Annotations to Your Solution

There are two primary ways of adding JavaScript PDF annotation capabilities to your solution, and this next section will outline them.

Building Your Own PDF Annotation JavaScript Solution

One of the challenges of building your own solution is that there are currently no open source annotation libraries available. In all likelihood, you’ll need to start from scratch when building annotations into your application or software.

It’s important to understand that PDF documents are incredibly complex file formats — the PDF specification has more than 1,000 pages. As such, you’ll need to have a general understanding of PDFs to build the logic of adding an annotation to a layer over a document at a precise point. This process typically involves deploying, testing, and continually refining the logic until it’s reasonably accurate.

Once you’ve developed the logic, you’ll need to design the UI and icons for the annotation function in your app. This process involves first defining your toolbar layout, and then building it to account for both small and large screens.

The last factor involves maintaining your custom annotation solution over time. We’ve heard from customers who have built their own custom annotation solution on top of PDF.js — which is an open source JavaScript library built by Mozilla — and then struggled maintaining the project. With each new release, they needed to divert developers from working on their current tasks to spending time fixing bugs.

Commercial Options for PDF Annotations with JavaScript

Since developing a document annotator can be a time-consuming endeavor, you may want to look for an out-of-the-box commercial solution. This is where PSPDFKit’s JavaScript PDF annotation library could be useful.

It offers:

  • 17 annotation types, with more to come.

  • The ability to customize annotations — for example, you can change the color, shape, and size.

  • The ability to sync annotations between devices (requires server deployment).

  • Comment threads you can enable so your user can have real-time conversations in a document (requires server deployment).

  • Tooltips, which can be designed according to your needs.

  • Support for a plethora of web frameworks. This means the client doesn’t have to write APIs for multiple programming languages.

You can view our demo to see what PSPDFKit is capable of and determine how you can use it for your project.

Integrating PSPDFKit Annotations in JavaScript

Now that you’ve learned all about annotations, in this section, you’ll use our JavaScript API to work with annotations.

Project Setup

As a first step, initialize a project with npm. You can do this with the following:

mkdir pspdfkit-tutorial
cd pspdfkit-tutorial
npm init -y       # Initialize the project.
mkdir src         # This folder will hold the main code.
cd src            # Navigate into the code folder.
touch index.html  # Client-side HTML will be served to the user.
touch index.js    # Server-side JavaScript will serve your PDF.

Next, since this app will be using the pspdfkit module, install it in your project:

npm install pspdfkit

# Get assets so you can use them in your project.
mkdir assets && cp -R ./node_modules/pspdfkit/dist/ ./src/assets/

When that’s done, navigate to your src folder. Here, add a PDF file of your choice for annotation purposes. You can use this demo document as an example.

Now, create a new file within your src folder called helperFunctions.js. As the name suggests, this file will hold utility methods needed to embed annotations into your project.

In the end, your folder structure should look like this:

Folder structure

Loading PDFs

In this section, you’ll write JavaScript instructions to serve your PDF file in the browser. Begin by entering the following code in index.js:

// src/index.js

import './assets/pspdfkit.js';

// You need to inform PSPDFKit where to look for its library assets, i.e. the location of the `pspdfkit-lib` directory.
const baseUrl = `${window.location.protocol}//${window.location.host}/assets/`;

async function renderPDF() {
	const instance = await PSPDFKit.load({
		baseUrl,
		container: '.pspdfkit', // Load the PDF into the `div` with the 'pspdfkit' class.
		document: './pspdfkit-web-demo.pdf', // Location of your document.
	});
	console.log('PSPDFKit loaded', instance);
}
await renderPDF();

Here, you created a function called renderPDF. When the program executes this method, JavaScript will call the PSPDFKit.load function. As a result, the library will display the document to the client.

All that’s left is to export this module and run it within HTML code. To do so, paste this snippet into index.html:

<!--- src/index.html --->

<html>
	<!--- Your PDF will go here --->
	<div class="pspdfkit" style="width: 100%; height: 100vh"></div>

	<!--- Load index.js file --->
	<script type="module" src="./index.js"></script>
</html>

Now it’s time to test it out! To do so, run the following command:

npx serve -l 8080 ./src # Build the `src` directory and run it on port 8080.

A loaded PDF you can annotate!

Adding Text Annotations

PSPDFKit uses the PSPDFKit.Annotations.TextAnnotation class for embedding text annotations.

To use text annotations, enter the following block of code in the helperFunctions.js file:

// src/helperFunctions.js

import './assets/pspdfkit.js';

async function createTextAnnotation(instance) {
	const annotation = new PSPDFKit.Annotations.TextAnnotation({
		pageIndex: 0, // Which page should have this annotation.
		text: {
			format: 'plain',
			value: 'Welcome to\nPSPDFKit',
		}, // Text to embed in the annotation.
		font: 'Helvetica',
		isBold: true,
		horizontalAlign: 'center', // Align your text to the center of the page.
		boundingBox: new PSPDFKit.Geometry.Rect({
			// The position of your annotation.
			left: 50,
			top: 200,
			width: 100,
			height: 80,
		}),
		fontColor: PSPDFKit.Color.BLACK, // Color of your chosen text.
	});
	const createdAnnotation = await instance.create(annotation); // In the end, attach this annotation to your PDF.
	return createdAnnotation;
}
export { createTextAnnotation }; // Export function so you can use it in your project.

When that’s done, all that’s left is to use your newly created helper function. To make this possible, insert this at the end of the renderPDF function in index.js:

// src/index.js

import { createTextAnnotation } from "./helperFunctions.js"; // Import this function.

function renderPDF() {
	// ... Extra code removed for brevity.
	await createTextAnnotation (instance); // Execute this function to embed text.
}

Adding text annotations to a PDF

Creating Ink Annotations

To insert ink annotations, you’ll use the PSPDFKit.Annotations.InkAnnotation class.

In helperFunctions.js, add the following:

// src/helperFunctions.js

async function createInkAnnotation({ instance, x1, y1, x2, y2 }) {
	// Extract the `List`, `DrawingPoint`, `Rect`, and `InkAnnotation` properties from PSPDFKit.
	// These are needed to render annotations onto the screen.
	const { List } = PSPDFKit.Immutable;
	const { DrawingPoint, Rect } = PSPDFKit.Geometry;
	const { InkAnnotation } = PSPDFKit.Annotations;
	// Initialize your ink annotation instance:
	const annotation = new InkAnnotation({
		pageIndex: 0,
		boundingBox: new Rect({ width: 400, height: 100 }), // Position of the annotation.
		strokeColor: new PSPDFKit.Color({ r: 100, b: 30, g: 255 }), // Sets the color of the stroke.
		lines: List([
			// Coordinates of the stroke.
			List([
				new DrawingPoint({ x: x1, y: y1 }), // Starting (x,y) coordinates of the line.
				new DrawingPoint({ x: x2, y: y2 }), // Final (x,y) coordinates.
			]),
		]),
	});
	const createdAnnotation = await instance.create(annotation);
	return createdAnnotation;
}
export { createTextAnnotation, createInkAnnotation }; // Now export the `createInkAnnotation` function.

As the last step, run the createInkAnnotation function in the index.js module:

// src/index.js

import { createInkAnnotation } from "./helperFunctions.js";

function renderPDF() {
	// More code:
	// Draw a line from (5,5) to (95,95).
   await createInkAnnotation({ instance, x1: 5, y1: 5, x2: 95, y2: 95 });
   // Draw another line from (95,5) to (5,95). This will draw a cross.
   await createInkAnnotation({ instance, x1: 95, y1: 5, x2: 5, y2: 95 });
}

This should be the outcome:

Adding ink annotations to a PDF

Exporting into XFDF Format

An XFDF file is an Adobe forms document that stores PDF-usable information. The data is written in XML.

You can export your modified document into XFDF via the exportXFDF method:

function renderPDF() {
	// More code.
	const XFDFData = await instance.exportXFDF(); // Retrieve the XFDF data of your document.
	console.log(XFDFData); // Log XML data into the console.
}

This will be the result:

XFDF version of the PDF

Setting the Annotations Author

You can set the annotation author’s name using the setAnnotationCreatorName method, instance.setAnnotationCreatorName("Name");.

Additional Resources

If you want to read more about annotations and what you can do with them, you can check out our guides:

Conclusion

In this post, you learned all about PDF annotations, and you saw how to use PSPDFKit’s solutions to implement programmatic annotations in JavaScript. To request a free trial of our SDK, click here. Alternatively, you can browse our demo page to see what our API is capable of.

If you’re interested, here’s the source code of this post.

FAQ

What are PDF annotations? PDF annotations are interactive elements that you can add to a PDF document, such as text comments, highlights, and drawings, without altering the original content.
Why should I use programmatic annotations? Programmatic annotations allow for automation, real-time updates, and customization, making them ideal for workflows that require efficiency and accuracy.
Can I sync annotations across devices? Yes, with the right implementation, annotations can be synchronized across devices, enabling seamless collaboration.
What types of annotations can I create? You can create various types of annotations, including text markup, drawing, widget, and stamp annotations, among others.
Is there an open source solution for PDF annotations? While there are some libraries available, using a commercial solution like PSPDFKit often provides better support and more features for complex use cases.

Explore related topics

Free trial Ready to get started?
Free trial