Pdfkit Guide: by Devon Govett
Pdfkit Guide: by Devon Govett
Pdfkit Guide: by Devon Govett
By Devon Govett
Version 0.9.0
Getting Started with PDFKit
Installation
Installation uses the npm package manager. Just type the following command after
installing npm.
Creating a document
Creating a PDFKit document is quite simple. Just require the pdfkit module in your
JavaScript source file and create an instance of the PDFDocument class.
PDFDocument instances are readable Node streams. They don't get saved anywhere
automatically, but you can call the pipe method to send the output of the PDF document to
another writable Node stream as it is being written. When you're done with your document,
call the end method to finalize it. Here is an example showing how to pipe to a file or an
HTTP response.
The write and output methods found in PDFKit before version 0.5 are now deprecated.
Using PDFKit in the browser
As of version 0.6, PDFKit can be used in the browser as well as in Node! There are two ways to
use PDFKit in the browser. The first is to use Browserify, which is a Node module packager
for the browser with the familiar require syntax. The second is to use a prebuilt version of
PDFKit, which you can download from Github.
Using PDFKit in the browser is exactly the same as using it in Node, except you'll want to
pipe the output to a destination supported in the browser, such as a Blob. Blobs can be used
to generate a URL to allow display of generated PDFs directly in the browser via an iframe, or
they can be used to upload the PDF to a server, or trigger a download in the user's browser.
To get a Blob from a PDFDocument, you should pipe it to a blob-stream, which is a module
that generates a Blob from any Node-style stream. The following example uses Browserify
to load PDFKit and blob-stream, but if you're not using Browserify, you can load them in
whatever way you'd like (e.g. script tags).
// require dependencies
const PDFDocument = require('pdfkit');
const blobStream = require('blob-stream');
Note that in order to Browserify a project using PDFKit, you need to install the brfs module
with npm, which is used to load built-in font data into the package. It is listed as a
devDependency in PDFKit's package.json, so it isn't installed by default for Node users. If
you forget to install it, Browserify will print an error message.
Adding pages
The first page of a PDFKit document is added for you automatically when you create the
document unless you provide autoFirstPage: false. Subsequent pages must be added by
you. Luckily, it is quite simple!
doc.addPage()
To add some content every time a page is created, either by calling addPage() or
automatically, you can use the pageAdded event.
You can also set some options for the page, such as its size and orientation.
The layout property can be either portrait (the default) or landscape. The size property
can be either an array specifying [width, height] in PDF points (72 per inch), or a string
specifying a predefined size. A list of the predefined paper sizes can be seen here. The default
is letter.
Passing a page options object to the PDFDocument constructor will set the default paper size
and layout for every page in the document, which is then overridden by individual options
passed to the addPage method.
You can set the page margins in two ways. The first is by setting the margin property
(singular) to a number, which applies that margin to all edges. The other way is to set the
margins property (plural) to an object with top, bottom, left, and right values. The default
is a 1 inch (72 point) margin on all sides.
For example:
PDFKit has a bufferPages option in versions v0.7.0 and later that allows you to control when
pages are flushed to the output file yourself rather than letting PDFKit handle that for you.
To use it, just pass bufferPages: true as an option to the PDFDocument constructor. Then,
you can call doc.switchToPage(pageNumber) to switch to a previous page (page numbers
start at 0).
When you're ready to flush the buffered pages to the output file, call flushPages. This
method is automatically called by doc.end(), so if you just want to buffer all pages in the
document, you never need to call it. Finally, there is a bufferedPageRange method, which
returns the range of pages that are currently buffered. Here is a small example that shows
how you might add page numbers to a document.
for (i = range.start, end = range.start + range.count, range.start <= end; i < end; i+
+;) {
doc.switchToPage(i);
doc.text(`Page ${i + 1} of ${range.count}`);
}
Here is a list of all of the properties you can add to the document metadata. According to the
PDF spec, each property must have its first letter capitalized.
To enable encryption, provide a user password when creating the PDFDocument in options
object. The PDF file will be encrypted when a user password is provided, and users will be
prompted to enter the password to decrypt the file when opening it.
To set access privileges for the PDF file, you need to provide an owner password and
permission settings in the option object when creating PDFDocument. By default, all
operations are disallowed. You need to explicitly allow certain operations.
You can specify either user password, owner password or both passwords. Behavior differs
according to passwords you provides:
When only user password is provided, users with user password are able to decrypt the file
and have full access to the document.
When only owner password is provided, users are able to decrypt and open the document
without providing any password, but the access is limited to those operations explicitly
permitted. Users with owner password have full access to the document.
When both passwords are provided, users with user password are able to decrypt the file but
only have limited access to the file according to permission settings. Users with owner
password have full access to the document.
Note that PDF file itself cannot enforce access privileges. When file is decrypted, PDF viewer
applications have full access to the file content, and it is up to viewer applications to respect
permission settings.
To choose encryption method, you need to specify PDF version. PDFKit will choose best
encryption method available in the PDF version you specified.
When using PDF version 1.7 ExtensionLevel 3, password is truncated to 127 bytes of its UTF-8
representation. In older versions, password is truncated to 32 bytes, and only Latin-1
characters are allowed.
Adding content
Once you've created a PDFDocument instance, you can add content to the document. Check
out the other sections described in this document to learn about each type of content you can
add.
That's the basics! Now let's move on to PDFKit's powerful vector graphics abilities.
Vector Graphics in PDFKit
An introduction to vector graphics
Unlike images which are defined by pixels, vector graphics are defined through a series of
drawing commands. This makes vector graphics scalable to any size without a reduction in
quality (pixelization). The PDF format was designed with vector graphics in mind, so creating
vector drawings is very easy. The PDFKit vector graphics APIs are very similar to that of the
HTML5 canvas element, so if you are familiar at all with that API, you will find PDFKit easy to
pick up.
One thing to notice about this example is the use of method chaining. All methods in PDFKit
are chainable, meaning that you can call one method right after the other without
referencing the doc variable again. Of course, this is an option, so if you don't like how the
code looks when chained, you don't have to write it that way.
SVG paths
PDFKit includes an SVG path parser, so you can include paths written in the SVG path syntax
in your PDF documents. This makes it simple to include vector graphics elements produced
in many popular editors such as Inkscape or Adobe Illustrator. The previous example could
also be written using the SVG path syntax like this.
The PDFKit SVG parser supports all of the command types supported by SVG, so any valid
SVG path you throw at it should work as expected.
Shape helpers
PDFKit also includes some helpers that make defining common shapes much easier. Here is
a list of the helpers.
The last one, polygon, allows you to pass in a list of points (arrays of x,y pairs), and it will
create the shape by moving to the first point, and then drawing lines to each consecutive
point. Here is how you'd draw a triangle with the polygon helper.
In order to make our drawings interesting, we really need to give them some style. PDFKit
has many methods designed to do just that.
lineWidth
lineCap
lineJoin
miterLimit
dash
fillColor
strokeColor
opacity
fillOpacity
strokeOpacity
Some of these are pretty self explanatory, but let's go through a few of them.
Line cap and line join
The lineCap and lineJoin properties accept constants describing what they should do. This
is best illustrated by an example.
doc.lineCap('round')
.moveTo(150, 20)
.lineTo(200, 20)
.stroke();
// square line cap shown with a circle instead of a line so you can see it
doc.lineCap('square')
.moveTo(250, 20)
.circle(275, 30, 15)
.stroke();
doc.lineJoin('round')
.rect(150, 100, 50, 50)
.stroke();
doc.lineJoin('bevel')
.rect(250, 100, 50, 50)
.stroke();
The space option defines the length of the space between each dash, and the phase option
defines the starting point of the sequence of dashes. By default the space attribute is equal
to the length and the phase attribute is set to 0. You can use the undash method to make the
line solid again.
The following example draws a circle with a dashed line where the space between the dashes
is double the length of each dash.
The fillColor and strokeColor methods accept an optional second argument as a shortcut
for setting the fillOpacity and strokeOpacity. Finally, the opacity method is a
convenience method that sets both the fill and stroke opacity to the same value.
The fill and stroke methods also accept a color as an argument so that you don't have to
call fillColor or strokeColor beforehand. The fillAndStroke method accepts both fill
and stroke colors as arguments.
There are two types of gradients: linear and radial. They are created by the linearGradient
and radialGradient methods. Their function signatures are listed below:
linearGradient(x1, y1, x2, y2) - x1,y1 is the start point, x2,y2 is the end point
radialGradient(x1, y1, r1, x2, y2, r2) - r1 is the inner radius, r2 is the outer radius
Once you have a gradient object, you need to create color stops at points along that gradient.
Stops are defined at percentage values (0 to 1), and take a color value (any usable by the
fillColor method), and an optional opacity.
You can see both linear and radial gradients in the following example:
// Initial setup
doc.fillColor('red')
.translate(-100, -50)
.scale(0.8);
You'll notice that I used the scale and translate transformations in this example. We'll
cover those in a minute. The output of this example, with some added labels, is below.
Saving and restoring the graphics stack
Once you start producing more complex vector drawings, you will want to be able to save and
restore the state of the graphics context. The graphics state is basically a snapshot of all the
styles and transformations (see below) that have been applied, and many states can be
created and stored on a stack. Every time the save method is called, the current graphics
state is pushed onto the stack, and when you call restore, the last state on the stack is
applied to the context again. This way, you can save the state, change some styles, and then
restore it to how it was before you made those changes.
Transformations
Transformations allow you to modify the look of a drawing without modifying the drawing
itself. There are three types of transformations available, as well as a method for setting the
transformation matrix yourself. They are translate, rotate and scale.
The translate transformation takes two arguments, x and y, and effectively moves the
origin of the page which is (0, 0) by default, to the left and right x and y units.
The rotate transformation takes an angle and optionally, an object with an origin property.
It rotates the document angle degrees around the passed origin or by default, around the
origin (top left corner) of the page.
The scale transformation takes a scale factor and an optional origin passed in an options
hash as with the rotate transformation. It is used to increase or decrease the size of the
units in the drawing, or change its size. For example, applying a scale of 0.5 would make the
drawing appear at half size, and a scale of 2 would make it appear twice as large.
If you are feeling particularly smart, you can modify the transformation matrix yourself
using the transform method.
We used the scale and translate transformations above, so here is an example of using the
rotate transformation. We'll set the origin of the rotation to the center of the rectangle.
If you want to "unclip", you can use the save method before the clipping, and then use
restore to retrieve access to the whole page.
That's it for vector graphics in PDFKit. Now let's move on to learning about PDFKit's text
support!
Text in PDFKit
The basics
PDFKit makes adding text to documents quite simple, and includes many options to
customize the display of the output. Adding text to a document is as simple as calling the
text method.
doc.text('Hello world!')
Internally, PDFKit keeps track of the current X and Y position of text as it is added to the
document. This way, subsequent calls to the text method will automatically appear as new
lines below the previous line. However, you can modify the position of text by passing X and
Y coordinates to the text method after the text itself.
If you want to move down or up by lines, just call the moveDown or moveUp method with the
number of lines you'd like to move (1 by default).
Line wrapping and justification
PDFKit includes support for line wrapping out of the box! If no options are given, text is
automatically wrapped within the page margins and placed in the document flow below any
previous text, or at the top of the page. PDFKit automatically inserts new pages as necessary
so you don't have to worry about doing that for long pieces of text. PDFKit can also
automatically wrap text into multiple columns.
The text will automatically wrap unless you set the lineBreak option to false. By default it
will wrap to the page margin, but the width option allows you to set a different width the
text should be wrapped to. If you set the height option, the text will be clipped to the
number of lines that can fit in that height.
When line wrapping is enabled, you can choose a text justification. There are four options:
left (the default), center, right, and justify. They work just as they do in your favorite
word processor, but here is an example showing their use in a text box.
const lorem = 'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Etiam in
suscipit purus. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices
posuere cubilia Curae; Vivamus nec hendrerit felis. Morbi aliquam facilisis risus eu
lacinia. Sed eu leo in turpis fringilla hendrerit. Ut nec accumsan nisl.';
doc.fontSize(8);
doc.text(`This text is left aligned. ${lorem}`, {
width: 410,
align: 'left'
}
);
doc.moveDown();
doc.text(`This text is centered. ${lorem}`, {
width: 410,
align: 'center'
}
);
doc.moveDown();
doc.text(`This text is right aligned. ${lorem}`, {
width: 410,
align: 'right'
}
);
doc.moveDown();
doc.text(`This text is justified. ${lorem}`, {
width: 410,
align: 'justify'
}
);
This text is left aligned. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Etiam in suscipit purus. Vestibulum
ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; Vivamus nec hendrerit felis. Morbi
aliquam facilisis risus eu lacinia. Sed eu leo in turpis fringilla hendrerit. Ut nec accumsan nisl.
This text is centered. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Etiam in suscipit purus. Vestibulum
ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; Vivamus nec hendrerit felis. Morbi
aliquam facilisis risus eu lacinia. Sed eu leo in turpis fringilla hendrerit. Ut nec accumsan nisl.
Text styling
PDFKit has many options for controlling the look of text added to PDF documents, which can
be passed to the text method. They are enumerated below.
Additionally, the fill and stroke color and opacity methods described in the vector graphics
section are applied to text content as well.
Here is an example combining some of the options above, wrapping a piece of text into three
columns, in a specified width and height.
const lorem = 'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Etiam in
suscipit purus. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices
posuere cubilia Curae; Vivamus nec hendrerit felis. Morbi aliquam facilisis risus eu
lacinia. Sed eu leo in turpis fringilla hendrerit. Ut nec accumsan nisl. Suspendisse
rhoncus nisl posuere tortor tempus et dapibus elit porta. Cras leo neque, elementum a
rhoncus ut, vestibulum non nibh. Phasellus pretium justo turpis. Etiam vulputate,
odio vitae tincidunt ultricies, eros odio dapibus nisi, ut tincidunt lacus arcu eu
elit. Aenean velit erat, vehicula eget lacinia ut, dignissim non tellus. Aliquam nec
lacus mi, sed vestibulum nunc. Suspendisse potenti. Curabitur vitae sem turpis.
Vestibulum sed neque eget dolor dapibus porttitor at sit amet sem. Fusce a turpis
lorem. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere
cubilia Curae;';
doc.text(lorem, {
columns: 3,
columnGap: 15,
height: 100,
width: 465,
align: 'justify'
});
Lorem ipsum dolor sit amet, Suspendisse rhoncus nisl posuere dignissim non tellus. Aliquam nec
consectetur adipiscing elit. Etiam in tortor tempus et dapibus elit porta. lacus mi, sed vestibulum nunc.
suscipit purus. Vestibulum ante Cras leo neque, elementum a Suspendisse potenti. Curabitur
ipsum primis in faucibus orci luctus rhoncus ut, vestibulum non nibh. vitae sem turpis. Vestibulum sed
et ultrices posuere cubilia Curae; Phasellus pretium justo turpis. neque eget dolor dapibus porttitor
Vivamus nec hendrerit felis. Morbi Etiam vulputate, odio vitae tincidunt at sit amet sem. Fusce a turpis
aliquam facilisis risus eu lacinia. ultricies, eros odio dapibus nisi, ut lorem. Vestibulum ante ipsum
Sed eu leo in turpis fringilla tincidunt lacus arcu eu elit. Aenean primis in faucibus orci luctus et
hendrerit. Ut nec accumsan nisl. velit erat, vehicula eget lacinia ut, ultrices posuere cubilia Curae;
Text measurements
If you're working with documents that require precise layout, you may need to know the size
of a piece of text. PDFKit has two methods to achieve this: widthOfString(text, options)
and heightOfString(text, options). Both methods use the same options described in the
Text styling section, and take into account the eventual line wrapping.
Lists
The list method creates a bulleted list. It accepts as arguments an array of strings, and the
optional x, y position. You can create complex multilevel lists by using nested arrays. Lists
use the following additional options:
bulletRadius
textIndent
bulletIndent
Rich Text
As mentioned above, PDFKit supports a simple form of rich text via the continued option.
When set to true, PDFKit will retain the text wrapping state between text calls. This way,
when you call text again after changing the text styles, the wrapping will continue right
where it left off.
The options given to the first text call are also retained for subsequent calls after a
continued one, but of course you can override them. In the following example, the width
option from the first text call is retained by the second call.
doc.fillColor('green')
.text(lorem.slice(0, 500), {
width: 465,
continued: true
}).fillColor('red')
.text(lorem.slice(500));
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Etiam in suscipit purus. Vestibulum ante ipsum primis in
faucibus orci luctus et ultrices posuere cubilia Curae; Vivamus nec hendrerit felis. Morbi aliquam facilisis risus eu
lacinia. Sed eu leo in turpis fringilla hendrerit. Ut nec accumsan nisl. Suspendisse rhoncus nisl posuere tortor
tempus et dapibus elit porta. Cras leo neque, elementum a rhoncus ut, vestibulum non nibh. Phasellus pretium justo
turpis. Etiam vulputate, odio vitae tincidunt ultricies, eros odio dapibus nisi, ut tincidunt lacus arcu eu elit. Aenean
velit erat, vehicula eget lacinia ut, dignissim non tellus. Aliquam nec lacus mi, sed vestibulum nunc. Suspendisse
potenti. Curabitur vitae sem turpis. Vestibulum sed neque eget dolor dapibus porttitor at sit amet sem. Fusce a turpis
lorem. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae;
Fonts
The PDF format defines 14 standard fonts that can be used in PDF documents. PDFKit
supports each of them out of the box. Besides Symbol and Zapf Dingbats this includes 4
styles (regular, bold, italic/oblique, bold+italic) of Helvetica, Courier, and Times. To switch
between standard fonts, call the font method with the corresponding Label:
'Courier'
'Courier-Bold'
'Courier-Oblique'
'Courier-BoldOblique'
'Helvetica'
'Helvetica-Bold'
'Helvetica-Oblique'
'Helvetica-BoldOblique'
'Symbol'
'Times-Roman'
'Times-Bold'
'Times-Italic'
'Times-BoldItalic'
'ZapfDingbats'
The PDF format also allows fonts to be embedded right in the document. PDFKit supports
embedding TrueType (.ttf), OpenType (.otf), WOFF, WOFF2, TrueType Collection (.ttc),
and Datafork TrueType (.dfont) fonts.
To change the font used to render text, just call the font method. If you are using a standard
PDF font, just pass the name to the font method. Otherwise, pass the path to the font file, or
a Buffer containing the font data. If the font is a collection font (.ttc and .dfont files),
meaning that it contains multiple styles in the same file, you should pass the name of the
style to be extracted from the collection.
// Register a font
doc.registerFont('Heading Font', 'fonts/Chalkboard.ttc', 'Chalkboard-Bold');
That's about all there is too it for text in PDFKit. Let's move on now to images.
Images in PDFKit
Adding images to PDFKit documents is an easy task. Just pass an image path, buffer, or data
uri with base64 encoded data to the image method along with some optional arguments.
PDFKit supports the JPEG and PNG formats. If an X and Y position are not provided, the
image is rendered at the current point in the text flow (below the last line of text). Otherwise,
it is positioned absolutely at the specified point. The image will be scaled according to the
following options.
When a fit or cover array is provided, PDFKit accepts these additional options:
align - horizontally align the image, the possible values are 'left', 'center' and 'right'
valign - vertically align the image, the possible values are 'top', 'center' and 'bottom'
// Fit the image in the dimensions, and center it both horizontally and vertically
doc.image('images/test.jpeg', 430, 15, {fit: [100, 100], align: 'center', valign:
'center'})
.rect(430, 15, 100, 100).stroke()
.text('Centered', 430, 0);
This example produces the following output:
Stretch
Scale
That is all there is to adding images to your PDF documents with PDFKit. Now let's look at
adding outlines.
Outlines in PDFKit
Outlines are the heirachical bookmarks that display in some PDF readers. Currently only
page bookmarks are supported, but more may be added in the future. They are simple to add
and only require a single method:
addItem(title, options)
// Add a sub-section
top.addItem('Sub-section');
Options
The options parameter currently only has one property: expanded. If this value is set to true
then all of that section's children will be visible by default. This value defaults to false.
In this example the 'Top Level' section will be expanded to show 'Sub-section'.
// Add a sub-section
top.addItem('Sub-section');
Annotations in PDFKit
Annotations are interactive features of the PDF format, and they make it possible to include
things like links and attached notes, or to highlight, underline or strikeout portions of text.
Annotations are added using the various helper methods, and each type of annotation is
defined by a rectangle and some other properties. Here is a list of the available annotation
methods:
Many of the annotations have a color option that you can specify. You can use an array of
RGB values, a hex color, or a named CSS color value for that option.
If you are adding an annotation to a piece of text, such as a link or underline, you will need to
know the width and height of the text in order to create the required rectangle for the
annotation. There are two methods that you can use to do that. To get the width of any piece
of text in the current font, just call the widthOfString method with the string you want to
measure. To get the line height in the current font, just call the currentLineHeight method.
You must remember that annotations have a stacking order. If you are putting more than one
annotation on a single area and one of those annotations is a link, make sure that the link is
the last one you add, otherwise it will be covered by another annotation and the user won't
be able to click it.
Here is an example that uses a few of the annotation types.
This is a link!
STRIKE!
Annotations are currently not the easiest things to add to PDF documents, but that is the
fault of the PDF spec itself. Calculating a rectangle manually isn't fun, but PDFKit makes it
easier for a few common annotations applied to text, including links, underlines, and
strikes. Here's an example showing two of them:
doc.fontSize(20)
.fillColor('red')
.text('Another link!', 20, 0, {
link: 'http://apple.com/',
underline: true
}
);
The output is as you'd expect:
Another link!
You made it!
That's all there is to creating PDF documents in PDFKit. It's really quite simple to create
beautiful multi-page printable documents using Node.js!
This guide was generated from Markdown files using a PDFKit generation script. The
examples are actually run to generate the output shown inline. The script generates both the
website and the PDF guide, and can be found on Github. Check it out if you want to see an
example of a slightly more complicated renderer using a parser for Markdown and a syntax
highlighter.
If you have any questions about what you've learned in this guide, please don't hesitate to
ask the author or post an issue on Github. Enjoy!