pdfmake
Client/server side PDF printing in pure JavaScript
Check out the playground or soon... read the docs
Features
- line-wrapping,
- text-alignments (left, right, centered, justified),
- numbered and bulleted lists,
- tables and columns
- auto/fixed/star-sized widths,
- col-spans and row-spans,
- headers automatically repeated in case of a page-break,
- images and vector graphics,
- convenient styling and style inheritance,
- page headers and footers:
- static or dynamic content,
- access to current page number and page count,
- page dimensions and orientations,
- margins,
- custom page breaks,
- font embedding,
- support for complex, multi-level (nested) structures,
- helper methods for opening/printing/downloading the generated PDF.
Getting Started
warning! links to the documentation do NOT work yet
This document will walk you through the basics of pdfmake and will show you how to create PDF files in the browser. If you're interested in server-side printing, read getting started with pdfMake under NodeJS.
To begin with the default configuration, you should include two files:
<!doctype html>
<html lang='en'>
<head>
<meta charset='utf-8'>
<title>my first pdfmake example</title>
<script src='build/pdfmake.min.js'></script>
<script src='build/vfs_fonts.js'></script>
</head>
<body>
...
You can get both files using bower:
bower install pdfmake
or copy them directly from the build directory from the repository.
Document-definition-object
pdfmake follows a declarative approach. It basically means, you'll never have to calculate positions manually or use commands like: writeText(text, x, y)
, moveDown
etc..., as you would with a lot of other libraries.
The most fundamental concept to be mastered is the document-definition-object which can be as simple as:
var docDefinition = { content: 'This is an sample PDF printed with pdfMake' };
or become pretty complex (having multi-level tables, images, lists, paragraphs, margins, styles etc...).
As soon as you have the document-definition-object, you're ready to create and open/print/download the PDF:
pdfMake.createPdf(docDefinition).open();
pdfMake.createPdf(docDefinition).download();
Styling
pdfmake makes it possible to style any paragraph or its part:
var docDefinition = {
content: [
'This is a standard paragraph, using default style',
{ text: 'This paragraph will have a bigger font', fontSize: 15 },
{
text: [
'This paragraph is defined as an array of elements to make it possible to ',
{ text: 'restyle part of it and make it bigger ', fontSize: 15 },
'than the rest.'
]
}
]
};
Style dictionaries
It's also possible to define a dictionary of reusable styles:
var docDefinition = {
content: [
{ text: 'This is a header', style: 'header' },
'No styling here, this is a standard paragraph',
{ text: 'Another text', style: 'anotherStyle' },
{ text: 'Multiple styles applied', style: [ 'header', 'anotherStyle' ] }
],
styles: {
header: {
fontSize: 22,
bold: true
},
anotherStyle: {
italic: true,
alignment: 'right'
}
}
};
To have a deeper understanding of styling in pdfmake, style inheritance and local-style-overrides check this example and the resulting PDF or open it in playground.
Columns
By default paragraphs are rendered as a vertical stack of elements (one below another). It is possible however to divide available space into columns.
var docDefinition = {
content: [
'This paragraph fills full width, as there are no columns. Next paragraph however consists of three columns',
{
columns: [
{
width: 'auto',
text: 'First column'
},
{
width: '*',
text: 'Second column'
},
{
width: 100,
text: 'Third column'
}
],
columnGap: 10
},
'This paragraph goes below all columns and has full width'
]
};
Column content is not limited to a simple text. It can actually contain any valid pdfmake element. See a complete example and the resulting pdf or open it in playground
Tables
Conceptually tables are similar to columns. They can however have headers, borders and cells spanning over multiple columns/rows.
var docDefinition = {
content: [
{
table: {
headerRows: 1,
widths: [ '*', 'auto', 100, '*' ],
body: [
[ 'First', 'Second', 'Third', 'The last one' ],
[ 'Value 1', 'Value 2', 'Value 3', 'Value 4' ],
[ { text: 'Bold value', bold: true }, 'Val 2', 'Val 3', 'Val 4' ]
]
}
}
]
};
All concepts related to tables are covered by this example and the resulting PDF. You can also open it in playground.
Lists
pdfMake supports both numbered and bulleted lists:
var docDefinition = {
content: [
'Bulleted list example:',
{
ul: [
'Item 1',
'Item 2',
'Item 3',
{ text: 'Item 4', bold: true },
]
},
'Numbered list example:',
{
ol: [
'Item 1',
'Item 2',
'Item 3'
]
}
]
};
Headers and footers
Page headers and footers in pdfmake can be: static or dynamic.
They use the same syntax:
var docDefinition = {
header: 'simple text',
footer: {
columns: [
'Left part',
{ text: 'Right part', alignment: 'right' }
]
},
content: (...)
};
For dynamically generated content (including page numbers and page count) you can pass a function to the header or footer:
var docDefinition = {
footer: function(currentPage, pageCount) { return currentPage.toString() + ' of ' + pageCount; },
header: function(currentPage, pageCount) {
return { text: 'simple text', alignment: (currentPage % 2) ? 'left' : 'right' };
},
(...)
};
Margins
Any element in pdfMake can have a margin:
(...)
{ text: 'sample', margin: [ 5, 2, 10, 20 ] },
{ text: 'another text', margin: [5, 2] },
{ text: 'last one', margin: 5 }
(...)
Stack of paragraphs
You could have figured out by now (from the examples), that if you set the content
key to an array, the document becomes a stack of paragraphs.
You'll quite often reuse this structure in a nested element, like in the following example:
var docDefinition = {
content: [
'paragraph 1',
'paragraph 2',
{
columns: [
'first column is a simple text',
[
'paragraph A',
'paragraph B',
'these paragraphs will be rendered one below another inside the column'
]
]
}
]
};
The problem with an array is that you cannot add styling properties to it (to change fontSize for example).
The good news is - array is just a shortcut in pdfMake for { stack: [] }, so if you want to restyle the whole stack, you can do it using the expanded definition:
var docDefinition = {
content: [
'paragraph 1',
'paragraph 2',
{
columns: [
'first column is a simple text',
{
stack: [
'paragraph A',
'paragraph B',
'these paragraphs will be rendered one below another inside the column'
],
fontSize: 15
}
]
}
]
};
Page dimensions, orientation and margins
var docDefinition = {
pageSize: 'A5',
pageOrientation: 'landscape',
pageMargins: [ 40, 60, 40, 60 ],
};
If you set pageSize
to a string, you can use one of the following values:
- '4A0', '2A0', 'A0', 'A1', 'A2', 'A3', 'A4', 'A5', 'A6', 'A7', 'A8', 'A9', 'A10',
- 'B0', 'B1', 'B2', 'B3', 'B4', 'B5', 'B6', 'B7', 'B8', 'B9', 'B10',
- 'C0', 'C1', 'C2', 'C3', 'C4', 'C5', 'C6', 'C7', 'C8', 'C9', 'C10',
- 'RA0', 'RA1', 'RA2', 'RA3', 'RA4',
- 'SRA0', 'SRA1', 'SRA2', 'SRA3', 'SRA4',
- 'EXECUTIVE', 'FOLIO', 'LEGAL', 'LETTER', 'TABLOID'
Does the above description cover everything?
Not at all.
If you really want to learn pdfMake, go ahead and check the:
Coming soon
Hmmm... let me know what you need ;)
The goal is quite simple - make pdfmake useful for a looooooooot of people and help building responsive HTML5 apps with printing support.
There's one thing on the roadmap for v2 (no deadline however) - make the library hackable, so you can write plugins to:
- extend document-definition-model (with things like { chart: ... }),
- add syntax translators (like the provided [ ... ] -> { stack: [ ... ] }
- build custom DSLs on top of document-definition-model (this is actually possible at the moment).
License
MIT
pdfmake is based on a truly amazing library pdfkit.org - credits to @devongovett