1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534 |
- /**
- * @licstart The following is the entire license notice for the
- * Javascript code in this page
- *
- * Copyright 2020 Mozilla Foundation
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- *
- * @licend The above is the entire license notice for the
- * Javascript code in this page
- */
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.getQuadPoints = getQuadPoints;
- exports.MarkupAnnotation = exports.AnnotationFactory = exports.AnnotationBorderStyle = exports.Annotation = void 0;
- var _util = require("../shared/util.js");
- var _obj = require("./obj.js");
- var _primitives = require("./primitives.js");
- var _colorspace = require("./colorspace.js");
- var _core_utils = require("./core_utils.js");
- var _operator_list = require("./operator_list.js");
- var _stream = require("./stream.js");
- var _writer = require("./writer.js");
- class AnnotationFactory {
- static create(xref, ref, pdfManager, idFactory) {
- return pdfManager.ensureCatalog("acroForm").then(acroForm => {
- return pdfManager.ensure(this, "_create", [xref, ref, pdfManager, idFactory, acroForm]);
- });
- }
- static _create(xref, ref, pdfManager, idFactory, acroForm) {
- const dict = xref.fetchIfRef(ref);
- if (!(0, _primitives.isDict)(dict)) {
- return undefined;
- }
- const id = (0, _primitives.isRef)(ref) ? ref.toString() : `annot_${idFactory.createObjId()}`;
- let subtype = dict.get("Subtype");
- subtype = (0, _primitives.isName)(subtype) ? subtype.name : null;
- const parameters = {
- xref,
- ref,
- dict,
- subtype,
- id,
- pdfManager,
- acroForm: acroForm instanceof _primitives.Dict ? acroForm : _primitives.Dict.empty
- };
- switch (subtype) {
- case "Link":
- return new LinkAnnotation(parameters);
- case "Text":
- return new TextAnnotation(parameters);
- case "Widget":
- let fieldType = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "FT"
- });
- fieldType = (0, _primitives.isName)(fieldType) ? fieldType.name : null;
- switch (fieldType) {
- case "Tx":
- return new TextWidgetAnnotation(parameters);
- case "Btn":
- return new ButtonWidgetAnnotation(parameters);
- case "Ch":
- return new ChoiceWidgetAnnotation(parameters);
- }
- (0, _util.warn)('Unimplemented widget field type "' + fieldType + '", ' + "falling back to base field type.");
- return new WidgetAnnotation(parameters);
- case "Popup":
- return new PopupAnnotation(parameters);
- case "FreeText":
- return new FreeTextAnnotation(parameters);
- case "Line":
- return new LineAnnotation(parameters);
- case "Square":
- return new SquareAnnotation(parameters);
- case "Circle":
- return new CircleAnnotation(parameters);
- case "PolyLine":
- return new PolylineAnnotation(parameters);
- case "Polygon":
- return new PolygonAnnotation(parameters);
- case "Caret":
- return new CaretAnnotation(parameters);
- case "Ink":
- return new InkAnnotation(parameters);
- case "Highlight":
- return new HighlightAnnotation(parameters);
- case "Underline":
- return new UnderlineAnnotation(parameters);
- case "Squiggly":
- return new SquigglyAnnotation(parameters);
- case "StrikeOut":
- return new StrikeOutAnnotation(parameters);
- case "Stamp":
- return new StampAnnotation(parameters);
- case "FileAttachment":
- return new FileAttachmentAnnotation(parameters);
- default:
- if (!subtype) {
- (0, _util.warn)("Annotation is missing the required /Subtype.");
- } else {
- (0, _util.warn)('Unimplemented annotation type "' + subtype + '", ' + "falling back to base annotation.");
- }
- return new Annotation(parameters);
- }
- }
- }
- exports.AnnotationFactory = AnnotationFactory;
- function getQuadPoints(dict, rect) {
- if (!dict.has("QuadPoints")) {
- return null;
- }
- const quadPoints = dict.getArray("QuadPoints");
- if (!Array.isArray(quadPoints) || quadPoints.length % 8 > 0) {
- return null;
- }
- const quadPointsLists = [];
- for (let i = 0, ii = quadPoints.length / 8; i < ii; i++) {
- quadPointsLists.push([]);
- for (let j = i * 8, jj = i * 8 + 8; j < jj; j += 2) {
- const x = quadPoints[j];
- const y = quadPoints[j + 1];
- if (x < rect[0] || x > rect[2] || y < rect[1] || y > rect[3]) {
- return null;
- }
- quadPointsLists[i].push({
- x,
- y
- });
- }
- }
- return quadPointsLists;
- }
- function getTransformMatrix(rect, bbox, matrix) {
- const [minX, minY, maxX, maxY] = _util.Util.getAxialAlignedBoundingBox(bbox, matrix);
- if (minX === maxX || minY === maxY) {
- return [1, 0, 0, 1, rect[0], rect[1]];
- }
- const xRatio = (rect[2] - rect[0]) / (maxX - minX);
- const yRatio = (rect[3] - rect[1]) / (maxY - minY);
- return [xRatio, 0, 0, yRatio, rect[0] - minX * xRatio, rect[1] - minY * yRatio];
- }
- class Annotation {
- constructor(params) {
- const dict = params.dict;
- this.setContents(dict.get("Contents"));
- this.setModificationDate(dict.get("M"));
- this.setFlags(dict.get("F"));
- this.setRectangle(dict.getArray("Rect"));
- this.setColor(dict.getArray("C"));
- this.setBorderStyle(dict);
- this.setAppearance(dict);
- this.data = {
- annotationFlags: this.flags,
- borderStyle: this.borderStyle,
- color: this.color,
- contents: this.contents,
- hasAppearance: !!this.appearance,
- id: params.id,
- modificationDate: this.modificationDate,
- rect: this.rectangle,
- subtype: params.subtype
- };
- }
- _hasFlag(flags, flag) {
- return !!(flags & flag);
- }
- _isViewable(flags) {
- return !this._hasFlag(flags, _util.AnnotationFlag.INVISIBLE) && !this._hasFlag(flags, _util.AnnotationFlag.HIDDEN) && !this._hasFlag(flags, _util.AnnotationFlag.NOVIEW);
- }
- _isPrintable(flags) {
- return this._hasFlag(flags, _util.AnnotationFlag.PRINT) && !this._hasFlag(flags, _util.AnnotationFlag.INVISIBLE) && !this._hasFlag(flags, _util.AnnotationFlag.HIDDEN);
- }
- get viewable() {
- if (this.flags === 0) {
- return true;
- }
- return this._isViewable(this.flags);
- }
- get printable() {
- if (this.flags === 0) {
- return false;
- }
- return this._isPrintable(this.flags);
- }
- setContents(contents) {
- this.contents = (0, _util.stringToPDFString)(contents || "");
- }
- setModificationDate(modificationDate) {
- this.modificationDate = (0, _util.isString)(modificationDate) ? modificationDate : null;
- }
- setFlags(flags) {
- this.flags = Number.isInteger(flags) && flags > 0 ? flags : 0;
- }
- hasFlag(flag) {
- return this._hasFlag(this.flags, flag);
- }
- setRectangle(rectangle) {
- if (Array.isArray(rectangle) && rectangle.length === 4) {
- this.rectangle = _util.Util.normalizeRect(rectangle);
- } else {
- this.rectangle = [0, 0, 0, 0];
- }
- }
- setColor(color) {
- const rgbColor = new Uint8ClampedArray(3);
- if (!Array.isArray(color)) {
- this.color = rgbColor;
- return;
- }
- switch (color.length) {
- case 0:
- this.color = null;
- break;
- case 1:
- _colorspace.ColorSpace.singletons.gray.getRgbItem(color, 0, rgbColor, 0);
- this.color = rgbColor;
- break;
- case 3:
- _colorspace.ColorSpace.singletons.rgb.getRgbItem(color, 0, rgbColor, 0);
- this.color = rgbColor;
- break;
- case 4:
- _colorspace.ColorSpace.singletons.cmyk.getRgbItem(color, 0, rgbColor, 0);
- this.color = rgbColor;
- break;
- default:
- this.color = rgbColor;
- break;
- }
- }
- setBorderStyle(borderStyle) {
- this.borderStyle = new AnnotationBorderStyle();
- if (!(0, _primitives.isDict)(borderStyle)) {
- return;
- }
- if (borderStyle.has("BS")) {
- const dict = borderStyle.get("BS");
- const dictType = dict.get("Type");
- if (!dictType || (0, _primitives.isName)(dictType, "Border")) {
- this.borderStyle.setWidth(dict.get("W"), this.rectangle);
- this.borderStyle.setStyle(dict.get("S"));
- this.borderStyle.setDashArray(dict.getArray("D"));
- }
- } else if (borderStyle.has("Border")) {
- const array = borderStyle.getArray("Border");
- if (Array.isArray(array) && array.length >= 3) {
- this.borderStyle.setHorizontalCornerRadius(array[0]);
- this.borderStyle.setVerticalCornerRadius(array[1]);
- this.borderStyle.setWidth(array[2], this.rectangle);
- if (array.length === 4) {
- this.borderStyle.setDashArray(array[3]);
- }
- }
- } else {
- this.borderStyle.setWidth(0);
- }
- }
- setAppearance(dict) {
- this.appearance = null;
- const appearanceStates = dict.get("AP");
- if (!(0, _primitives.isDict)(appearanceStates)) {
- return;
- }
- const normalAppearanceState = appearanceStates.get("N");
- if ((0, _primitives.isStream)(normalAppearanceState)) {
- this.appearance = normalAppearanceState;
- return;
- }
- if (!(0, _primitives.isDict)(normalAppearanceState)) {
- return;
- }
- const as = dict.get("AS");
- if (!(0, _primitives.isName)(as) || !normalAppearanceState.has(as.name)) {
- return;
- }
- this.appearance = normalAppearanceState.get(as.name);
- }
- loadResources(keys) {
- return this.appearance.dict.getAsync("Resources").then(resources => {
- if (!resources) {
- return undefined;
- }
- const objectLoader = new _obj.ObjectLoader(resources, keys, resources.xref);
- return objectLoader.load().then(function () {
- return resources;
- });
- });
- }
- getOperatorList(evaluator, task, renderForms, annotationStorage) {
- if (!this.appearance) {
- return Promise.resolve(new _operator_list.OperatorList());
- }
- const appearance = this.appearance;
- const data = this.data;
- const appearanceDict = appearance.dict;
- const resourcesPromise = this.loadResources(["ExtGState", "ColorSpace", "Pattern", "Shading", "XObject", "Font"]);
- const bbox = appearanceDict.getArray("BBox") || [0, 0, 1, 1];
- const matrix = appearanceDict.getArray("Matrix") || [1, 0, 0, 1, 0, 0];
- const transform = getTransformMatrix(data.rect, bbox, matrix);
- return resourcesPromise.then(resources => {
- const opList = new _operator_list.OperatorList();
- opList.addOp(_util.OPS.beginAnnotation, [data.rect, transform, matrix]);
- return evaluator.getOperatorList({
- stream: appearance,
- task,
- resources,
- operatorList: opList
- }).then(() => {
- opList.addOp(_util.OPS.endAnnotation, []);
- appearance.reset();
- return opList;
- });
- });
- }
- async save(evaluator, task, annotationStorage) {
- return null;
- }
- }
- exports.Annotation = Annotation;
- class AnnotationBorderStyle {
- constructor() {
- this.width = 1;
- this.style = _util.AnnotationBorderStyleType.SOLID;
- this.dashArray = [3];
- this.horizontalCornerRadius = 0;
- this.verticalCornerRadius = 0;
- }
- setWidth(width, rect = [0, 0, 0, 0]) {
- if ((0, _primitives.isName)(width)) {
- this.width = 0;
- return;
- }
- if (Number.isInteger(width)) {
- if (width > 0) {
- const maxWidth = (rect[2] - rect[0]) / 2;
- const maxHeight = (rect[3] - rect[1]) / 2;
- if (maxWidth > 0 && maxHeight > 0 && (width > maxWidth || width > maxHeight)) {
- (0, _util.warn)(`AnnotationBorderStyle.setWidth - ignoring width: ${width}`);
- width = 1;
- }
- }
- this.width = width;
- }
- }
- setStyle(style) {
- if (!(0, _primitives.isName)(style)) {
- return;
- }
- switch (style.name) {
- case "S":
- this.style = _util.AnnotationBorderStyleType.SOLID;
- break;
- case "D":
- this.style = _util.AnnotationBorderStyleType.DASHED;
- break;
- case "B":
- this.style = _util.AnnotationBorderStyleType.BEVELED;
- break;
- case "I":
- this.style = _util.AnnotationBorderStyleType.INSET;
- break;
- case "U":
- this.style = _util.AnnotationBorderStyleType.UNDERLINE;
- break;
- default:
- break;
- }
- }
- setDashArray(dashArray) {
- if (Array.isArray(dashArray) && dashArray.length > 0) {
- let isValid = true;
- let allZeros = true;
- for (const element of dashArray) {
- const validNumber = +element >= 0;
- if (!validNumber) {
- isValid = false;
- break;
- } else if (element > 0) {
- allZeros = false;
- }
- }
- if (isValid && !allZeros) {
- this.dashArray = dashArray;
- } else {
- this.width = 0;
- }
- } else if (dashArray) {
- this.width = 0;
- }
- }
- setHorizontalCornerRadius(radius) {
- if (Number.isInteger(radius)) {
- this.horizontalCornerRadius = radius;
- }
- }
- setVerticalCornerRadius(radius) {
- if (Number.isInteger(radius)) {
- this.verticalCornerRadius = radius;
- }
- }
- }
- exports.AnnotationBorderStyle = AnnotationBorderStyle;
- class MarkupAnnotation extends Annotation {
- constructor(parameters) {
- super(parameters);
- const dict = parameters.dict;
- if (dict.has("IRT")) {
- const rawIRT = dict.getRaw("IRT");
- this.data.inReplyTo = (0, _primitives.isRef)(rawIRT) ? rawIRT.toString() : null;
- const rt = dict.get("RT");
- this.data.replyType = (0, _primitives.isName)(rt) ? rt.name : _util.AnnotationReplyType.REPLY;
- }
- if (this.data.replyType === _util.AnnotationReplyType.GROUP) {
- const parent = dict.get("IRT");
- this.data.title = (0, _util.stringToPDFString)(parent.get("T") || "");
- this.setContents(parent.get("Contents"));
- this.data.contents = this.contents;
- if (!parent.has("CreationDate")) {
- this.data.creationDate = null;
- } else {
- this.setCreationDate(parent.get("CreationDate"));
- this.data.creationDate = this.creationDate;
- }
- if (!parent.has("M")) {
- this.data.modificationDate = null;
- } else {
- this.setModificationDate(parent.get("M"));
- this.data.modificationDate = this.modificationDate;
- }
- this.data.hasPopup = parent.has("Popup");
- if (!parent.has("C")) {
- this.data.color = null;
- } else {
- this.setColor(parent.getArray("C"));
- this.data.color = this.color;
- }
- } else {
- this.data.title = (0, _util.stringToPDFString)(dict.get("T") || "");
- this.setCreationDate(dict.get("CreationDate"));
- this.data.creationDate = this.creationDate;
- this.data.hasPopup = dict.has("Popup");
- if (!dict.has("C")) {
- this.data.color = null;
- }
- }
- }
- setCreationDate(creationDate) {
- this.creationDate = (0, _util.isString)(creationDate) ? creationDate : null;
- }
- }
- exports.MarkupAnnotation = MarkupAnnotation;
- class WidgetAnnotation extends Annotation {
- constructor(params) {
- super(params);
- const dict = params.dict;
- const data = this.data;
- this.ref = params.ref;
- data.annotationType = _util.AnnotationType.WIDGET;
- data.fieldName = this._constructFieldName(dict);
- const fieldValue = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "V",
- getArray: true
- });
- data.fieldValue = this._decodeFormValue(fieldValue);
- data.alternativeText = (0, _util.stringToPDFString)(dict.get("TU") || "");
- data.defaultAppearance = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "DA"
- }) || params.acroForm.get("DA") || "";
- const fieldType = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "FT"
- });
- data.fieldType = (0, _primitives.isName)(fieldType) ? fieldType.name : null;
- this.fieldResources = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "DR"
- }) || params.acroForm.get("DR") || _primitives.Dict.empty;
- data.fieldFlags = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "Ff"
- });
- if (!Number.isInteger(data.fieldFlags) || data.fieldFlags < 0) {
- data.fieldFlags = 0;
- }
- data.readOnly = this.hasFieldFlag(_util.AnnotationFieldFlag.READONLY);
- if (data.fieldType === "Sig") {
- data.fieldValue = null;
- this.setFlags(_util.AnnotationFlag.HIDDEN);
- }
- }
- _constructFieldName(dict) {
- if (!dict.has("T") && !dict.has("Parent")) {
- (0, _util.warn)("Unknown field name, falling back to empty field name.");
- return "";
- }
- if (!dict.has("Parent")) {
- return (0, _util.stringToPDFString)(dict.get("T"));
- }
- const fieldName = [];
- if (dict.has("T")) {
- fieldName.unshift((0, _util.stringToPDFString)(dict.get("T")));
- }
- let loopDict = dict;
- while (loopDict.has("Parent")) {
- loopDict = loopDict.get("Parent");
- if (!(0, _primitives.isDict)(loopDict)) {
- break;
- }
- if (loopDict.has("T")) {
- fieldName.unshift((0, _util.stringToPDFString)(loopDict.get("T")));
- }
- }
- return fieldName.join(".");
- }
- _decodeFormValue(formValue) {
- if (Array.isArray(formValue)) {
- return formValue.filter(item => (0, _util.isString)(item)).map(item => (0, _util.stringToPDFString)(item));
- } else if ((0, _primitives.isName)(formValue)) {
- return (0, _util.stringToPDFString)(formValue.name);
- } else if ((0, _util.isString)(formValue)) {
- return (0, _util.stringToPDFString)(formValue);
- }
- return null;
- }
- hasFieldFlag(flag) {
- return !!(this.data.fieldFlags & flag);
- }
- getOperatorList(evaluator, task, renderForms, annotationStorage) {
- if (renderForms) {
- return Promise.resolve(new _operator_list.OperatorList());
- }
- if (!this._hasText) {
- return super.getOperatorList(evaluator, task, renderForms, annotationStorage);
- }
- return this._getAppearance(evaluator, task, annotationStorage).then(content => {
- if (this.appearance && content === null) {
- return super.getOperatorList(evaluator, task, renderForms, annotationStorage);
- }
- const operatorList = new _operator_list.OperatorList();
- if (!this.data.defaultAppearance || content === null) {
- return operatorList;
- }
- const matrix = [1, 0, 0, 1, 0, 0];
- const bbox = [0, 0, this.data.rect[2] - this.data.rect[0], this.data.rect[3] - this.data.rect[1]];
- const transform = getTransformMatrix(this.data.rect, bbox, matrix);
- operatorList.addOp(_util.OPS.beginAnnotation, [this.data.rect, transform, matrix]);
- const stream = new _stream.StringStream(content);
- return evaluator.getOperatorList({
- stream,
- task,
- resources: this.fieldResources,
- operatorList
- }).then(function () {
- operatorList.addOp(_util.OPS.endAnnotation, []);
- return operatorList;
- });
- });
- }
- async save(evaluator, task, annotationStorage) {
- if (this.data.fieldValue === annotationStorage[this.data.id]) {
- return null;
- }
- let appearance = await this._getAppearance(evaluator, task, annotationStorage);
- if (appearance === null) {
- return null;
- }
- const dict = evaluator.xref.fetchIfRef(this.ref);
- if (!(0, _primitives.isDict)(dict)) {
- return null;
- }
- const bbox = [0, 0, this.data.rect[2] - this.data.rect[0], this.data.rect[3] - this.data.rect[1]];
- const newRef = evaluator.xref.getNewRef();
- const AP = new _primitives.Dict(evaluator.xref);
- AP.set("N", newRef);
- const value = annotationStorage[this.data.id];
- const encrypt = evaluator.xref.encrypt;
- let originalTransform = null;
- let newTransform = null;
- if (encrypt) {
- originalTransform = encrypt.createCipherTransform(this.ref.num, this.ref.gen);
- newTransform = encrypt.createCipherTransform(newRef.num, newRef.gen);
- appearance = newTransform.encryptString(appearance);
- }
- dict.set("V", value);
- dict.set("AP", AP);
- dict.set("M", `D:${(0, _util.getModificationDate)()}`);
- const appearanceDict = new _primitives.Dict(evaluator.xref);
- appearanceDict.set("Length", appearance.length);
- appearanceDict.set("Subtype", _primitives.Name.get("Form"));
- appearanceDict.set("Resources", this.fieldResources);
- appearanceDict.set("BBox", bbox);
- const bufferOriginal = [`${this.ref.num} ${this.ref.gen} obj\n`];
- (0, _writer.writeDict)(dict, bufferOriginal, originalTransform);
- bufferOriginal.push("\nendobj\n");
- const bufferNew = [`${newRef.num} ${newRef.gen} obj\n`];
- (0, _writer.writeDict)(appearanceDict, bufferNew, newTransform);
- bufferNew.push(" stream\n");
- bufferNew.push(appearance);
- bufferNew.push("\nendstream\nendobj\n");
- return [{
- ref: this.ref,
- data: bufferOriginal.join("")
- }, {
- ref: newRef,
- data: bufferNew.join("")
- }];
- }
- async _getAppearance(evaluator, task, annotationStorage) {
- const isPassword = this.hasFieldFlag(_util.AnnotationFieldFlag.PASSWORD);
- if (!annotationStorage || isPassword) {
- return null;
- }
- const value = annotationStorage[this.data.id];
- if (value === "") {
- return "";
- }
- const defaultPadding = 2;
- const hPadding = defaultPadding;
- const totalHeight = this.data.rect[3] - this.data.rect[1];
- const totalWidth = this.data.rect[2] - this.data.rect[0];
- const fontInfo = await this._getFontData(evaluator, task);
- const [font, fontName] = fontInfo;
- let fontSize = fontInfo[2];
- fontSize = this._computeFontSize(font, fontName, fontSize, totalHeight);
- let descent = font.descent;
- if (isNaN(descent)) {
- descent = 0;
- }
- const vPadding = defaultPadding + Math.abs(descent) * fontSize;
- const defaultAppearance = this.data.defaultAppearance;
- const alignment = this.data.textAlignment;
- if (this.data.comb) {
- return this._getCombAppearance(defaultAppearance, value, totalWidth, hPadding, vPadding);
- }
- if (this.data.multiLine) {
- return this._getMultilineAppearance(defaultAppearance, value, font, fontSize, totalWidth, totalHeight, alignment, hPadding, vPadding);
- }
- if (alignment === 0 || alignment > 2) {
- return "/Tx BMC q BT " + defaultAppearance + ` 1 0 0 1 ${hPadding} ${vPadding} Tm (${(0, _util.escapeString)(value)}) Tj` + " ET Q EMC";
- }
- const renderedText = this._renderText(value, font, fontSize, totalWidth, alignment, hPadding, vPadding);
- return "/Tx BMC q BT " + defaultAppearance + ` 1 0 0 1 0 0 Tm ${renderedText}` + " ET Q EMC";
- }
- async _getFontData(evaluator, task) {
- const operatorList = new _operator_list.OperatorList();
- const initialState = {
- fontSize: 0,
- font: null,
- fontName: null,
- clone() {
- return this;
- }
- };
- await evaluator.getOperatorList({
- stream: new _stream.StringStream(this.data.defaultAppearance),
- task,
- resources: this.fieldResources,
- operatorList,
- initialState
- });
- return [initialState.font, initialState.fontName, initialState.fontSize];
- }
- _computeFontSize(font, fontName, fontSize, height) {
- if (fontSize === null || fontSize === 0) {
- const em = font.charsToGlyphs("M", true)[0].width / 1000;
- const capHeight = 0.7 * em;
- fontSize = Math.max(1, Math.floor(height / (1.5 * capHeight)));
- let fontRegex = new RegExp(`/${fontName}\\s+[0-9\.]+\\s+Tf`);
- if (this.data.defaultAppearance.search(fontRegex) === -1) {
- fontRegex = new RegExp(`/${fontName}\\s+Tf`);
- }
- this.data.defaultAppearance = this.data.defaultAppearance.replace(fontRegex, `/${fontName} ${fontSize} Tf`);
- }
- return fontSize;
- }
- _renderText(text, font, fontSize, totalWidth, alignment, hPadding, vPadding) {
- const glyphs = font.charsToGlyphs(text);
- const scale = fontSize / 1000;
- let width = 0;
- for (const glyph of glyphs) {
- width += glyph.width * scale;
- }
- let shift;
- if (alignment === 1) {
- shift = (totalWidth - width) / 2;
- } else if (alignment === 2) {
- shift = totalWidth - width - hPadding;
- } else {
- shift = hPadding;
- }
- shift = shift.toFixed(2);
- vPadding = vPadding.toFixed(2);
- return `${shift} ${vPadding} Td (${(0, _util.escapeString)(text)}) Tj`;
- }
- }
- class TextWidgetAnnotation extends WidgetAnnotation {
- constructor(params) {
- super(params);
- this._hasText = true;
- const dict = params.dict;
- if (!(0, _util.isString)(this.data.fieldValue)) {
- this.data.fieldValue = "";
- }
- let alignment = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "Q"
- });
- if (!Number.isInteger(alignment) || alignment < 0 || alignment > 2) {
- alignment = null;
- }
- this.data.textAlignment = alignment;
- let maximumLength = (0, _core_utils.getInheritableProperty)({
- dict,
- key: "MaxLen"
- });
- if (!Number.isInteger(maximumLength) || maximumLength < 0) {
- maximumLength = null;
- }
- this.data.maxLen = maximumLength;
- this.data.multiLine = this.hasFieldFlag(_util.AnnotationFieldFlag.MULTILINE);
- this.data.comb = this.hasFieldFlag(_util.AnnotationFieldFlag.COMB) && !this.hasFieldFlag(_util.AnnotationFieldFlag.MULTILINE) && !this.hasFieldFlag(_util.AnnotationFieldFlag.PASSWORD) && !this.hasFieldFlag(_util.AnnotationFieldFlag.FILESELECT) && this.data.maxLen !== null;
- }
- _getCombAppearance(defaultAppearance, text, width, hPadding, vPadding) {
- const combWidth = (width / this.data.maxLen).toFixed(2);
- const buf = [];
- for (const character of text) {
- buf.push(`(${(0, _util.escapeString)(character)}) Tj`);
- }
- const renderedComb = buf.join(` ${combWidth} 0 Td `);
- return "/Tx BMC q BT " + defaultAppearance + ` 1 0 0 1 ${hPadding} ${vPadding} Tm ${renderedComb}` + " ET Q EMC";
- }
- _getMultilineAppearance(defaultAppearance, text, font, fontSize, width, height, alignment, hPadding, vPadding) {
- const lines = text.split(/\r\n|\r|\n/);
- const buf = [];
- const totalWidth = width - 2 * hPadding;
- for (const line of lines) {
- const chunks = this._splitLine(line, font, fontSize, totalWidth);
- for (const chunk of chunks) {
- const padding = buf.length === 0 ? hPadding : 0;
- buf.push(this._renderText(chunk, font, fontSize, width, alignment, padding, -fontSize));
- }
- }
- const renderedText = buf.join("\n");
- return "/Tx BMC q BT " + defaultAppearance + ` 1 0 0 1 0 ${height} Tm ${renderedText}` + " ET Q EMC";
- }
- _splitLine(line, font, fontSize, width) {
- if (line.length <= 1) {
- return [line];
- }
- const scale = fontSize / 1000;
- const whitespace = font.charsToGlyphs(" ", true)[0].width * scale;
- const chunks = [];
- let lastSpacePos = -1,
- startChunk = 0,
- currentWidth = 0;
- for (let i = 0, ii = line.length; i < ii; i++) {
- const character = line.charAt(i);
- if (character === " ") {
- if (currentWidth + whitespace > width) {
- chunks.push(line.substring(startChunk, i));
- startChunk = i;
- currentWidth = whitespace;
- lastSpacePos = -1;
- } else {
- currentWidth += whitespace;
- lastSpacePos = i;
- }
- } else {
- const charWidth = font.charsToGlyphs(character, false)[0].width * scale;
- if (currentWidth + charWidth > width) {
- if (lastSpacePos !== -1) {
- chunks.push(line.substring(startChunk, lastSpacePos + 1));
- startChunk = i = lastSpacePos + 1;
- lastSpacePos = -1;
- currentWidth = 0;
- } else {
- chunks.push(line.substring(startChunk, i));
- startChunk = i;
- currentWidth = charWidth;
- }
- } else {
- currentWidth += charWidth;
- }
- }
- }
- if (startChunk < line.length) {
- chunks.push(line.substring(startChunk, line.length));
- }
- return chunks;
- }
- }
- class ButtonWidgetAnnotation extends WidgetAnnotation {
- constructor(params) {
- super(params);
- this.checkedAppearance = null;
- this.uncheckedAppearance = null;
- this.data.checkBox = !this.hasFieldFlag(_util.AnnotationFieldFlag.RADIO) && !this.hasFieldFlag(_util.AnnotationFieldFlag.PUSHBUTTON);
- this.data.radioButton = this.hasFieldFlag(_util.AnnotationFieldFlag.RADIO) && !this.hasFieldFlag(_util.AnnotationFieldFlag.PUSHBUTTON);
- this.data.pushButton = this.hasFieldFlag(_util.AnnotationFieldFlag.PUSHBUTTON);
- if (this.data.checkBox) {
- this._processCheckBox(params);
- } else if (this.data.radioButton) {
- this._processRadioButton(params);
- } else if (this.data.pushButton) {
- this._processPushButton(params);
- } else {
- (0, _util.warn)("Invalid field flags for button widget annotation");
- }
- }
- getOperatorList(evaluator, task, renderForms, annotationStorage) {
- if (this.data.pushButton) {
- return super.getOperatorList(evaluator, task, false, annotationStorage);
- }
- if (annotationStorage) {
- const value = annotationStorage[this.data.id] || false;
- let appearance;
- if (value) {
- appearance = this.checkedAppearance;
- } else {
- appearance = this.uncheckedAppearance;
- }
- if (appearance) {
- const savedAppearance = this.appearance;
- this.appearance = appearance;
- const operatorList = super.getOperatorList(evaluator, task, renderForms, annotationStorage);
- this.appearance = savedAppearance;
- return operatorList;
- }
- return Promise.resolve(new _operator_list.OperatorList());
- }
- return super.getOperatorList(evaluator, task, renderForms, annotationStorage);
- }
- async save(evaluator, task, annotationStorage) {
- if (this.data.checkBox) {
- return this._saveCheckbox(evaluator, task, annotationStorage);
- }
- if (this.data.radioButton) {
- return this._saveRadioButton(evaluator, task, annotationStorage);
- }
- return super.save(evaluator, task, annotationStorage);
- }
- async _saveCheckbox(evaluator, task, annotationStorage) {
- const defaultValue = this.data.fieldValue && this.data.fieldValue !== "Off";
- const value = annotationStorage[this.data.id];
- if (defaultValue === value) {
- return null;
- }
- const dict = evaluator.xref.fetchIfRef(this.ref);
- if (!(0, _primitives.isDict)(dict)) {
- return null;
- }
- const name = _primitives.Name.get(value ? this.data.exportValue : "Off");
- dict.set("V", name);
- dict.set("AS", name);
- dict.set("M", `D:${(0, _util.getModificationDate)()}`);
- const encrypt = evaluator.xref.encrypt;
- let originalTransform = null;
- if (encrypt) {
- originalTransform = encrypt.createCipherTransform(this.ref.num, this.ref.gen);
- }
- const buffer = [`${this.ref.num} ${this.ref.gen} obj\n`];
- (0, _writer.writeDict)(dict, buffer, originalTransform);
- buffer.push("\nendobj\n");
- return [{
- ref: this.ref,
- data: buffer.join("")
- }];
- }
- async _saveRadioButton(evaluator, task, annotationStorage) {
- const defaultValue = this.data.fieldValue === this.data.buttonValue;
- const value = annotationStorage[this.data.id];
- if (defaultValue === value) {
- return null;
- }
- const dict = evaluator.xref.fetchIfRef(this.ref);
- if (!(0, _primitives.isDict)(dict)) {
- return null;
- }
- const name = _primitives.Name.get(value ? this.data.buttonValue : "Off");
- let parentBuffer = null;
- const encrypt = evaluator.xref.encrypt;
- if (value) {
- if ((0, _primitives.isRef)(this.parent)) {
- const parent = evaluator.xref.fetch(this.parent);
- let parentTransform = null;
- if (encrypt) {
- parentTransform = encrypt.createCipherTransform(this.parent.num, this.parent.gen);
- }
- parent.set("V", name);
- parentBuffer = [`${this.parent.num} ${this.parent.gen} obj\n`];
- (0, _writer.writeDict)(parent, parentBuffer, parentTransform);
- parentBuffer.push("\nendobj\n");
- } else if ((0, _primitives.isDict)(this.parent)) {
- this.parent.set("V", name);
- }
- }
- dict.set("AS", name);
- dict.set("M", `D:${(0, _util.getModificationDate)()}`);
- let originalTransform = null;
- if (encrypt) {
- originalTransform = encrypt.createCipherTransform(this.ref.num, this.ref.gen);
- }
- const buffer = [`${this.ref.num} ${this.ref.gen} obj\n`];
- (0, _writer.writeDict)(dict, buffer, originalTransform);
- buffer.push("\nendobj\n");
- const newRefs = [{
- ref: this.ref,
- data: buffer.join("")
- }];
- if (parentBuffer !== null) {
- newRefs.push({
- ref: this.parent,
- data: parentBuffer.join("")
- });
- }
- return newRefs;
- }
- _processCheckBox(params) {
- const customAppearance = params.dict.get("AP");
- if (!(0, _primitives.isDict)(customAppearance)) {
- return;
- }
- const normalAppearance = customAppearance.get("N");
- if (!(0, _primitives.isDict)(normalAppearance)) {
- return;
- }
- const exportValues = normalAppearance.getKeys();
- if (!exportValues.includes("Off")) {
- exportValues.push("Off");
- }
- if (exportValues.length !== 2) {
- return;
- }
- this.data.exportValue = exportValues[0] === "Off" ? exportValues[1] : exportValues[0];
- this.checkedAppearance = normalAppearance.get(this.data.exportValue);
- this.uncheckedAppearance = normalAppearance.get("Off") || null;
- }
- _processRadioButton(params) {
- this.data.fieldValue = this.data.buttonValue = null;
- const fieldParent = params.dict.get("Parent");
- if ((0, _primitives.isDict)(fieldParent) && fieldParent.has("V")) {
- const fieldParentValue = fieldParent.get("V");
- if ((0, _primitives.isName)(fieldParentValue)) {
- this.parent = params.dict.getRaw("Parent");
- this.data.fieldValue = this._decodeFormValue(fieldParentValue);
- }
- }
- const appearanceStates = params.dict.get("AP");
- if (!(0, _primitives.isDict)(appearanceStates)) {
- return;
- }
- const normalAppearance = appearanceStates.get("N");
- if (!(0, _primitives.isDict)(normalAppearance)) {
- return;
- }
- for (const key of normalAppearance.getKeys()) {
- if (key !== "Off") {
- this.data.buttonValue = key;
- break;
- }
- }
- this.checkedAppearance = normalAppearance.get(this.data.buttonValue);
- this.uncheckedAppearance = normalAppearance.get("Off") || null;
- }
- _processPushButton(params) {
- if (!params.dict.has("A")) {
- (0, _util.warn)("Push buttons without action dictionaries are not supported");
- return;
- }
- _obj.Catalog.parseDestDictionary({
- destDict: params.dict,
- resultObj: this.data,
- docBaseUrl: params.pdfManager.docBaseUrl
- });
- }
- }
- class ChoiceWidgetAnnotation extends WidgetAnnotation {
- constructor(params) {
- super(params);
- this.data.options = [];
- const options = (0, _core_utils.getInheritableProperty)({
- dict: params.dict,
- key: "Opt"
- });
- if (Array.isArray(options)) {
- const xref = params.xref;
- for (let i = 0, ii = options.length; i < ii; i++) {
- const option = xref.fetchIfRef(options[i]);
- const isOptionArray = Array.isArray(option);
- this.data.options[i] = {
- exportValue: this._decodeFormValue(isOptionArray ? xref.fetchIfRef(option[0]) : option),
- displayValue: this._decodeFormValue(isOptionArray ? xref.fetchIfRef(option[1]) : option)
- };
- }
- }
- if ((0, _util.isString)(this.data.fieldValue)) {
- this.data.fieldValue = [this.data.fieldValue];
- } else if (!this.data.fieldValue) {
- this.data.fieldValue = [];
- }
- this.data.combo = this.hasFieldFlag(_util.AnnotationFieldFlag.COMBO);
- this.data.multiSelect = this.hasFieldFlag(_util.AnnotationFieldFlag.MULTISELECT);
- this._hasText = true;
- }
- }
- class TextAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- const DEFAULT_ICON_SIZE = 22;
- super(parameters);
- const dict = parameters.dict;
- this.data.annotationType = _util.AnnotationType.TEXT;
- if (this.data.hasAppearance) {
- this.data.name = "NoIcon";
- } else {
- this.data.rect[1] = this.data.rect[3] - DEFAULT_ICON_SIZE;
- this.data.rect[2] = this.data.rect[0] + DEFAULT_ICON_SIZE;
- this.data.name = dict.has("Name") ? dict.get("Name").name : "Note";
- }
- if (dict.has("State")) {
- this.data.state = dict.get("State") || null;
- this.data.stateModel = dict.get("StateModel") || null;
- } else {
- this.data.state = null;
- this.data.stateModel = null;
- }
- }
- }
- class LinkAnnotation extends Annotation {
- constructor(params) {
- super(params);
- this.data.annotationType = _util.AnnotationType.LINK;
- const quadPoints = getQuadPoints(params.dict, this.rectangle);
- if (quadPoints) {
- this.data.quadPoints = quadPoints;
- }
- _obj.Catalog.parseDestDictionary({
- destDict: params.dict,
- resultObj: this.data,
- docBaseUrl: params.pdfManager.docBaseUrl
- });
- }
- }
- class PopupAnnotation extends Annotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.POPUP;
- let parentItem = parameters.dict.get("Parent");
- if (!parentItem) {
- (0, _util.warn)("Popup annotation has a missing or invalid parent annotation.");
- return;
- }
- const parentSubtype = parentItem.get("Subtype");
- this.data.parentType = (0, _primitives.isName)(parentSubtype) ? parentSubtype.name : null;
- const rawParent = parameters.dict.getRaw("Parent");
- this.data.parentId = (0, _primitives.isRef)(rawParent) ? rawParent.toString() : null;
- const rt = parentItem.get("RT");
- if ((0, _primitives.isName)(rt, _util.AnnotationReplyType.GROUP)) {
- parentItem = parentItem.get("IRT");
- }
- if (!parentItem.has("M")) {
- this.data.modificationDate = null;
- } else {
- this.setModificationDate(parentItem.get("M"));
- this.data.modificationDate = this.modificationDate;
- }
- if (!parentItem.has("C")) {
- this.data.color = null;
- } else {
- this.setColor(parentItem.getArray("C"));
- this.data.color = this.color;
- }
- if (!this.viewable) {
- const parentFlags = parentItem.get("F");
- if (this._isViewable(parentFlags)) {
- this.setFlags(parentFlags);
- }
- }
- this.data.title = (0, _util.stringToPDFString)(parentItem.get("T") || "");
- this.data.contents = (0, _util.stringToPDFString)(parentItem.get("Contents") || "");
- }
- }
- class FreeTextAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.FREETEXT;
- }
- }
- class LineAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.LINE;
- this.data.lineCoordinates = _util.Util.normalizeRect(parameters.dict.getArray("L"));
- }
- }
- class SquareAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.SQUARE;
- }
- }
- class CircleAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.CIRCLE;
- }
- }
- class PolylineAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.POLYLINE;
- const rawVertices = parameters.dict.getArray("Vertices");
- this.data.vertices = [];
- for (let i = 0, ii = rawVertices.length; i < ii; i += 2) {
- this.data.vertices.push({
- x: rawVertices[i],
- y: rawVertices[i + 1]
- });
- }
- }
- }
- class PolygonAnnotation extends PolylineAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.POLYGON;
- }
- }
- class CaretAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.CARET;
- }
- }
- class InkAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.INK;
- const xref = parameters.xref;
- const originalInkLists = parameters.dict.getArray("InkList");
- this.data.inkLists = [];
- for (let i = 0, ii = originalInkLists.length; i < ii; ++i) {
- this.data.inkLists.push([]);
- for (let j = 0, jj = originalInkLists[i].length; j < jj; j += 2) {
- this.data.inkLists[i].push({
- x: xref.fetchIfRef(originalInkLists[i][j]),
- y: xref.fetchIfRef(originalInkLists[i][j + 1])
- });
- }
- }
- }
- }
- class HighlightAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.HIGHLIGHT;
- const quadPoints = getQuadPoints(parameters.dict, this.rectangle);
- if (quadPoints) {
- this.data.quadPoints = quadPoints;
- }
- }
- }
- class UnderlineAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.UNDERLINE;
- const quadPoints = getQuadPoints(parameters.dict, this.rectangle);
- if (quadPoints) {
- this.data.quadPoints = quadPoints;
- }
- }
- }
- class SquigglyAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.SQUIGGLY;
- const quadPoints = getQuadPoints(parameters.dict, this.rectangle);
- if (quadPoints) {
- this.data.quadPoints = quadPoints;
- }
- }
- }
- class StrikeOutAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.STRIKEOUT;
- const quadPoints = getQuadPoints(parameters.dict, this.rectangle);
- if (quadPoints) {
- this.data.quadPoints = quadPoints;
- }
- }
- }
- class StampAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- this.data.annotationType = _util.AnnotationType.STAMP;
- }
- }
- class FileAttachmentAnnotation extends MarkupAnnotation {
- constructor(parameters) {
- super(parameters);
- const file = new _obj.FileSpec(parameters.dict.get("FS"), parameters.xref);
- this.data.annotationType = _util.AnnotationType.FILEATTACHMENT;
- this.data.file = file.serializable;
- }
- }
|