UNPKG

pdf-lib

Version:

Create and modify PDF files with JavaScript

572 lines (517 loc) 20.4 kB
import PDFDocument from 'src/api/PDFDocument'; import PDFPage from 'src/api/PDFPage'; import PDFFont from 'src/api/PDFFont'; import PDFField, { FieldAppearanceOptions, assertFieldAppearanceOptions, } from 'src/api/form/PDFField'; import { AppearanceProviderFor, normalizeAppearance, defaultOptionListAppearanceProvider, } from 'src/api/form/appearances'; import { rgb } from 'src/api/colors'; import { degrees } from 'src/api/rotations'; import { PDFRef, PDFHexString, PDFString, PDFStream, PDFAcroListBox, AcroChoiceFlags, PDFWidgetAnnotation, } from 'src/core'; import { assertIs, assertIsSubset, assertOrUndefined, assertPositive, } from 'src/utils'; /** * Represents an option list field of a [[PDFForm]]. * * [[PDFOptionList]] fields are interactive lists of options. The purpose of an * option list is to enable users to select one or more options from a set of * possible options. Users are able to see the full set of options without * first having to click on the field (though scrolling may be necessary). * Clicking an option in the list will cause it to be selected and displayed * with a highlighted background. Some option lists allow users to select * more than one option (see [[PDFOptionList.isMultiselect]]). */ export default class PDFOptionList extends PDFField { /** * > **NOTE:** You probably don't want to call this method directly. Instead, * > consider using the [[PDFForm.getOptionList]] method, which will create * > an instance of [[PDFOptionList]] for you. * * Create an instance of [[PDFOptionList]] from an existing acroListBox and * ref * * @param acroComboBox The underlying `PDFAcroListBox` for this option list. * @param ref The unique reference for this option list. * @param doc The document to which this option list will belong. */ static of = (acroListBox: PDFAcroListBox, ref: PDFRef, doc: PDFDocument) => new PDFOptionList(acroListBox, ref, doc); /** The low-level PDFAcroListBox wrapped by this option list. */ readonly acroField: PDFAcroListBox; private constructor( acroListBox: PDFAcroListBox, ref: PDFRef, doc: PDFDocument, ) { super(acroListBox, ref, doc); assertIs(acroListBox, 'acroListBox', [[PDFAcroListBox, 'PDFAcroListBox']]); this.acroField = acroListBox; } /** * Get the list of available options for this option list. These options will * be displayed to users who view this option list in a PDF reader. * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * const options = optionList.getOptions() * console.log('Option List options:', options) * ``` * @returns The options for this option list. */ getOptions(): string[] { const rawOptions = this.acroField.getOptions(); const options = new Array<string>(rawOptions.length); for (let idx = 0, len = options.length; idx < len; idx++) { const { display, value } = rawOptions[idx]; options[idx] = (display ?? value).decodeText(); } return options; } /** * Get the selected options for this option list. These are the values that * were selected by a human user via a PDF reader, or programatically via * software. * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * const selections = optionList.getSelected() * console.log('Option List selections:', selections) * ``` * @returns The selected options for this option list. */ getSelected(): string[] { const values = this.acroField.getValues(); const selected = new Array<string>(values.length); for (let idx = 0, len = values.length; idx < len; idx++) { selected[idx] = values[idx].decodeText(); } return selected; } /** * Set the list of options that are available for this option list. These are * the values that will be available for users to select when they view this * option list in a PDF reader. Note that preexisting options for this * option list will be removed. Only the values passed as `options` will be * available to select. * * For example: * ```js * const optionList = form.getOptionList('planets.optionList') * optionList.setOptions(['Earth', 'Mars', 'Pluto', 'Venus']) * ``` * * This method will mark this option list as dirty, causing its appearance * streams to be updated when either [[PDFDocument.save]] or * [[PDFForm.updateFieldAppearances]] is called. The updated streams will * display the options this field contains inside the widgets of this text * field (with selected options highlighted). * * **IMPORTANT:** The default font used to update appearance streams is * [[StandardFonts.Helvetica]]. Note that this is a WinAnsi font. This means * that encoding errors will be thrown if this field contains any options * with characters outside the WinAnsi character set (the latin alphabet). * * Embedding a custom font and passing it to * [[PDFForm.updateFieldAppearances]] or [[PDFOptionList.updateAppearances]] * allows you to generate appearance streams with characters outside the * latin alphabet (assuming the custom font supports them). * * @param options The options that should be available in this option list. */ setOptions(options: string[]) { assertIs(options, 'options', [Array]); this.markAsDirty(); const optionObjects = new Array<{ value: PDFHexString }>(options.length); for (let idx = 0, len = options.length; idx < len; idx++) { optionObjects[idx] = { value: PDFHexString.fromText(options[idx]) }; } this.acroField.setOptions(optionObjects); } /** * Add to the list of options that are available for this option list. Users * will be able to select these values in a PDF reader. In addition to the * values passed as `options`, any preexisting options for this option list * will still be available for users to select. * For example: * ```js * const optionList = form.getOptionList('rockets.optionList') * optionList.addOptions(['Saturn IV', 'Falcon Heavy']) * ``` * This method will mark this option list as dirty. See * [[PDFOptionList.setOptions]] for more details about what this means. * @param options New options that should be available in this option list. */ addOptions(options: string | string[]) { assertIs(options, 'options', ['string', Array]); this.markAsDirty(); const optionsArr = Array.isArray(options) ? options : [options]; const existingOptions: { value: PDFString | PDFHexString; display?: PDFString | PDFHexString; }[] = this.acroField.getOptions(); const newOptions = new Array<{ value: PDFHexString }>(optionsArr.length); for (let idx = 0, len = optionsArr.length; idx < len; idx++) { newOptions[idx] = { value: PDFHexString.fromText(optionsArr[idx]) }; } this.acroField.setOptions(existingOptions.concat(newOptions)); } /** * Select one or more values for this option list. This operation is analogous * to a human user opening the option list in a PDF reader and clicking on one * or more values to select them. This method will update the underlying state * of the option list to indicate which values have been selected. PDF * libraries and readers will be able to extract these values from the saved * document and determine which values were selected. * For example: * ```js * const optionList = form.getOptionList('best.superheroes.optionList') * optionList.select(['One Punch Man', 'Iron Man']) * ``` * This method will mark this option list as dirty. See * [[PDFOptionList.setOptions]] for more details about what this means. * @param options The options to be selected. * @param merge Whether or not existing selections should be preserved. */ select(options: string | string[], merge = false) { assertIs(options, 'options', ['string', Array]); assertIs(merge, 'merge', ['boolean']); const optionsArr = Array.isArray(options) ? options : [options]; const validOptions = this.getOptions(); assertIsSubset(optionsArr, 'option', validOptions); this.markAsDirty(); if (optionsArr.length > 1 || (optionsArr.length === 1 && merge)) { this.enableMultiselect(); } const values = new Array<PDFHexString>(optionsArr.length); for (let idx = 0, len = optionsArr.length; idx < len; idx++) { values[idx] = PDFHexString.fromText(optionsArr[idx]); } if (merge) { const existingValues = this.acroField.getValues(); this.acroField.setValues(existingValues.concat(values)); } else { this.acroField.setValues(values); } } /** * Clear all selected values for this option list. This operation is * equivalent to selecting an empty list. This method will update the * underlying state of the option list to indicate that no values have been * selected. * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.clear() * ``` * This method will mark this option list as dirty. See * [[PDFOptionList.setOptions]] for more details about what this means. */ clear() { this.markAsDirty(); this.acroField.setValues([]); } /** * Set the font size for the text in this field. There needs to be a * default appearance string (DA) set with a font value specified * for this to work. For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.setFontSize(4); * ``` * @param fontSize The font size to set the font to. */ /** * Set the font size for this field. Larger font sizes will result in larger * text being displayed when PDF readers render this option list. Font sizes * may be integer or floating point numbers. Supplying a negative font size * will cause this method to throw an error. * * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.setFontSize(4) * optionList.setFontSize(15.7) * ``` * * > This method depends upon the existence of a default appearance * > (`/DA`) string. If this field does not have a default appearance string, * > or that string does not contain a font size (via the `Tf` operator), * > then this method will throw an error. * * @param fontSize The font size to be used when rendering text in this field. */ setFontSize(fontSize: number) { assertPositive(fontSize, 'fontSize'); this.acroField.setFontSize(fontSize); this.markAsDirty(); } /** * Returns `true` if the options of this option list are always displayed * in alphabetical order, irrespective of the order in which the options * were added to the option list. See [[PDFOptionList.enableSorting]] and * [[PDFOptionList.disableSorting]]. For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * if (optionList.isSorted()) console.log('Sorting is enabled') * ``` * @returns Whether or not this option list is sorted. */ isSorted(): boolean { return this.acroField.hasFlag(AcroChoiceFlags.Sort); } /** * Always display the options of this option list in alphabetical order, * irrespective of the order in which the options were added to this option * list. * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.enableSorting() * ``` */ enableSorting() { this.acroField.setFlagTo(AcroChoiceFlags.Sort, true); } /** * Do not always display the options of this option list in alphabetical * order. Instead, display the options in whichever order they were added * to this option list. For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.disableSorting() * ``` */ disableSorting() { this.acroField.setFlagTo(AcroChoiceFlags.Sort, false); } /** * Returns `true` if multiple options can be selected from this option list. * See [[PDFOptionList.enableMultiselect]] and * [[PDFOptionList.disableMultiselect]]. For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * if (optionList.isMultiselect()) console.log('Multiselect is enabled') * ``` * @returns Whether or not multiple options can be selected. */ isMultiselect(): boolean { return this.acroField.hasFlag(AcroChoiceFlags.MultiSelect); } /** * Allow users to select more than one option from this option list. * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.enableMultiselect() * ``` */ enableMultiselect() { this.acroField.setFlagTo(AcroChoiceFlags.MultiSelect, true); } /** * Do not allow users to select more than one option from this option list. * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.disableMultiselect() * ``` */ disableMultiselect() { this.acroField.setFlagTo(AcroChoiceFlags.MultiSelect, false); } /** * Returns `true` if the option selected by a user is stored, or "committed", * when the user clicks the option. The alternative is that the user's * selection is stored when the user leaves this option list field (by * clicking outside of it - on another field, for example). See * [[PDFOptionList.enableSelectOnClick]] and * [[PDFOptionList.disableSelectOnClick]]. For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * if (optionList.isSelectOnClick()) console.log('Select on click is enabled') * ``` * @returns Whether or not options are selected immediately after they are * clicked. */ isSelectOnClick(): boolean { return this.acroField.hasFlag(AcroChoiceFlags.CommitOnSelChange); } /** * Store the option selected by a user immediately after the user clicks the * option. Do not wait for the user to leave this option list field (by * clicking outside of it - on another field, for example). For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.enableSelectOnClick() * ``` */ enableSelectOnClick() { this.acroField.setFlagTo(AcroChoiceFlags.CommitOnSelChange, true); } /** * Wait to store the option selected by a user until they leave this option * list field (by clicking outside of it - on another field, for example). * For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * optionList.disableSelectOnClick() * ``` */ disableSelectOnClick() { this.acroField.setFlagTo(AcroChoiceFlags.CommitOnSelChange, false); } /** * Show this option list on the specified page. For example: * ```js * const ubuntuFont = await pdfDoc.embedFont(ubuntuFontBytes) * const page = pdfDoc.addPage() * * const form = pdfDoc.getForm() * const optionList = form.createOptionList('best.gundams') * optionList.setOptions(['Exia', 'Dynames', 'Kyrios', 'Virtue']) * optionList.select(['Exia', 'Virtue']) * * optionList.addToPage(page, { * x: 50, * y: 75, * width: 200, * height: 100, * textColor: rgb(1, 0, 0), * backgroundColor: rgb(0, 1, 0), * borderColor: rgb(0, 0, 1), * borderWidth: 2, * rotate: degrees(90), * font: ubuntuFont, * }) * ``` * This will create a new widget for this option list field. * @param page The page to which this option list widget should be added. * @param options The options to be used when adding this option list widget. */ addToPage(page: PDFPage, options?: FieldAppearanceOptions) { assertIs(page, 'page', [[PDFPage, 'PDFPage']]); assertFieldAppearanceOptions(options); if (!options) options = {}; if (!('textColor' in options)) options.textColor = rgb(0, 0, 0); if (!('backgroundColor' in options)) options.backgroundColor = rgb(1, 1, 1); if (!('borderColor' in options)) options.borderColor = rgb(0, 0, 0); if (!('borderWidth' in options)) options.borderWidth = 1; // Create a widget for this option list const widget = this.createWidget({ x: options.x ?? 0, y: options.y ?? 0, width: options.width ?? 200, height: options.height ?? 100, textColor: options.textColor, backgroundColor: options.backgroundColor, borderColor: options.borderColor, borderWidth: options.borderWidth ?? 0, rotate: options.rotate ?? degrees(0), hidden: options.hidden, page: page.ref, }); const widgetRef = this.doc.context.register(widget.dict); // Add widget to this field this.acroField.addWidget(widgetRef); // Set appearance streams for widget const font = options.font ?? this.doc.getForm().getDefaultFont(); this.updateWidgetAppearance(widget, font); // Add widget to the given page page.node.addAnnot(widgetRef); } /** * Returns `true` if this option list has been marked as dirty, or if any of * this option list's widgets do not have an appearance stream. For example: * ```js * const optionList = form.getOptionList('some.optionList.field') * if (optionList.needsAppearancesUpdate()) console.log('Needs update') * ``` * @returns Whether or not this option list needs an appearance update. */ needsAppearancesUpdate(): boolean { if (this.isDirty()) return true; const widgets = this.acroField.getWidgets(); for (let idx = 0, len = widgets.length; idx < len; idx++) { const widget = widgets[idx]; const hasAppearances = widget.getAppearances()?.normal instanceof PDFStream; if (!hasAppearances) return true; } return false; } /** * Update the appearance streams for each of this option list's widgets using * the default appearance provider for option lists. For example: * ```js * const helvetica = await pdfDoc.embedFont(StandardFonts.Helvetica) * const optionList = form.getOptionList('some.optionList.field') * optionList.defaultUpdateAppearances(helvetica) * ``` * @param font The font to be used for creating the appearance streams. */ defaultUpdateAppearances(font: PDFFont) { assertIs(font, 'font', [[PDFFont, 'PDFFont']]); this.updateAppearances(font); } /** * Update the appearance streams for each of this option list's widgets using * the given appearance provider. If no `provider` is passed, the default * appearance provider for option lists will be used. For example: * ```js * const helvetica = await pdfDoc.embedFont(StandardFonts.Helvetica) * const optionList = form.getOptionList('some.optionList.field') * optionList.updateAppearances(helvetica, (field, widget, font) => { * ... * return drawOptionList(...) * }) * ``` * @param font The font to be used for creating the appearance streams. * @param provider Optionally, the appearance provider to be used for * generating the contents of the appearance streams. */ updateAppearances( font: PDFFont, provider?: AppearanceProviderFor<PDFOptionList>, ) { assertIs(font, 'font', [[PDFFont, 'PDFFont']]); assertOrUndefined(provider, 'provider', [Function]); const widgets = this.acroField.getWidgets(); for (let idx = 0, len = widgets.length; idx < len; idx++) { const widget = widgets[idx]; this.updateWidgetAppearance(widget, font, provider); } this.markAsClean(); } // getOption(index: number): string {} // getSelectedIndices(): number[] {} // removeOptions(option: string | string[]) {} // removeIndices(option: number[]) {} // deselect(options: string | string[]) {} // deselectIndices(optionIndices: number[]) {} private updateWidgetAppearance( widget: PDFWidgetAnnotation, font: PDFFont, provider?: AppearanceProviderFor<PDFOptionList>, ) { const apProvider = provider ?? defaultOptionListAppearanceProvider; const appearances = normalizeAppearance(apProvider(this, widget, font)); this.updateWidgetAppearanceWithFont(widget, font, appearances); } }