This article was first published in October 2018 and was updated in August 2024.
PDF documents are the preferred format for many things, including sharing information with formatting and enhanced data visualization. Therefore, it’s not surprising that the ability to open and annotate PDF documents has become an increasingly demanded feature for any web application as soon as it grows in size and complexity.
However, adding such a feature to a web application usually means incrementing the number of “moving parts” of a codebase by orders of magnitude: PDF is a complex file format, which may deem the task overwhelming for any development team.
You can simplify this task significantly by making use of PSPDFKit for Web, a JavaScript PDF library that can be used with (and without) any JavaScript framework, including Vue.js. It supports all modern mobile and desktop browsers (Chrome, Firefox, Safari, and Edge) and multiple languages, and it makes good use of the latest technologies available — like WebAssembly — to make the experience as performant as possible.
PSPDFKit for Web comes in two flavors: server-backed and standalone. This means you can set it up as a shared collaboration tool that’s integrated with your server backend, or as a client-side library with all the features you may need for your PDF document handling.
To allow developers to easily embed our PDF library in their applications, there are several integration examples available. In this article, you’ll learn how to integrate our Vue.js PDF library, which you can clone from the public PSPDFKit repository. You’ll build a small app in a single HTML file that will fetch all the assets needed to load and run PSPDFKit for Web in a Vue.js app.
The final result will look like the image below in your browser. It’ll consist of a simple UI that allows you to open, view, and annotate PDF documents from within your Vue.js app.
Opening a PDF in Vue.js
To get the example running, you need the following tools:
-
Latest version of Node.js
-
PSPDFKit for Web (start your free trial)
1. Setting Up Vue.js CLI
-
Install the Vue.js CLI
npm install -g @vue/cli
-
Create a new Vue.js project
vue create my-app
Select Vue 3, depending on your preference.
-
Navigate to the project directory
cd my-app
2. Installing PSPDFKit for Web
-
Add the PSPDFKit dependency
yarn add pspdfkit
# or
npm install pspdfkit
-
Prepare the PSPDFKit library
-
Create a directory under public:
mkdir -p public/js
-
Copy the PSPDFKit library:
cp -R ./node_modules/pspdfkit/dist/pspdfkit-lib public/js/pspdfkit-lib
-
Configure the MIME types
Ensure your server has the Content-Type: application/wasm
MIME type set. This is required for PSPDFKit.
3. Creating the PSPDFKit Vue Component
-
Create the component file
-
In the
src/components
directory, create a file namedPSPDFKitContainer.vue
.
-
Set up the props
Start by defining the pdfFile
prop in your component. This prop is required and will hold the path to the PDF file you want to display. By passing this as a prop, you can dynamically load different PDF files into the viewer from outside the component:
export default { props: { pdfFile: { type: String, required: true, }, }, };
-
Create the
loadPSPDFKit
method
Next, define the loadPSPDFKit
method. This method is responsible for loading the PSPDFKit viewer with the PDF file provided through the pdfFile
prop. Since PSPDFKit.load()
returns a Promise
, define the method as async.
Make sure to unload any existing PSPDFKit instance before loading a new one. This prevents memory leaks and ensures the correct PDF is always displayed:
methods: { async loadPSPDFKit() { PSPDFKit.unload('.pdf-container'); return PSPDFKit.load({ document: this.pdfFile, container: '.pdf-container', }); }, },
-
Clean up on component destruction
To avoid lingering references when the component is destroyed, use the beforeDestroy
lifecycle hook to unload PSPDFKit. This ensures the viewer instance is properly cleaned up:
beforeDestroy() {
PSPDFKit.unload('.pdf-container');
},
-
Load PSPDFKit when the component is mounted
To make sure PSPDFKit is loaded as soon as the component is mounted, call the loadPSPDFKit
method within the mounted lifecycle hook. Once the method resolves with the instance of PSPDFKit, emit a loaded event with the instance as the payload. This event enables you to use the instance elsewhere in your application:
mounted() { this.loadPSPDFKit().then((instance) => { this.$emit('loaded', instance); }); },
-
Watch for changes to the
pdfFile
prop
To handle dynamic changes to the pdfFile
prop, use a watch handler. This allows the component to reload the PDF whenever the prop changes. Additionally, include a check to ensure the new value is valid before attempting to reload PSPDFKit:
watch: { pdfFile(val) { if (val) this.loadPSPDFKit(); }, },
Full Code Overview
Here’s the final code that combines all these steps:
<template> <div class="pdf-container"></div> </template> <script> import PSPDFKit from 'pspdfkit'; export default { props: { pdfFile: { type: String, required: true, }, }, methods: { async loadPSPDFKit() { PSPDFKit.unload('.pdf-container'); return PSPDFKit.load({ document: this.pdfFile, container: '.pdf-container', }); }, }, beforeUnmount() { PSPDFKit.unload('.pdf-container'); }, mounted() { this.loadPSPDFKit().then((instance) => { this.$emit('loaded', instance); }); }, watch: { pdfFile(val) { if (val) this.loadPSPDFKit(); }, }, }; </script> <style scoped> .pdf-container { height: 100vh; } </style>
This component is now fully capable of loading PSPDFKit, rendering PDFs, unloading when destroyed, and dynamically switching PDFs based on the provided pdfFile
prop.
Use the scoped
flag to make sure this CSS rule only applies to DOM nodes within the component itself. You can still overwrite CSS rules from outside, but it’s good practice to prevent accidental overrides across your project. This way, the component can also be freely used between various projects without having to reapply styles in each of them.
Importing and Using the PSPDFKitContainer.vue Component
Now that you’ve created the PSPDFKitContainer.vue
component, the next step is to import and use it within the main Vue.js page, where you want to display the PDF.
-
Set up the template
In your main Vue component (e.g. App.vue
), add the template structure that includes a file input for selecting the PDF file and the PSPDFKitContainer
component to display the selected file:
<template> <div id="app"> <input type="file" @change="openDocument" /> <PSPDFKitContainer :pdfFile="pdfFile" /> </div> </template>
-
The
<input>
element is used to select the PDF file. -
The
PSPDFKitContainer
component displays the PDF, withpdfFile
as a prop containing the file path.
-
Import the PSPDFKitContainer component
In the script section, import the PSPDFKitContainer
component so that it can be used in the template:
<script> import PSPDFKitContainer from '@/components/PSPDFKitContainer'; export default { data() { return { pdfFile: this.pdfFile || '/example.pdf', // Default PDF file to display. }; }, components: { PSPDFKitContainer, }, methods: { openDocument() { if (this.pdfFile) { window.URL.revokeObjectURL(this.pdfFile); // Clean up old file reference. } // Update `pdfFile` with the new file selected by the user. this.pdfFile = window.URL.createObjectURL(event.target.files[0]); }, }, }; </script>
-
The
pdfFile
data property stores the path to the currently selected PDF. -
The
openDocument
method handles the file input change, generating a URL for the selected file and assigning it topdfFile
.
-
Prepare your PDF document
Add the example.pdf
file to the public
directory of your Vue.js project. The public directory serves static files directly at the root of your project, making it easy to access resources like images, styles, and PDF files.
-
Run the application
With everything set up, run your Vue application:
npm run serve
When you load a PDF, the PSPDFKit viewer will display the document, allowing you to read, annotate, and print it directly from your browser.
Dynamic Loading and Event Handling
While the setup above works well for displaying a PDF, it’s also essential to manage the PSPDFKit instance when switching between different PDF files. To handle this, watch for changes in the pdfFile
prop and reload the viewer when a new file is selected. You also need to handle the instance returned by PSPDFKit.load()
for advanced functionality like annotations.
-
Update the template to handle loaded events
Modify the template to listen for the loaded
event emitted by the PSPDFKitContainer
component:
<template> <div id="app"> <input type="file" @change="openDocument" /> <PSPDFKitContainer :pdfFile="pdfFile" @loaded="handleLoaded" /> </div> </template>
The @loaded="handleLoaded"
directive listens for when the PSPDFKit instance is ready.
-
Handle the loaded event
In the script, add a method to handle the loaded event, which provides the PSPDFKit instance for further interaction:
<script> import PSPDFKitContainer from '@/components/PSPDFKitContainer'; export default { data() { return { pdfFile: this.pdfFile || '/example.pdf', }; }, components: { PSPDFKitContainer, }, methods: { openDocument(event) { if (this.pdfFile) { window.URL.revokeObjectURL(this.pdfFile); } this.pdfFile = window.URL.createObjectURL(event.target.files[0]); }, handleLoaded(instance) { console.log("PSPDFKit has loaded: ", instance); // Perform any operations with the PSPDFKit instance, like adding annotations. }, }, }; </script>
Annotating the PDF in Vue.js
Here’s a step-by-step guide to adding custom annotations to your PDF in a Vue.js app using PSPDFKit.
Step 1 — Set Up the Component Data
First, add a data()
function to your component to store the PSPDFKit instance:
export default { data() { return { instance: null, // This will hold the PSPDFKit instance }; }, // ... };
Step 2 — Update the loadPSPDFKit Method
Customize the toolbar to include a custom button that adds an ink annotation. Modify the loadPSPDFKit
method to include the custom toolbar button, and set up the onPress
handler:
methods: { async loadPSPDFKit() { try { PSPDFKit.unload('.pdf-container'); const instance = await PSPDFKit.load({ document: this.pdfFile, container: '.pdf-container', toolbarItems: [ { type: 'custom', title: 'Add Ink Annotation', className: 'addInkAnnotation', name: 'addInkAnnotation', onPress: async () => { if (!instance) { console.error('PSPDFKit instance is not available'); return; } const inkAnnotation = new PSPDFKit.Annotations.InkAnnotation({ pageIndex: 0, lines: PSPDFKit.Immutable.List([ PSPDFKit.Immutable.List([ new PSPDFKit.Geometry.DrawingPoint({ x: 0, y: 0 }), new PSPDFKit.Geometry.DrawingPoint({ x: 100, y: 100 }), ]), ]), }); try { const createdAnnotations = await instance.create(inkAnnotation); console.log('Created Ink Annotations:', createdAnnotations); } catch (error) { console.error('Error creating ink annotation:', error); } }, }, ], }); this.instance = instance; this.$emit('loaded', instance); } catch (error) { console.error('Error loading PSPDFKit:', error); } }, }
Step 3 — Handle the instance in the Component Lifecycle
Update the mounted
lifecycle hook and the watch
property to ensure the instance is updated when needed:
mounted() { this.loadPSPDFKit().then((instance) => { this.instance = instance; // Store the PSPDFKit instance. this.$emit('loaded', instance); // Emit an event when loaded. }); }, watch: { pdfFile(val) { if (val) { this.loadPSPDFKit().then((instance) => { this.instance = instance; // Update the instance when the PDF changes. }); } }, },
Rebuild your Vue.js app and load a PDF. The toolbar will now include a button labeled Add Ink Annotation. When clicked, it adds an ink annotation with predefined drawing points to the first page of your PDF. Adjust the drawing points and other parameters as necessary to fit your specific requirements.
This post focuses on adding an ink annotation, but PSPDFKit supports various annotation types, like text and images. Find out more about how to use PSPDFKit.Instance#create()
by browsing the PSPDFKit for Web API reference.
Conclusion
In this blog, you learned how to integrate PSPDFKit’s JavaScript PDF library with the Vue.js framework. Once you’ve got it up and running, you can enable additional features and customizations in your application:
-
And much more
At PSPDFKit, we offer a commercial, feature-rich, and completely customizable Vue.js PDF library that’s easy to integrate and comes with well-documented APIs to handle advanced use cases. Try it for free, or visit our web demo to see it in action.