UNPKG

pdf-lib

Version:

Create and modify PDF files with JavaScript

788 lines 35.9 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var tslib_1 = require("tslib"); var errors_1 = require("./errors"); var PDFFont_1 = tslib_1.__importDefault(require("./PDFFont")); var PDFImage_1 = tslib_1.__importDefault(require("./PDFImage")); var PDFPage_1 = tslib_1.__importDefault(require("./PDFPage")); var sizes_1 = require("./sizes"); var core_1 = require("../core"); var utils_1 = require("../utils"); var ParseSpeeds; (function (ParseSpeeds) { ParseSpeeds[ParseSpeeds["Fastest"] = Infinity] = "Fastest"; ParseSpeeds[ParseSpeeds["Fast"] = 1500] = "Fast"; ParseSpeeds[ParseSpeeds["Medium"] = 500] = "Medium"; ParseSpeeds[ParseSpeeds["Slow"] = 100] = "Slow"; })(ParseSpeeds = exports.ParseSpeeds || (exports.ParseSpeeds = {})); /** * Represents a PDF document. */ var PDFDocument = /** @class */ (function () { function PDFDocument(context, ignoreEncryption) { var _this = this; /** The default word breaks used in PDFPage.drawText */ this.defaultWordBreaks = [' ']; this.computePages = function () { var pages = []; _this.catalog.Pages().traverse(function (node, ref) { if (node instanceof core_1.PDFPageLeaf) { var page = _this.pageMap.get(node); if (!page) { page = PDFPage_1.default.of(node, ref, _this); _this.pageMap.set(node, page); } pages.push(page); } }); return pages; }; utils_1.assertIs(context, 'context', [[core_1.PDFContext, 'PDFContext']]); utils_1.assertIs(ignoreEncryption, 'ignoreEncryption', ['boolean']); this.context = context; this.catalog = context.lookup(context.trailerInfo.Root); this.isEncrypted = !!context.lookup(context.trailerInfo.Encrypt); this.pageCache = utils_1.Cache.populatedBy(this.computePages); this.pageMap = new Map(); this.fonts = []; this.images = []; if (!ignoreEncryption && this.isEncrypted) throw new errors_1.EncryptedPDFError(); this.updateInfoDict(); } /** * Load an existing [[PDFDocument]]. The input data can be provided in * multiple formats: * * | Type | Contents | * | ------------- | ------------------------------------------------------ | * | `string` | A base64 encoded string (or data URI) containing a PDF | * | `Uint8Array` | The raw bytes of a PDF | * | `ArrayBuffer` | The raw bytes of a PDF | * * For example: * ```js * import { PDFDocument } from 'pdf-lib' * * // pdf=string * const base64 = * 'JVBERi0xLjcKJYGBgYEKCjUgMCBvYmoKPDwKL0ZpbHRlciAvRmxhdGVEZWNvZGUKL0xlbm' + * 'd0aCAxMDQKPj4Kc3RyZWFtCniccwrhMlAAwaJ0Ln2P1Jyy1JLM5ERdc0MjCwUjE4WQNC4Q' + * '6cNlCFZkqGCqYGSqEJLLZWNuYGZiZmbkYuZsZmlmZGRgZmluDCQNzc3NTM2NzdzMXMxMjQ' + * 'ztFEKyuEK0uFxDuAAOERdVCmVuZHN0cmVhbQplbmRvYmoKCjYgMCBvYmoKPDwKL0ZpbHRl' + * 'ciAvRmxhdGVEZWNvZGUKL1R5cGUgL09ialN0bQovTiA0Ci9GaXJzdCAyMAovTGVuZ3RoID' + * 'IxNQo+PgpzdHJlYW0KeJxVj9GqwjAMhu/zFHkBzTo3nCCCiiKIHPEICuJF3cKoSCu2E8/b' + * '20wPIr1p8v9/8kVhgilmGfawX2CGaVrgcAi0/bsy0lrX7IGWpvJ4iJYEN3gEmrrGBlQwGs' + * 'HHO9VBX1wNrxAqMX87RBD5xpJuddqwd82tjAHxzV1U5LPgy52DKXWnr1Lheg+j/c/pzGVr' + * 'iqV0VlwZPXGPCJjElw/ybkwUmeoWgxesDXGhHJC/D/iikp1Av80ptKU0FdBEe25pPihAM1' + * 'u6ytgaaWfs2Hrz35CJT1+EWmAKZW5kc3RyZWFtCmVuZG9iagoKNyAwIG9iago8PAovU2l6' + * 'ZSA4Ci9Sb290IDIgMCBSCi9GaWx0ZXIgL0ZsYXRlRGVjb2RlCi9UeXBlIC9YUmVmCi9MZW' + * '5ndGggMzgKL1cgWyAxIDIgMiBdCi9JbmRleCBbIDAgOCBdCj4+CnN0cmVhbQp4nBXEwREA' + * 'EBAEsCwz3vrvRmOOyyOoGhZdutHN2MT55fIAVocD+AplbmRzdHJlYW0KZW5kb2JqCgpzdG' + * 'FydHhyZWYKNTEwCiUlRU9G' * * const dataUri = 'data:application/pdf;base64,' + base64 * * const pdfDoc1 = await PDFDocument.load(base64) * const pdfDoc2 = await PDFDocument.load(dataUri) * * // pdf=Uint8Array * import fs from 'fs' * const uint8Array = fs.readFileSync('with_update_sections.pdf') * const pdfDoc3 = await PDFDocument.load(uint8Array) * * // pdf=ArrayBuffer * const url = 'https://pdf-lib.js.org/assets/with_update_sections.pdf' * const arrayBuffer = await fetch(url).then(res => res.arrayBuffer()) * const pdfDoc4 = await PDFDocument.load(arrayBuffer) * * ``` * * @param pdf The input data containing a PDF document. * @param options The options to be used when loading the document. * @returns Resolves with a document loaded from the input. */ PDFDocument.load = function (pdf, options) { if (options === void 0) { options = {}; } return tslib_1.__awaiter(this, void 0, void 0, function () { var _a, ignoreEncryption, _b, parseSpeed, bytes, context; return tslib_1.__generator(this, function (_c) { switch (_c.label) { case 0: _a = options.ignoreEncryption, ignoreEncryption = _a === void 0 ? false : _a, _b = options.parseSpeed, parseSpeed = _b === void 0 ? ParseSpeeds.Slow : _b; utils_1.assertIs(pdf, 'pdf', ['string', Uint8Array, ArrayBuffer]); utils_1.assertIs(ignoreEncryption, 'ignoreEncryption', ['boolean']); utils_1.assertIs(parseSpeed, 'parseSpeed', ['number']); bytes = utils_1.toUint8Array(pdf); return [4 /*yield*/, core_1.PDFParser.forBytesWithOptions(bytes, parseSpeed).parseDocument()]; case 1: context = _c.sent(); return [2 /*return*/, new PDFDocument(context, ignoreEncryption)]; } }); }); }; /** * Create a new [[PDFDocument]]. * @returns Resolves with the newly created document. */ PDFDocument.create = function () { return tslib_1.__awaiter(this, void 0, void 0, function () { var context, pageTree, pageTreeRef, catalog; return tslib_1.__generator(this, function (_a) { context = core_1.PDFContext.create(); pageTree = core_1.PDFPageTree.withContext(context); pageTreeRef = context.register(pageTree); catalog = core_1.PDFCatalog.withContextAndPages(context, pageTreeRef); context.trailerInfo.Root = context.register(catalog); return [2 /*return*/, new PDFDocument(context, false)]; }); }); }; /** * Register a fontkit instance. This must be done before custom fonts can * be embedded. See [here](https://github.com/Hopding/pdf-lib/tree/Rewrite#fontkit-installation) * for instructions on how to install and register a fontkit instance. * * > You do **not** need to call this method to embed standard fonts. * * @param fontkit The fontkit instance to be registered. */ PDFDocument.prototype.registerFontkit = function (fontkit) { this.fontkit = fontkit; }; /** * Set this document's title metadata. The title will appear in the * "Document Properties" section of most PDF readers. For example: * ```js * pdfDoc.setTitle('🥚 The Life of an Egg 🍳') * ``` * @param title The title of this document. */ PDFDocument.prototype.setTitle = function (title) { utils_1.assertIs(title, 'title', ['string']); var key = core_1.PDFName.of('Title'); this.getInfoDict().set(key, core_1.PDFHexString.fromText(title)); }; /** * Set this document's author metadata. The author will appear in the * "Document Properties" section of most PDF readers. For example: * ```js * pdfDoc.setAuthor('Humpty Dumpty') * ``` * @param author The author of this document. */ PDFDocument.prototype.setAuthor = function (author) { utils_1.assertIs(author, 'author', ['string']); var key = core_1.PDFName.of('Author'); this.getInfoDict().set(key, core_1.PDFHexString.fromText(author)); }; /** * Set this document's subject metadata. The subject will appear in the * "Document Properties" section of most PDF readers. For example: * ```js * pdfDoc.setSubject('📘 An Epic Tale of Woe 📖') * ``` * @param subject The subject of this document. */ PDFDocument.prototype.setSubject = function (subject) { utils_1.assertIs(subject, 'author', ['string']); var key = core_1.PDFName.of('Subject'); this.getInfoDict().set(key, core_1.PDFHexString.fromText(subject)); }; /** * Set this document's keyword metadata. These keywords will appear in the * "Document Properties" section of most PDF readers. For example: * ```js * pdfDoc.setKeywords(['eggs', 'wall', 'fall', 'king', 'horses', 'men']) * ``` * @param keywords An array of keywords associated with this document. */ PDFDocument.prototype.setKeywords = function (keywords) { utils_1.assertIs(keywords, 'keywords', [Array]); var key = core_1.PDFName.of('Keywords'); this.getInfoDict().set(key, core_1.PDFHexString.fromText(keywords.join(' '))); }; /** * Set this document's creator metadata. The creator will appear in the * "Document Properties" section of most PDF readers. For example: * ```js * pdfDoc.setCreator('PDF App 9000 🤖') * ``` * @param creator The creator of this document. */ PDFDocument.prototype.setCreator = function (creator) { utils_1.assertIs(creator, 'creator', ['string']); var key = core_1.PDFName.of('Creator'); this.getInfoDict().set(key, core_1.PDFHexString.fromText(creator)); }; /** * Set this document's producer metadata. The producer will appear in the * "Document Properties" section of most PDF readers. For example: * ```js * pdfDoc.setProducer('PDF App 9000 🤖') * ``` * @param producer The producer of this document. */ PDFDocument.prototype.setProducer = function (producer) { utils_1.assertIs(producer, 'creator', ['string']); var key = core_1.PDFName.of('Producer'); this.getInfoDict().set(key, core_1.PDFHexString.fromText(producer)); }; /** * Set this document's creation date metadata. The creation date will appear * in the "Document Properties" section of most PDF readers. For example: * ```js * pdfDoc.setCreationDate(new Date()) * ``` * @param creationDate The date this document was created. */ PDFDocument.prototype.setCreationDate = function (creationDate) { utils_1.assertIs(creationDate, 'creationDate', [[Date, 'Date']]); var key = core_1.PDFName.of('CreationDate'); this.getInfoDict().set(key, core_1.PDFString.fromDate(creationDate)); }; /** * Set this document's modification date metadata. The modification date will * appear in the "Document Properties" section of most PDF readers. For * example: * ```js * pdfDoc.setModificationDate(new Date()) * ``` * @param modificationDate The date this document was last modified. */ PDFDocument.prototype.setModificationDate = function (modificationDate) { utils_1.assertIs(modificationDate, 'modificationDate', [[Date, 'Date']]); var key = core_1.PDFName.of('ModDate'); this.getInfoDict().set(key, core_1.PDFString.fromDate(modificationDate)); }; /** * Get the number of pages contained in this document. For example: * ```js * const totalPages = pdfDoc.getPageCount() * ``` * @returns The number of pages in this document. */ PDFDocument.prototype.getPageCount = function () { if (this.pageCount === undefined) this.pageCount = this.getPages().length; return this.pageCount; }; /** * Get an array of all the pages contained in this document. The pages are * stored in the array in the same order that they are rendered in the * document. For example: * ```js * const pages = pdfDoc.getPages() * pages[0] // The first page of the document * pages[2] // The third page of the document * pages[197] // The 198th page of the document * ``` * @returns An array of all the pages contained in this document. */ PDFDocument.prototype.getPages = function () { return this.pageCache.access(); }; /** * Get an array of indices for all the pages contained in this document. The * array will contain a range of integers from * `0..pdfDoc.getPageCount() - 1`. For example: * ```js * const pdfDoc = await PDFDocument.create() * pdfDoc.addPage() * pdfDoc.addPage() * pdfDoc.addPage() * * const indices = pdfDoc.getPageIndices() * indices // => [0, 1, 2] * ``` * @returns An array of indices for all pages contained in this document. */ PDFDocument.prototype.getPageIndices = function () { return utils_1.range(0, this.getPageCount()); }; /** * Remove the page at a given index from this document. For example: * ```js * pdfDoc.removePage(0) // Remove the first page of the document * pdfDoc.removePage(2) // Remove the third page of the document * pdfDoc.removePage(197) // Remove the 198th page of the document * ``` * Once a page has been removed, it will no longer be rendered at that index * in the document. * @param index The index of the page to be removed. */ PDFDocument.prototype.removePage = function (index) { var pageCount = this.getPageCount(); if (this.pageCount === 0) throw new errors_1.RemovePageFromEmptyDocumentError(); utils_1.assertRange(index, 'index', 0, pageCount - 1); this.catalog.removeLeafNode(index); this.pageCount = pageCount - 1; }; /** * Add a page to the end of this document. This method accepts three * different value types for the `page` parameter: * * | Type | Behavior | * | ------------------ | ----------------------------------------------------------------------------------- | * | `undefined` | Create a new page and add it to the end of this document | * | `[number, number]` | Create a new page with the given dimensions and add it to the end of this document | * | `PDFPage` | Add the existing page to the end of this document | * * For example: * ```js * // page=undefined * const newPage = pdfDoc.addPage() * * // page=[number, number] * import { PageSizes } from 'pdf-lib' * const newPage1 = pdfDoc.addPage(PageSizes.A7) * const newPage2 = pdfDoc.addPage(PageSizes.Letter) * const newPage3 = pdfDoc.addPage([500, 750]) * * // page=PDFPage * const pdfDoc1 = await PDFDocument.create() * const pdfDoc2 = await PDFDocument.load(...) * const [existingPage] = await pdfDoc1.copyPages(pdfDoc2, [0]) * pdfDoc1.addPage(existingPage) * ``` * * @param page Optionally, the desired dimensions or existing page. * @returns The newly created (or existing) page. */ PDFDocument.prototype.addPage = function (page) { utils_1.assertIs(page, 'page', ['undefined', [PDFPage_1.default, 'PDFPage'], Array]); return this.insertPage(this.getPageCount(), page); }; /** * Insert a page at a given index within this document. This method accepts * three different value types for the `page` parameter: * * | Type | Behavior | * | ------------------ | ------------------------------------------------------------------------------ | * | `undefined` | Create a new page and insert it into this document | * | `[number, number]` | Create a new page with the given dimensions and insert it into this document | * | `PDFPage` | Insert the existing page into this document | * * For example: * ```js * // page=undefined * const newPage = pdfDoc.insertPage(2) * * // page=[number, number] * import { PageSizes } from 'pdf-lib' * const newPage1 = pdfDoc.insertPage(2, PageSizes.A7) * const newPage2 = pdfDoc.insertPage(0, PageSizes.Letter) * const newPage3 = pdfDoc.insertPage(198, [500, 750]) * * // page=PDFPage * const pdfDoc1 = await PDFDocument.create() * const pdfDoc2 = await PDFDocument.load(...) * const [existingPage] = await pdfDoc1.copyPages(pdfDoc2, [0]) * pdfDoc1.insertPage(0, existingPage) * ``` * * @param index The index at which the page should be inserted (zero-based). * @param page Optionally, the desired dimensions or existing page. * @returns The newly created (or existing) page. */ PDFDocument.prototype.insertPage = function (index, page) { var pageCount = this.getPageCount(); utils_1.assertRange(index, 'index', 0, pageCount); utils_1.assertIs(page, 'page', ['undefined', [PDFPage_1.default, 'PDFPage'], Array]); if (!page || Array.isArray(page)) { var dims = Array.isArray(page) ? page : sizes_1.PageSizes.A4; page = PDFPage_1.default.create(this); page.setSize.apply(page, dims); } else if (page.doc !== this) { throw new errors_1.ForeignPageError(); } var parentRef = this.catalog.insertLeafNode(page.ref, index); page.node.setParent(parentRef); this.pageMap.set(page.node, page); this.pageCache.invalidate(); this.pageCount = pageCount + 1; return page; }; /** * Copy pages from a source document into this document. Allows pages to be * copied between different [[PDFDocument]] instances. For example: * ```js * const pdfDoc = await PDFDocument.create() * const srcDoc = await PDFDocument.load(...) * * const copiedPages = await pdfDoc.copyPages(srcDoc, [0, 3, 89]) * const [firstPage, fourthPage, ninetiethPage] = copiedPages; * * pdfDoc.addPage(fourthPage) * pdfDoc.insertPage(0, ninetiethPage) * pdfDoc.addPage(firstPage) * ``` * @param srcDoc The document from which pages should be copied. * @param indices The indices of the pages that should be copied. * @returns Resolves with an array of pages copied into this document. */ PDFDocument.prototype.copyPages = function (srcDoc, indices) { return tslib_1.__awaiter(this, void 0, void 0, function () { var copier, srcPages, copiedPages, idx, len, srcPage, copiedPage, ref; return tslib_1.__generator(this, function (_a) { switch (_a.label) { case 0: utils_1.assertIs(srcDoc, 'srcDoc', [[PDFDocument, 'PDFDocument']]); utils_1.assertIs(indices, 'indices', [Array]); return [4 /*yield*/, srcDoc.flush()]; case 1: _a.sent(); copier = core_1.PDFObjectCopier.for(srcDoc.context, this.context); srcPages = srcDoc.getPages(); copiedPages = new Array(indices.length); for (idx = 0, len = indices.length; idx < len; idx++) { srcPage = srcPages[indices[idx]]; copiedPage = copier.copy(srcPage.node); ref = this.context.register(copiedPage); copiedPages[idx] = PDFPage_1.default.of(copiedPage, ref, this); } return [2 /*return*/, copiedPages]; } }); }); }; /** * Embed a font into this document. The input data can be provided in multiple * formats: * * | Type | Contents | * | --------------- | ------------------------------------------------------- | * | `StandardFonts` | One of the standard 14 fonts | * | `string` | A base64 encoded string (or data URI) containing a font | * | `Uint8Array` | The raw bytes of a font | * | `ArrayBuffer` | The raw bytes of a font | * * For example: * ```js * // font=StandardFonts * import { StandardFonts } from 'pdf-lib' * const font1 = await pdfDoc.embedFont(StandardFonts.Helvetica) * * // font=string * const font2 = await pdfDoc.embedFont('AAEAAAAVAQAABABQRFNJRx/upe...') * const font3 = await pdfDoc.embedFont('data:font/opentype;base64,AAEAAA...') * * // font=Uint8Array * import fs from 'fs' * const font4 = await pdfDoc.embedFont(fs.readFileSync('Ubuntu-R.ttf')) * * // font=ArrayBuffer * const url = 'https://pdf-lib.js.org/assets/ubuntu/Ubuntu-R.ttf' * const ubuntuBytes = await fetch(url).then(res => res.arrayBuffer()) * const font5 = await pdfDoc.embedFont(ubuntuBytes) * ``` * See also: [[registerFontkit]] * @param font The input data for a font. * @param options The options to be used when embedding the font. * @returns Resolves with the embedded font. */ PDFDocument.prototype.embedFont = function (font, options) { if (options === void 0) { options = {}; } return tslib_1.__awaiter(this, void 0, void 0, function () { var _a, subset, embedder, bytes, fontkit, _b, ref, pdfFont; return tslib_1.__generator(this, function (_c) { switch (_c.label) { case 0: _a = options.subset, subset = _a === void 0 ? false : _a; utils_1.assertIs(font, 'font', ['string', Uint8Array, ArrayBuffer]); utils_1.assertIs(subset, 'subset', ['boolean']); if (!utils_1.isStandardFont(font)) return [3 /*break*/, 1]; embedder = core_1.StandardFontEmbedder.for(font); return [3 /*break*/, 7]; case 1: if (!utils_1.canBeConvertedToUint8Array(font)) return [3 /*break*/, 6]; bytes = utils_1.toUint8Array(font); fontkit = this.assertFontkit(); if (!subset) return [3 /*break*/, 3]; return [4 /*yield*/, core_1.CustomFontSubsetEmbedder.for(fontkit, bytes)]; case 2: _b = _c.sent(); return [3 /*break*/, 5]; case 3: return [4 /*yield*/, core_1.CustomFontEmbedder.for(fontkit, bytes)]; case 4: _b = _c.sent(); _c.label = 5; case 5: embedder = _b; return [3 /*break*/, 7]; case 6: throw new TypeError('`font` must be one of `StandardFonts | string | Uint8Array | ArrayBuffer`'); case 7: ref = this.context.nextRef(); pdfFont = PDFFont_1.default.of(ref, this, embedder); this.fonts.push(pdfFont); return [2 /*return*/, pdfFont]; } }); }); }; /** * Embed a standard font into this document. * For example: * ```js * import { StandardFonts } from 'pdf-lib' * const helveticaFont = pdfDoc.embedFont(StandardFonts.Helvetica) * ``` * @param font The standard font to be embedded. * @returns The embedded font. */ PDFDocument.prototype.embedStandardFont = function (font) { utils_1.assertIs(font, 'font', ['string']); if (!utils_1.isStandardFont(font)) { throw new TypeError('`font` must be one of type `StandardFontsr`'); } var embedder = core_1.StandardFontEmbedder.for(font); var ref = this.context.nextRef(); var pdfFont = PDFFont_1.default.of(ref, this, embedder); this.fonts.push(pdfFont); return pdfFont; }; /** * Embed a JPEG image into this document. The input data can be provided in * multiple formats: * * | Type | Contents | * | ------------- | ------------------------------------------------------------- | * | `string` | A base64 encoded string (or data URI) containing a JPEG image | * | `Uint8Array` | The raw bytes of a JPEG image | * | `ArrayBuffer` | The raw bytes of a JPEG image | * * For example: * ```js * // jpg=string * const image1 = await pdfDoc.embedJpg('/9j/4AAQSkZJRgABAQAAAQABAAD/2wBD...') * const image2 = await pdfDoc.embedJpg('...') * * // jpg=Uint8Array * import fs from 'fs' * const uint8Array = fs.readFileSync('cat_riding_unicorn.jpg') * const image3 = await pdfDoc.embedJpg(uint8Array) * * // jpg=ArrayBuffer * const url = 'https://pdf-lib.js.org/assets/cat_riding_unicorn.jpg' * const arrayBuffer = await fetch(url).then(res => res.arrayBuffer()) * const image4 = await pdfDoc.embedJpg(arrayBuffer) * ``` * * @param jpg The input data for a JPEG image. * @returns Resolves with the embedded image. */ PDFDocument.prototype.embedJpg = function (jpg) { return tslib_1.__awaiter(this, void 0, void 0, function () { var bytes, embedder, ref, pdfImage; return tslib_1.__generator(this, function (_a) { switch (_a.label) { case 0: utils_1.assertIs(jpg, 'jpg', ['string', Uint8Array, ArrayBuffer]); bytes = utils_1.toUint8Array(jpg); return [4 /*yield*/, core_1.JpegEmbedder.for(bytes)]; case 1: embedder = _a.sent(); ref = this.context.nextRef(); pdfImage = PDFImage_1.default.of(ref, this, embedder); this.images.push(pdfImage); return [2 /*return*/, pdfImage]; } }); }); }; /** * Embed a PNG image into this document. The input data can be provided in * multiple formats: * * | Type | Contents | * | ------------- | ------------------------------------------------------------ | * | `string` | A base64 encoded string (or data URI) containing a PNG image | * | `Uint8Array` | The raw bytes of a PNG image | * | `ArrayBuffer` | The raw bytes of a PNG image | * * For example: * ```js * // png=string * const image1 = await pdfDoc.embedPng('iVBORw0KGgoAAAANSUhEUgAAAlgAAAF3...') * const image2 = await pdfDoc.embedPng('...') * * // png=Uint8Array * import fs from 'fs' * const uint8Array = fs.readFileSync('small_mario.png') * const image3 = await pdfDoc.embedPng(uint8Array) * * // png=ArrayBuffer * const url = 'https://pdf-lib.js.org/assets/small_mario.png' * const arrayBuffer = await fetch(url).then(res => res.arrayBuffer()) * const image4 = await pdfDoc.embedPng(arrayBuffer) * ``` * * @param png The input data for a PNG image. * @returns Resolves with the embedded image. */ PDFDocument.prototype.embedPng = function (png) { return tslib_1.__awaiter(this, void 0, void 0, function () { var bytes, embedder, ref, pdfImage; return tslib_1.__generator(this, function (_a) { switch (_a.label) { case 0: utils_1.assertIs(png, 'png', ['string', Uint8Array, ArrayBuffer]); bytes = utils_1.toUint8Array(png); return [4 /*yield*/, core_1.PngEmbedder.for(bytes)]; case 1: embedder = _a.sent(); ref = this.context.nextRef(); pdfImage = PDFImage_1.default.of(ref, this, embedder); this.images.push(pdfImage); return [2 /*return*/, pdfImage]; } }); }); }; /** * > **NOTE:** You shouldn't need to call this method directly. The [[save]] * > and [[saveAsBase64]] methods will automatically ensure that all embedded * > assets are flushed before serializing the document. * * Flush all embedded fonts and images to this document's [[context]]. * * @returns Resolves when the flush is complete. */ PDFDocument.prototype.flush = function () { return tslib_1.__awaiter(this, void 0, void 0, function () { var idx, len, font, idx, len, image; return tslib_1.__generator(this, function (_a) { switch (_a.label) { case 0: idx = 0, len = this.fonts.length; _a.label = 1; case 1: if (!(idx < len)) return [3 /*break*/, 4]; font = this.fonts[idx]; return [4 /*yield*/, font.embed()]; case 2: _a.sent(); _a.label = 3; case 3: idx++; return [3 /*break*/, 1]; case 4: idx = 0, len = this.images.length; _a.label = 5; case 5: if (!(idx < len)) return [3 /*break*/, 8]; image = this.images[idx]; return [4 /*yield*/, image.embed()]; case 6: _a.sent(); _a.label = 7; case 7: idx++; return [3 /*break*/, 5]; case 8: return [2 /*return*/]; } }); }); }; /** * Serialize this document to an array of bytes making up a PDF file. * For example: * ```js * const pdfBytes = await pdfDoc.save() * ``` * * There are a number of things you can do with the serialized document, * depending on the JavaScript environment you're running in: * * Write it to a file in Node or React Native * * Download it as a Blob in the browser * * Render it in an `iframe` * * @param options The options to be used when saving the document. * @returns Resolves with the bytes of the serialized document. */ PDFDocument.prototype.save = function (options) { if (options === void 0) { options = {}; } return tslib_1.__awaiter(this, void 0, void 0, function () { var _a, useObjectStreams, _b, addDefaultPage, _c, objectsPerTick, Writer; return tslib_1.__generator(this, function (_d) { switch (_d.label) { case 0: _a = options.useObjectStreams, useObjectStreams = _a === void 0 ? true : _a, _b = options.addDefaultPage, addDefaultPage = _b === void 0 ? true : _b, _c = options.objectsPerTick, objectsPerTick = _c === void 0 ? 50 : _c; utils_1.assertIs(useObjectStreams, 'useObjectStreams', ['boolean']); utils_1.assertIs(addDefaultPage, 'addDefaultPage', ['boolean']); utils_1.assertIs(objectsPerTick, 'objectsPerTick', ['number']); if (addDefaultPage && this.pageCount === 0) this.addPage(); return [4 /*yield*/, this.flush()]; case 1: _d.sent(); Writer = useObjectStreams ? core_1.PDFStreamWriter : core_1.PDFWriter; return [2 /*return*/, Writer.forContext(this.context, objectsPerTick).serializeToBuffer()]; } }); }); }; /** * Serialize this document to a base64 encoded string or data URI making up a * PDF file. For example: * ```js * const base64String = await pdfDoc.saveAsBase64() * base64String // => 'JVBERi0xLjcKJYGBgYEKC...' * * const base64DataUri = await pdfDoc.saveAsBase64({ dataUri: true }) * base64DataUri // => 'data:application/pdf;base64,JVBERi0xLjcKJYGBgYEKC...' * ``` * * @param options The options to be used when saving the document. * @returns Resolves with a base64 encoded string or data URI of the * serialized document. */ PDFDocument.prototype.saveAsBase64 = function (options) { if (options === void 0) { options = {}; } return tslib_1.__awaiter(this, void 0, void 0, function () { var _a, dataUri, otherOptions, bytes, base64; return tslib_1.__generator(this, function (_b) { switch (_b.label) { case 0: _a = options.dataUri, dataUri = _a === void 0 ? false : _a, otherOptions = tslib_1.__rest(options, ["dataUri"]); utils_1.assertIs(dataUri, 'dataUri', ['boolean']); return [4 /*yield*/, this.save(otherOptions)]; case 1: bytes = _b.sent(); base64 = utils_1.encodeToBase64(bytes); return [2 /*return*/, dataUri ? "data:application/pdf;base64," + base64 : base64]; } }); }); }; PDFDocument.prototype.updateInfoDict = function () { var pdfLib = "pdf-lib (https://github.com/Hopding/pdf-lib)"; var now = new Date(); var info = this.getInfoDict(); this.setProducer(pdfLib); this.setModificationDate(now); if (!info.get(core_1.PDFName.of('Creator'))) this.setCreator(pdfLib); if (!info.get(core_1.PDFName.of('CreationDate'))) this.setCreationDate(now); }; PDFDocument.prototype.getInfoDict = function () { var existingInfo = this.context.lookup(this.context.trailerInfo.Info); if (existingInfo instanceof core_1.PDFDict) return existingInfo; var newInfo = this.context.obj({}); this.context.trailerInfo.Info = this.context.register(newInfo); return newInfo; }; PDFDocument.prototype.assertFontkit = function () { if (!this.fontkit) throw new errors_1.FontkitNotRegisteredError(); return this.fontkit; }; return PDFDocument; }()); exports.default = PDFDocument; //# sourceMappingURL=PDFDocument.js.map