cfb908c999
On my computer, it takes few tenths of a second to load a local font. Since a font can be used several times in a document, the cache will improve performances.
4650 lines
150 KiB
JavaScript
4650 lines
150 KiB
JavaScript
/* Copyright 2017 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.
|
||
*/
|
||
|
||
import {
|
||
Annotation,
|
||
AnnotationBorderStyle,
|
||
AnnotationFactory,
|
||
getQuadPoints,
|
||
MarkupAnnotation,
|
||
} from "../../src/core/annotation.js";
|
||
import {
|
||
AnnotationBorderStyleType,
|
||
AnnotationEditorType,
|
||
AnnotationFieldFlag,
|
||
AnnotationFlag,
|
||
AnnotationType,
|
||
OPS,
|
||
RenderingIntentFlag,
|
||
stringToBytes,
|
||
stringToUTF8String,
|
||
} from "../../src/shared/util.js";
|
||
import {
|
||
CMAP_URL,
|
||
createIdFactory,
|
||
STANDARD_FONT_DATA_URL,
|
||
XRefMock,
|
||
} from "./test_utils.js";
|
||
import {
|
||
DefaultCMapReaderFactory,
|
||
DefaultStandardFontDataFactory,
|
||
} from "../../src/display/api.js";
|
||
import { Dict, Name, Ref, RefSetCache } from "../../src/core/primitives.js";
|
||
import { Lexer, Parser } from "../../src/core/parser.js";
|
||
import { PartialEvaluator } from "../../src/core/evaluator.js";
|
||
import { StringStream } from "../../src/core/stream.js";
|
||
import { WorkerTask } from "../../src/core/worker.js";
|
||
|
||
describe("annotation", function () {
|
||
class PDFManagerMock {
|
||
constructor(params) {
|
||
this.docBaseUrl = params.docBaseUrl || null;
|
||
this.pdfDocument = {
|
||
catalog: {
|
||
acroForm: new Dict(),
|
||
},
|
||
};
|
||
this.evaluatorOptions = {
|
||
isOffscreenCanvasSupported: false,
|
||
};
|
||
}
|
||
|
||
ensure(obj, prop, args) {
|
||
return new Promise(function (resolve) {
|
||
const value = obj[prop];
|
||
if (typeof value === "function") {
|
||
resolve(value.apply(obj, args));
|
||
} else {
|
||
resolve(value);
|
||
}
|
||
});
|
||
}
|
||
|
||
ensureCatalog(prop, args) {
|
||
return this.ensure(this.pdfDocument.catalog, prop, args);
|
||
}
|
||
|
||
ensureDoc(prop, args) {
|
||
return this.ensure(this.pdfDocument, prop, args);
|
||
}
|
||
}
|
||
|
||
const fontDataReader = new DefaultStandardFontDataFactory({
|
||
baseUrl: STANDARD_FONT_DATA_URL,
|
||
});
|
||
|
||
function HandlerMock() {
|
||
this.inputs = [];
|
||
}
|
||
HandlerMock.prototype = {
|
||
send(name, data) {
|
||
this.inputs.push({ name, data });
|
||
},
|
||
sendWithPromise(name, data) {
|
||
if (name !== "FetchStandardFontData") {
|
||
return Promise.reject(new Error(`Unsupported mock ${name}.`));
|
||
}
|
||
return fontDataReader.fetch(data);
|
||
},
|
||
};
|
||
|
||
let pdfManagerMock, idFactoryMock, partialEvaluator;
|
||
|
||
beforeAll(async function () {
|
||
pdfManagerMock = new PDFManagerMock({
|
||
docBaseUrl: null,
|
||
});
|
||
|
||
const CMapReaderFactory = new DefaultCMapReaderFactory({
|
||
baseUrl: CMAP_URL,
|
||
});
|
||
|
||
const builtInCMapCache = new Map();
|
||
builtInCMapCache.set(
|
||
"UniJIS-UTF16-H",
|
||
await CMapReaderFactory.fetch({ name: "UniJIS-UTF16-H" })
|
||
);
|
||
builtInCMapCache.set(
|
||
"Adobe-Japan1-UCS2",
|
||
await CMapReaderFactory.fetch({ name: "Adobe-Japan1-UCS2" })
|
||
);
|
||
|
||
idFactoryMock = createIdFactory(/* pageIndex = */ 0);
|
||
partialEvaluator = new PartialEvaluator({
|
||
xref: new XRefMock(),
|
||
handler: new HandlerMock(),
|
||
pageIndex: 0,
|
||
idFactory: createIdFactory(/* pageIndex = */ 0),
|
||
fontCache: new RefSetCache(),
|
||
builtInCMapCache,
|
||
standardFontDataCache: new Map(),
|
||
systemFontCache: new Map(),
|
||
});
|
||
});
|
||
|
||
afterAll(function () {
|
||
pdfManagerMock = null;
|
||
idFactoryMock = null;
|
||
partialEvaluator = null;
|
||
});
|
||
|
||
describe("AnnotationFactory", function () {
|
||
it("should get id for annotation", async function () {
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
|
||
const annotationRef = Ref.get(10, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.id).toEqual("10R");
|
||
});
|
||
|
||
it(
|
||
"should handle, and get fallback IDs for, annotations that are not " +
|
||
"indirect objects (issue 7569)",
|
||
async function () {
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
|
||
const xref = new XRefMock();
|
||
const idFactory = createIdFactory(/* pageIndex = */ 0);
|
||
|
||
const annotation1 = AnnotationFactory.create(
|
||
xref,
|
||
annotationDict,
|
||
pdfManagerMock,
|
||
idFactory
|
||
).then(({ data }) => {
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.id).toEqual("annot_p0_1");
|
||
});
|
||
|
||
const annotation2 = AnnotationFactory.create(
|
||
xref,
|
||
annotationDict,
|
||
pdfManagerMock,
|
||
idFactory
|
||
).then(({ data }) => {
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.id).toEqual("annot_p0_2");
|
||
});
|
||
|
||
await Promise.all([annotation1, annotation2]);
|
||
}
|
||
);
|
||
|
||
it("should handle missing /Subtype", async function () {
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
|
||
const annotationRef = Ref.get(1, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toBeUndefined();
|
||
});
|
||
});
|
||
|
||
describe("getQuadPoints", function () {
|
||
let dict, rect;
|
||
|
||
beforeEach(function () {
|
||
dict = new Dict();
|
||
rect = [];
|
||
});
|
||
|
||
afterEach(function () {
|
||
dict = null;
|
||
rect = null;
|
||
});
|
||
|
||
it("should ignore missing quadpoints", function () {
|
||
expect(getQuadPoints(dict, rect)).toEqual(null);
|
||
});
|
||
|
||
it("should ignore non-array values", function () {
|
||
dict.set("QuadPoints", "foo");
|
||
expect(getQuadPoints(dict, rect)).toEqual(null);
|
||
});
|
||
|
||
it("should ignore arrays where the length is not a multiple of eight", function () {
|
||
dict.set("QuadPoints", [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
|
||
expect(getQuadPoints(dict, rect)).toEqual(null);
|
||
});
|
||
|
||
it("should ignore quadpoints if one coordinate lies outside the rectangle", function () {
|
||
rect = [10, 10, 20, 20];
|
||
const inputs = [
|
||
[11, 11, 12, 12, 9, 13, 14, 14], // Smaller than lower x coordinate.
|
||
[11, 11, 12, 12, 13, 9, 14, 14], // Smaller than lower y coordinate.
|
||
[11, 11, 12, 12, 21, 13, 14, 14], // Larger than upper x coordinate.
|
||
[11, 11, 12, 12, 13, 21, 14, 14], // Larger than upper y coordinate.
|
||
];
|
||
for (const input of inputs) {
|
||
dict.set("QuadPoints", input);
|
||
expect(getQuadPoints(dict, rect)).toEqual(null);
|
||
}
|
||
});
|
||
|
||
it("should process quadpoints in the standard order", function () {
|
||
rect = [10, 10, 20, 20];
|
||
dict.set(
|
||
"QuadPoints",
|
||
[10, 20, 20, 20, 10, 10, 20, 10, 11, 19, 19, 19, 11, 11, 19, 11]
|
||
);
|
||
expect(getQuadPoints(dict, rect)).toEqual([
|
||
[
|
||
{ x: 10, y: 20 },
|
||
{ x: 20, y: 20 },
|
||
{ x: 10, y: 10 },
|
||
{ x: 20, y: 10 },
|
||
],
|
||
[
|
||
{ x: 11, y: 19 },
|
||
{ x: 19, y: 19 },
|
||
{ x: 11, y: 11 },
|
||
{ x: 19, y: 11 },
|
||
],
|
||
]);
|
||
});
|
||
|
||
it("should normalize and process quadpoints in non-standard orders", function () {
|
||
rect = [10, 10, 20, 20];
|
||
const nonStandardOrders = [
|
||
// Bottom left, bottom right, top right and top left.
|
||
[10, 20, 20, 20, 20, 10, 10, 10],
|
||
|
||
// Top left, top right, bottom left and bottom right.
|
||
[10, 10, 20, 10, 10, 20, 20, 20],
|
||
|
||
// Top left, top right, bottom right and bottom left.
|
||
[10, 10, 20, 10, 20, 20, 10, 20],
|
||
];
|
||
|
||
for (const nonStandardOrder of nonStandardOrders) {
|
||
dict.set("QuadPoints", nonStandardOrder);
|
||
expect(getQuadPoints(dict, rect)).toEqual([
|
||
[
|
||
{ x: 10, y: 20 },
|
||
{ x: 20, y: 20 },
|
||
{ x: 10, y: 10 },
|
||
{ x: 20, y: 10 },
|
||
],
|
||
]);
|
||
}
|
||
});
|
||
});
|
||
|
||
describe("Annotation", function () {
|
||
let dict, ref;
|
||
|
||
beforeAll(function () {
|
||
dict = new Dict();
|
||
ref = Ref.get(1, 0);
|
||
});
|
||
|
||
afterAll(function () {
|
||
dict = ref = null;
|
||
});
|
||
|
||
it("should set and get valid contents", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setContents("Foo bar baz");
|
||
|
||
expect(annotation._contents).toEqual({ str: "Foo bar baz", dir: "ltr" });
|
||
});
|
||
|
||
it("should not set and get invalid contents", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setContents(undefined);
|
||
|
||
expect(annotation._contents).toEqual({ str: "", dir: "ltr" });
|
||
});
|
||
|
||
it("should set and get a valid modification date", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setModificationDate("D:20190422");
|
||
|
||
expect(annotation.modificationDate).toEqual("D:20190422");
|
||
});
|
||
|
||
it("should not set and get an invalid modification date", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setModificationDate(undefined);
|
||
|
||
expect(annotation.modificationDate).toEqual(null);
|
||
});
|
||
|
||
it("should set and get flags", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setFlags(13);
|
||
|
||
expect(annotation.hasFlag(AnnotationFlag.INVISIBLE)).toEqual(true);
|
||
expect(annotation.hasFlag(AnnotationFlag.NOZOOM)).toEqual(true);
|
||
expect(annotation.hasFlag(AnnotationFlag.PRINT)).toEqual(true);
|
||
expect(annotation.hasFlag(AnnotationFlag.READONLY)).toEqual(false);
|
||
expect(annotation.hasFlag(AnnotationFlag.HIDDEN)).toEqual(false);
|
||
});
|
||
|
||
it("should be viewable and not printable by default", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
|
||
expect(annotation.viewable).toEqual(true);
|
||
expect(annotation.printable).toEqual(false);
|
||
});
|
||
|
||
it("should set and get a valid rectangle", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setRectangle([117, 694, 164.298, 720]);
|
||
|
||
expect(annotation.rectangle).toEqual([117, 694, 164.298, 720]);
|
||
});
|
||
|
||
it("should not set and get an invalid rectangle", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setRectangle([117, 694, 164.298]);
|
||
|
||
expect(annotation.rectangle).toEqual([0, 0, 0, 0]);
|
||
});
|
||
|
||
it("should reject a color if it is not an array", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setColor("red");
|
||
|
||
expect(annotation.color).toEqual(new Uint8ClampedArray([0, 0, 0]));
|
||
});
|
||
|
||
it("should set and get a transparent color", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setColor([]);
|
||
|
||
expect(annotation.color).toEqual(null);
|
||
});
|
||
|
||
it("should set and get a grayscale color", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setColor([0.4]);
|
||
|
||
expect(annotation.color).toEqual(new Uint8ClampedArray([102, 102, 102]));
|
||
});
|
||
|
||
it("should set and get an RGB color", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setColor([0, 0, 1]);
|
||
|
||
expect(annotation.color).toEqual(new Uint8ClampedArray([0, 0, 255]));
|
||
});
|
||
|
||
it("should set and get a CMYK color", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setColor([0.1, 0.92, 0.84, 0.02]);
|
||
|
||
expect(annotation.color).toEqual(new Uint8ClampedArray([234, 59, 48]));
|
||
});
|
||
|
||
it("should not set and get an invalid color", function () {
|
||
const annotation = new Annotation({ dict, ref });
|
||
annotation.setColor([0.4, 0.6]);
|
||
|
||
expect(annotation.color).toEqual(new Uint8ClampedArray([0, 0, 0]));
|
||
});
|
||
});
|
||
|
||
describe("AnnotationBorderStyle", function () {
|
||
it("should set and get a valid width", function () {
|
||
const borderStyleInt = new AnnotationBorderStyle();
|
||
borderStyleInt.setWidth(3);
|
||
const borderStyleNum = new AnnotationBorderStyle();
|
||
borderStyleNum.setWidth(2.5);
|
||
|
||
expect(borderStyleInt.width).toEqual(3);
|
||
expect(borderStyleNum.width).toEqual(2.5);
|
||
});
|
||
|
||
it("should not set and get an invalid width", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setWidth("three");
|
||
|
||
expect(borderStyle.width).toEqual(1);
|
||
});
|
||
|
||
it("should set the width to zero, when the input is a `Name` (issue 10385)", function () {
|
||
const borderStyleZero = new AnnotationBorderStyle();
|
||
borderStyleZero.setWidth(Name.get("0"));
|
||
const borderStyleFive = new AnnotationBorderStyle();
|
||
borderStyleFive.setWidth(Name.get("5"));
|
||
|
||
expect(borderStyleZero.width).toEqual(0);
|
||
expect(borderStyleFive.width).toEqual(0);
|
||
});
|
||
|
||
it("should set and get a valid style", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setStyle(Name.get("D"));
|
||
|
||
expect(borderStyle.style).toEqual(AnnotationBorderStyleType.DASHED);
|
||
});
|
||
|
||
it("should not set and get an invalid style", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setStyle("Dashed");
|
||
|
||
expect(borderStyle.style).toEqual(AnnotationBorderStyleType.SOLID);
|
||
});
|
||
|
||
it("should set and get a valid dash array", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setDashArray([1, 2, 3]);
|
||
|
||
expect(borderStyle.dashArray).toEqual([1, 2, 3]);
|
||
});
|
||
|
||
it("should not set and get an invalid dash array", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setDashArray([0, 0]);
|
||
|
||
expect(borderStyle.dashArray).toEqual([3]);
|
||
});
|
||
|
||
it("should set and get a valid horizontal corner radius", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setHorizontalCornerRadius(3);
|
||
|
||
expect(borderStyle.horizontalCornerRadius).toEqual(3);
|
||
});
|
||
|
||
it("should not set and get an invalid horizontal corner radius", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setHorizontalCornerRadius("three");
|
||
|
||
expect(borderStyle.horizontalCornerRadius).toEqual(0);
|
||
});
|
||
|
||
it("should set and get a valid vertical corner radius", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setVerticalCornerRadius(3);
|
||
|
||
expect(borderStyle.verticalCornerRadius).toEqual(3);
|
||
});
|
||
|
||
it("should not set and get an invalid vertical corner radius", function () {
|
||
const borderStyle = new AnnotationBorderStyle();
|
||
borderStyle.setVerticalCornerRadius("three");
|
||
|
||
expect(borderStyle.verticalCornerRadius).toEqual(0);
|
||
});
|
||
});
|
||
|
||
describe("MarkupAnnotation", function () {
|
||
let dict, ref;
|
||
|
||
beforeAll(function () {
|
||
dict = new Dict();
|
||
ref = Ref.get(1, 0);
|
||
});
|
||
|
||
afterAll(function () {
|
||
dict = ref = null;
|
||
});
|
||
|
||
it("should set and get a valid creation date", function () {
|
||
const markupAnnotation = new MarkupAnnotation({ dict, ref });
|
||
markupAnnotation.setCreationDate("D:20190422");
|
||
|
||
expect(markupAnnotation.creationDate).toEqual("D:20190422");
|
||
});
|
||
|
||
it("should not set and get an invalid creation date", function () {
|
||
const markupAnnotation = new MarkupAnnotation({ dict, ref });
|
||
markupAnnotation.setCreationDate(undefined);
|
||
|
||
expect(markupAnnotation.creationDate).toEqual(null);
|
||
});
|
||
|
||
it("should not parse IRT/RT when not defined", async function () {
|
||
dict.set("Type", Name.get("Annot"));
|
||
dict.set("Subtype", Name.get("Text"));
|
||
|
||
const xref = new XRefMock([{ ref, data: dict }]);
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
ref,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.inReplyTo).toBeUndefined();
|
||
expect(data.replyType).toBeUndefined();
|
||
});
|
||
|
||
it("should parse IRT and set default RT when not defined", async function () {
|
||
const annotationRef = Ref.get(819, 0);
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Text"));
|
||
|
||
const replyRef = Ref.get(820, 0);
|
||
const replyDict = new Dict();
|
||
replyDict.set("Type", Name.get("Annot"));
|
||
replyDict.set("Subtype", Name.get("Text"));
|
||
replyDict.set("IRT", annotationRef);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
{ ref: replyRef, data: replyDict },
|
||
]);
|
||
annotationDict.assignXref(xref);
|
||
replyDict.assignXref(xref);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
replyRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.inReplyTo).toEqual(annotationRef.toString());
|
||
expect(data.replyType).toEqual("R");
|
||
});
|
||
|
||
it("should parse IRT/RT for a group type", async function () {
|
||
const annotationRef = Ref.get(819, 0);
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Text"));
|
||
annotationDict.set("T", "ParentTitle");
|
||
annotationDict.set("Contents", "ParentText");
|
||
annotationDict.set("CreationDate", "D:20180423");
|
||
annotationDict.set("M", "D:20190423");
|
||
annotationDict.set("C", [0, 0, 1]);
|
||
|
||
const popupRef = Ref.get(820, 0);
|
||
const popupDict = new Dict();
|
||
popupDict.set("Type", Name.get("Annot"));
|
||
popupDict.set("Subtype", Name.get("Popup"));
|
||
popupDict.set("Parent", annotationRef);
|
||
annotationDict.set("Popup", popupRef);
|
||
|
||
const replyRef = Ref.get(821, 0);
|
||
const replyDict = new Dict();
|
||
replyDict.set("Type", Name.get("Annot"));
|
||
replyDict.set("Subtype", Name.get("Text"));
|
||
replyDict.set("IRT", annotationRef);
|
||
replyDict.set("RT", Name.get("Group"));
|
||
replyDict.set("T", "ReplyTitle");
|
||
replyDict.set("Contents", "ReplyText");
|
||
replyDict.set("CreationDate", "D:20180523");
|
||
replyDict.set("M", "D:20190523");
|
||
replyDict.set("C", [0.4]);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
{ ref: popupRef, data: popupDict },
|
||
{ ref: replyRef, data: replyDict },
|
||
]);
|
||
annotationDict.assignXref(xref);
|
||
popupDict.assignXref(xref);
|
||
replyDict.assignXref(xref);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
replyRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.inReplyTo).toEqual(annotationRef.toString());
|
||
expect(data.replyType).toEqual("Group");
|
||
expect(data.titleObj).toEqual({ str: "ParentTitle", dir: "ltr" });
|
||
expect(data.contentsObj).toEqual({ str: "ParentText", dir: "ltr" });
|
||
expect(data.creationDate).toEqual("D:20180423");
|
||
expect(data.modificationDate).toEqual("D:20190423");
|
||
expect(data.color).toEqual(new Uint8ClampedArray([0, 0, 255]));
|
||
expect(data.hasPopup).toEqual(true);
|
||
});
|
||
|
||
it("should parse IRT/RT for a reply type", async function () {
|
||
const annotationRef = Ref.get(819, 0);
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Text"));
|
||
annotationDict.set("T", "ParentTitle");
|
||
annotationDict.set("Contents", "ParentText");
|
||
annotationDict.set("CreationDate", "D:20180423");
|
||
annotationDict.set("M", "D:20190423");
|
||
annotationDict.set("C", [0, 0, 1]);
|
||
|
||
const popupRef = Ref.get(820, 0);
|
||
const popupDict = new Dict();
|
||
popupDict.set("Type", Name.get("Annot"));
|
||
popupDict.set("Subtype", Name.get("Popup"));
|
||
popupDict.set("Parent", annotationRef);
|
||
annotationDict.set("Popup", popupRef);
|
||
|
||
const replyRef = Ref.get(821, 0);
|
||
const replyDict = new Dict();
|
||
replyDict.set("Type", Name.get("Annot"));
|
||
replyDict.set("Subtype", Name.get("Text"));
|
||
replyDict.set("IRT", annotationRef);
|
||
replyDict.set("RT", Name.get("R"));
|
||
replyDict.set("T", "ReplyTitle");
|
||
replyDict.set("Contents", "ReplyText");
|
||
replyDict.set("CreationDate", "D:20180523");
|
||
replyDict.set("M", "D:20190523");
|
||
replyDict.set("C", [0.4]);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
{ ref: popupRef, data: popupDict },
|
||
{ ref: replyRef, data: replyDict },
|
||
]);
|
||
annotationDict.assignXref(xref);
|
||
popupDict.assignXref(xref);
|
||
replyDict.assignXref(xref);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
replyRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.inReplyTo).toEqual(annotationRef.toString());
|
||
expect(data.replyType).toEqual("R");
|
||
expect(data.titleObj).toEqual({ str: "ReplyTitle", dir: "ltr" });
|
||
expect(data.contentsObj).toEqual({ str: "ReplyText", dir: "ltr" });
|
||
expect(data.creationDate).toEqual("D:20180523");
|
||
expect(data.modificationDate).toEqual("D:20190523");
|
||
expect(data.color).toEqual(new Uint8ClampedArray([102, 102, 102]));
|
||
expect(data.hasPopup).toEqual(false);
|
||
});
|
||
});
|
||
|
||
describe("TextAnnotation", function () {
|
||
it("should not parse state model and state when not defined", async function () {
|
||
const annotationRef = Ref.get(819, 0);
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Text"));
|
||
annotationDict.set("Contents", "TestText");
|
||
|
||
const replyRef = Ref.get(820, 0);
|
||
const replyDict = new Dict();
|
||
replyDict.set("Type", Name.get("Annot"));
|
||
replyDict.set("Subtype", Name.get("Text"));
|
||
replyDict.set("IRT", annotationRef);
|
||
replyDict.set("RT", Name.get("R"));
|
||
replyDict.set("Contents", "ReplyText");
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
{ ref: replyRef, data: replyDict },
|
||
]);
|
||
annotationDict.assignXref(xref);
|
||
replyDict.assignXref(xref);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
replyRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.stateModel).toBeNull();
|
||
expect(data.state).toBeNull();
|
||
});
|
||
|
||
it("should correctly parse state model and state when defined", async function () {
|
||
const annotationRef = Ref.get(819, 0);
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Text"));
|
||
|
||
const replyRef = Ref.get(820, 0);
|
||
const replyDict = new Dict();
|
||
replyDict.set("Type", Name.get("Annot"));
|
||
replyDict.set("Subtype", Name.get("Text"));
|
||
replyDict.set("IRT", annotationRef);
|
||
replyDict.set("RT", Name.get("R"));
|
||
replyDict.set("StateModel", "Review");
|
||
replyDict.set("State", "Rejected");
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
{ ref: replyRef, data: replyDict },
|
||
]);
|
||
annotationDict.assignXref(xref);
|
||
replyDict.assignXref(xref);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
replyRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.stateModel).toEqual("Review");
|
||
expect(data.state).toEqual("Rejected");
|
||
});
|
||
});
|
||
|
||
describe("LinkAnnotation", function () {
|
||
it("should correctly parse a URI action", async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("URI"));
|
||
actionDict.set("URI", "http://www.ctan.org/tex-archive/info/lshort");
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(820, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual("http://www.ctan.org/tex-archive/info/lshort");
|
||
expect(data.unsafeUrl).toEqual(
|
||
"http://www.ctan.org/tex-archive/info/lshort"
|
||
);
|
||
expect(data.dest).toBeUndefined();
|
||
});
|
||
|
||
it(
|
||
"should correctly parse a URI action, where the URI entry " +
|
||
"is missing a protocol",
|
||
async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("URI"));
|
||
actionDict.set("URI", "www.hmrc.gov.uk");
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(353, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual("http://www.hmrc.gov.uk/");
|
||
expect(data.unsafeUrl).toEqual("www.hmrc.gov.uk");
|
||
expect(data.dest).toBeUndefined();
|
||
}
|
||
);
|
||
|
||
it(
|
||
"should correctly parse a URI action, where the URI entry " +
|
||
"has an incorrect encoding (bug 1122280)",
|
||
async function () {
|
||
const actionStream = new StringStream(
|
||
"<<\n" +
|
||
"/Type /Action\n" +
|
||
"/S /URI\n" +
|
||
"/URI (http://www.example.com/\\303\\274\\303\\266\\303\\244)\n" +
|
||
">>\n"
|
||
);
|
||
const parser = new Parser({
|
||
lexer: new Lexer(actionStream),
|
||
xref: null,
|
||
});
|
||
const actionDict = parser.getObj();
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(8, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual(
|
||
new URL(
|
||
stringToUTF8String(
|
||
"http://www.example.com/\xC3\xBC\xC3\xB6\xC3\xA4"
|
||
)
|
||
).href
|
||
);
|
||
expect(data.unsafeUrl).toEqual(
|
||
"http://www.example.com/\xC3\xBC\xC3\xB6\xC3\xA4"
|
||
);
|
||
expect(data.dest).toBeUndefined();
|
||
}
|
||
);
|
||
|
||
it("should correctly parse a GoTo action", async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("GoTo"));
|
||
actionDict.set("D", "page.157");
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(798, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toBeUndefined();
|
||
expect(data.unsafeUrl).toBeUndefined();
|
||
expect(data.dest).toEqual("page.157");
|
||
});
|
||
|
||
it(
|
||
"should correctly parse a GoToR action, where the FileSpec entry " +
|
||
"is a string containing a relative URL",
|
||
async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("GoToR"));
|
||
actionDict.set("F", "../../0013/001346/134685E.pdf");
|
||
actionDict.set("D", "4.3");
|
||
actionDict.set("NewWindow", true);
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(489, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toBeUndefined();
|
||
expect(data.unsafeUrl).toEqual("../../0013/001346/134685E.pdf#4.3");
|
||
expect(data.dest).toBeUndefined();
|
||
expect(data.newWindow).toEqual(true);
|
||
}
|
||
);
|
||
|
||
it(
|
||
"should correctly parse a GoToR action, containing a relative URL, " +
|
||
'with the "docBaseUrl" parameter specified',
|
||
async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("GoToR"));
|
||
actionDict.set("F", "../../0013/001346/134685E.pdf");
|
||
actionDict.set("D", "4.3");
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(489, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
]);
|
||
const pdfManager = new PDFManagerMock({
|
||
docBaseUrl: "http://www.example.com/test/pdfs/qwerty.pdf",
|
||
});
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManager,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual(
|
||
"http://www.example.com/0013/001346/134685E.pdf#4.3"
|
||
);
|
||
expect(data.unsafeUrl).toEqual("../../0013/001346/134685E.pdf#4.3");
|
||
expect(data.dest).toBeUndefined();
|
||
}
|
||
);
|
||
|
||
it("should correctly parse a GoToR action, with named destination", async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("GoToR"));
|
||
actionDict.set("F", "http://www.example.com/test.pdf");
|
||
actionDict.set("D", "15");
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(495, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual("http://www.example.com/test.pdf#15");
|
||
expect(data.unsafeUrl).toEqual("http://www.example.com/test.pdf#15");
|
||
expect(data.dest).toBeUndefined();
|
||
expect(data.newWindow).toBeFalsy();
|
||
});
|
||
|
||
it("should correctly parse a GoToR action, with explicit destination array", async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("GoToR"));
|
||
actionDict.set("F", "http://www.example.com/test.pdf");
|
||
actionDict.set("D", [14, Name.get("XYZ"), null, 298.043, null]);
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(489, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual(
|
||
new URL(
|
||
"http://www.example.com/test.pdf#" +
|
||
'[14,{"name":"XYZ"},null,298.043,null]'
|
||
).href
|
||
);
|
||
expect(data.unsafeUrl).toEqual(
|
||
"http://www.example.com/test.pdf#" +
|
||
'[14,{"name":"XYZ"},null,298.043,null]'
|
||
);
|
||
expect(data.dest).toBeUndefined();
|
||
expect(data.newWindow).toBeFalsy();
|
||
});
|
||
|
||
it(
|
||
"should correctly parse a Launch action, where the FileSpec dict " +
|
||
'contains a relative URL, with the "docBaseUrl" parameter specified',
|
||
async function () {
|
||
const fileSpecDict = new Dict();
|
||
fileSpecDict.set("Type", Name.get("FileSpec"));
|
||
fileSpecDict.set("F", "Part II/Part II.pdf");
|
||
fileSpecDict.set("UF", "Part II/Part II.pdf");
|
||
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("Launch"));
|
||
actionDict.set("F", fileSpecDict);
|
||
actionDict.set("NewWindow", true);
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(88, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
]);
|
||
const pdfManager = new PDFManagerMock({
|
||
docBaseUrl: "http://www.example.com/test/pdfs/qwerty.pdf",
|
||
});
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManager,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual(
|
||
new URL("http://www.example.com/test/pdfs/Part II/Part II.pdf").href
|
||
);
|
||
expect(data.unsafeUrl).toEqual("Part II/Part II.pdf");
|
||
expect(data.dest).toBeUndefined();
|
||
expect(data.newWindow).toEqual(true);
|
||
}
|
||
);
|
||
|
||
it(
|
||
"should recover valid URLs from JavaScript actions having certain " +
|
||
"white-listed formats",
|
||
async function () {
|
||
function checkJsAction(params) {
|
||
const jsEntry = params.jsEntry;
|
||
const expectedUrl = params.expectedUrl;
|
||
const expectedUnsafeUrl = params.expectedUnsafeUrl;
|
||
const expectedNewWindow = params.expectedNewWindow;
|
||
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("JavaScript"));
|
||
actionDict.set("JS", jsEntry);
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(46, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
]);
|
||
|
||
return AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
).then(({ data }) => {
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toEqual(expectedUrl);
|
||
expect(data.unsafeUrl).toEqual(expectedUnsafeUrl);
|
||
expect(data.dest).toBeUndefined();
|
||
expect(data.newWindow).toEqual(expectedNewWindow);
|
||
});
|
||
}
|
||
|
||
// Check that we reject a 'JS' entry containing arbitrary JavaScript.
|
||
const annotation1 = checkJsAction({
|
||
jsEntry: 'function someFun() { return "qwerty"; } someFun();',
|
||
expectedUrl: undefined,
|
||
expectedUnsafeUrl: undefined,
|
||
expectedNewWindow: undefined,
|
||
});
|
||
|
||
// Check that we accept a white-listed {string} 'JS' entry.
|
||
const annotation2 = checkJsAction({
|
||
jsEntry: "window.open('http://www.example.com/test.pdf')",
|
||
expectedUrl: new URL("http://www.example.com/test.pdf").href,
|
||
expectedUnsafeUrl: "http://www.example.com/test.pdf",
|
||
expectedNewWindow: false,
|
||
});
|
||
|
||
// Check that we accept a white-listed {Stream} 'JS' entry.
|
||
const annotation3 = checkJsAction({
|
||
jsEntry: new StringStream(
|
||
'app.launchURL("http://www.example.com/test.pdf", true)'
|
||
),
|
||
expectedUrl: new URL("http://www.example.com/test.pdf").href,
|
||
expectedUnsafeUrl: "http://www.example.com/test.pdf",
|
||
expectedNewWindow: true,
|
||
});
|
||
|
||
await Promise.all([annotation1, annotation2, annotation3]);
|
||
}
|
||
);
|
||
|
||
it("should correctly parse a Named action", async function () {
|
||
const actionDict = new Dict();
|
||
actionDict.set("Type", Name.get("Action"));
|
||
actionDict.set("S", Name.get("Named"));
|
||
actionDict.set("N", Name.get("GoToPage"));
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("A", actionDict);
|
||
|
||
const annotationRef = Ref.get(12, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toBeUndefined();
|
||
expect(data.unsafeUrl).toBeUndefined();
|
||
expect(data.action).toEqual("GoToPage");
|
||
});
|
||
|
||
it("should correctly parse a simple Dest", async function () {
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("Dest", Name.get("LI0"));
|
||
|
||
const annotationRef = Ref.get(583, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toBeUndefined();
|
||
expect(data.unsafeUrl).toBeUndefined();
|
||
expect(data.dest).toEqual("LI0");
|
||
});
|
||
|
||
it("should correctly parse a simple Dest, with explicit destination array", async function () {
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("Dest", [
|
||
Ref.get(17, 0),
|
||
Name.get("XYZ"),
|
||
0,
|
||
841.89,
|
||
null,
|
||
]);
|
||
|
||
const annotationRef = Ref.get(10, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toBeUndefined();
|
||
expect(data.unsafeUrl).toBeUndefined();
|
||
expect(data.dest).toEqual([
|
||
Ref.get(17, 0),
|
||
Name.get("XYZ"),
|
||
0,
|
||
841.89,
|
||
null,
|
||
]);
|
||
});
|
||
|
||
it(
|
||
"should correctly parse a Dest, which violates the specification " +
|
||
"by containing a dictionary",
|
||
async function () {
|
||
const destDict = new Dict();
|
||
destDict.set("Type", Name.get("Action"));
|
||
destDict.set("S", Name.get("GoTo"));
|
||
destDict.set("D", "page.157");
|
||
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
// The /Dest must be a Name or an Array, refer to ISO 32000-1:2008
|
||
// section 12.3.3, but there are PDF files where it's a dictionary.
|
||
annotationDict.set("Dest", destDict);
|
||
|
||
const annotationRef = Ref.get(798, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.url).toBeUndefined();
|
||
expect(data.unsafeUrl).toBeUndefined();
|
||
expect(data.dest).toEqual("page.157");
|
||
}
|
||
);
|
||
|
||
it("should not set quadpoints if not defined", async function () {
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
|
||
const annotationRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.quadPoints).toBeUndefined();
|
||
});
|
||
|
||
it("should set quadpoints if defined", async function () {
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Link"));
|
||
annotationDict.set("Rect", [10, 10, 20, 20]);
|
||
annotationDict.set("QuadPoints", [10, 20, 20, 20, 10, 10, 20, 10]);
|
||
|
||
const annotationRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: annotationRef, data: annotationDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
annotationRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
||
expect(data.quadPoints).toEqual([
|
||
[
|
||
{ x: 10, y: 20 },
|
||
{ x: 20, y: 20 },
|
||
{ x: 10, y: 10 },
|
||
{ x: 20, y: 10 },
|
||
],
|
||
]);
|
||
});
|
||
});
|
||
|
||
describe("WidgetAnnotation", function () {
|
||
let widgetDict;
|
||
|
||
beforeEach(function () {
|
||
widgetDict = new Dict();
|
||
widgetDict.set("Type", Name.get("Annot"));
|
||
widgetDict.set("Subtype", Name.get("Widget"));
|
||
});
|
||
|
||
afterEach(function () {
|
||
widgetDict = null;
|
||
});
|
||
|
||
it("should handle unknown field names", async function () {
|
||
const widgetRef = Ref.get(20, 0);
|
||
const xref = new XRefMock([{ ref: widgetRef, data: widgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
widgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.fieldName).toEqual("");
|
||
});
|
||
|
||
it("should construct the field name when there are no ancestors", async function () {
|
||
widgetDict.set("T", "foo");
|
||
|
||
const widgetRef = Ref.get(21, 0);
|
||
const xref = new XRefMock([{ ref: widgetRef, data: widgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
widgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.fieldName).toEqual("foo");
|
||
});
|
||
|
||
it("should construct the field name when there are ancestors", async function () {
|
||
const firstParent = new Dict();
|
||
firstParent.set("T", "foo");
|
||
|
||
const secondParent = new Dict();
|
||
secondParent.set("Parent", firstParent);
|
||
secondParent.set("T", "bar");
|
||
|
||
widgetDict.set("Parent", secondParent);
|
||
widgetDict.set("T", "baz");
|
||
|
||
const widgetRef = Ref.get(22, 0);
|
||
const xref = new XRefMock([{ ref: widgetRef, data: widgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
widgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.fieldName).toEqual("foo.bar.baz");
|
||
});
|
||
|
||
it(
|
||
"should construct the field name if a parent is not a dictionary " +
|
||
"(issue 8143)",
|
||
async function () {
|
||
const parentDict = new Dict();
|
||
parentDict.set("Parent", null);
|
||
parentDict.set("T", "foo");
|
||
|
||
widgetDict.set("Parent", parentDict);
|
||
widgetDict.set("T", "bar");
|
||
|
||
const widgetRef = Ref.get(22, 0);
|
||
const xref = new XRefMock([{ ref: widgetRef, data: widgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
widgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.fieldName).toEqual("foo.bar");
|
||
}
|
||
);
|
||
});
|
||
|
||
describe("TextWidgetAnnotation", function () {
|
||
let textWidgetDict, helvRefObj, gothRefObj;
|
||
|
||
beforeEach(function () {
|
||
textWidgetDict = new Dict();
|
||
textWidgetDict.set("Type", Name.get("Annot"));
|
||
textWidgetDict.set("Subtype", Name.get("Widget"));
|
||
textWidgetDict.set("FT", Name.get("Tx"));
|
||
|
||
const helvDict = new Dict();
|
||
helvDict.set("BaseFont", Name.get("Helvetica"));
|
||
helvDict.set("Type", Name.get("Font"));
|
||
helvDict.set("Subtype", Name.get("Type1"));
|
||
|
||
const gothDict = new Dict();
|
||
gothDict.set("BaseFont", Name.get("MSGothic"));
|
||
gothDict.set("Type", Name.get("Font"));
|
||
gothDict.set("Subtype", Name.get("Type0"));
|
||
gothDict.set("Encoding", Name.get("UniJIS-UTF16-H"));
|
||
gothDict.set("Name", Name.get("MSGothic"));
|
||
|
||
const cidSysInfoDict = new Dict();
|
||
cidSysInfoDict.set("Ordering", "Japan1");
|
||
cidSysInfoDict.set("Registry", "Adobe");
|
||
cidSysInfoDict.set("Supplement", "5");
|
||
|
||
const fontDescriptorDict = new Dict();
|
||
fontDescriptorDict.set("FontName", Name.get("MSGothic"));
|
||
fontDescriptorDict.set("CapHeight", "680");
|
||
|
||
const gothDescendantDict = new Dict();
|
||
gothDescendantDict.set("BaseFont", Name.get("MSGothic"));
|
||
gothDescendantDict.set("CIDSystemInfo", cidSysInfoDict);
|
||
gothDescendantDict.set("Subtype", Name.get("CIDFontType2"));
|
||
gothDescendantDict.set("Type", Name.get("Font"));
|
||
gothDescendantDict.set("FontDescriptor", fontDescriptorDict);
|
||
|
||
gothDict.set("DescendantFonts", [gothDescendantDict]);
|
||
|
||
const helvRef = Ref.get(314, 0);
|
||
const gothRef = Ref.get(159, 0);
|
||
helvRefObj = { ref: helvRef, data: helvDict };
|
||
gothRefObj = { ref: gothRef, data: gothDict };
|
||
const resourceDict = new Dict();
|
||
const fontDict = new Dict();
|
||
fontDict.set("Helv", helvRef);
|
||
resourceDict.set("Font", fontDict);
|
||
|
||
textWidgetDict.set("DA", "/Helv 5 Tf");
|
||
textWidgetDict.set("DR", resourceDict);
|
||
textWidgetDict.set("Rect", [0, 0, 32, 10]);
|
||
});
|
||
|
||
afterEach(function () {
|
||
textWidgetDict = helvRefObj = gothRefObj = null;
|
||
});
|
||
|
||
it("should handle unknown text alignment, maximum length and flags", async function () {
|
||
textWidgetDict.set("DV", "foo");
|
||
|
||
const textWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([{ ref: textWidgetRef, data: textWidgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.textAlignment).toEqual(null);
|
||
expect(data.maxLen).toEqual(0);
|
||
expect(data.readOnly).toEqual(false);
|
||
expect(data.hidden).toEqual(false);
|
||
expect(data.multiLine).toEqual(false);
|
||
expect(data.comb).toEqual(false);
|
||
expect(data.defaultFieldValue).toEqual("foo");
|
||
});
|
||
|
||
it("should not set invalid text alignment, maximum length and flags", async function () {
|
||
textWidgetDict.set("Q", "center");
|
||
textWidgetDict.set("MaxLen", "five");
|
||
textWidgetDict.set("Ff", "readonly");
|
||
|
||
const textWidgetRef = Ref.get(43, 0);
|
||
const xref = new XRefMock([{ ref: textWidgetRef, data: textWidgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.textAlignment).toEqual(null);
|
||
expect(data.maxLen).toEqual(0);
|
||
expect(data.readOnly).toEqual(false);
|
||
expect(data.hidden).toEqual(false);
|
||
expect(data.multiLine).toEqual(false);
|
||
expect(data.comb).toEqual(false);
|
||
});
|
||
|
||
it("should set valid text alignment, maximum length and flags", async function () {
|
||
textWidgetDict.set("Q", 1);
|
||
textWidgetDict.set("MaxLen", 20);
|
||
textWidgetDict.set(
|
||
"Ff",
|
||
AnnotationFieldFlag.READONLY + AnnotationFieldFlag.MULTILINE
|
||
);
|
||
|
||
const textWidgetRef = Ref.get(84, 0);
|
||
const xref = new XRefMock([{ ref: textWidgetRef, data: textWidgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.textAlignment).toEqual(1);
|
||
expect(data.maxLen).toEqual(20);
|
||
expect(data.readOnly).toEqual(true);
|
||
expect(data.hidden).toEqual(false);
|
||
expect(data.multiLine).toEqual(true);
|
||
});
|
||
|
||
it("should reject comb fields without a maximum length", async function () {
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.COMB);
|
||
|
||
const textWidgetRef = Ref.get(46, 0);
|
||
const xref = new XRefMock([{ ref: textWidgetRef, data: textWidgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.comb).toEqual(false);
|
||
});
|
||
|
||
it("should accept comb fields with a maximum length", async function () {
|
||
textWidgetDict.set("MaxLen", 20);
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.COMB);
|
||
|
||
const textWidgetRef = Ref.get(46, 0);
|
||
const xref = new XRefMock([{ ref: textWidgetRef, data: textWidgetDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.comb).toEqual(true);
|
||
});
|
||
|
||
it("should only accept comb fields when the flags are valid", async function () {
|
||
const invalidFieldFlags = [
|
||
AnnotationFieldFlag.MULTILINE,
|
||
AnnotationFieldFlag.PASSWORD,
|
||
AnnotationFieldFlag.FILESELECT,
|
||
];
|
||
|
||
// Start with all invalid flags set and remove them one by one.
|
||
// The field may only use combs when all invalid flags are unset.
|
||
let flags =
|
||
AnnotationFieldFlag.COMB +
|
||
AnnotationFieldFlag.MULTILINE +
|
||
AnnotationFieldFlag.PASSWORD +
|
||
AnnotationFieldFlag.FILESELECT;
|
||
|
||
let promise = Promise.resolve();
|
||
for (let i = 0, ii = invalidFieldFlags.length; i <= ii; i++) {
|
||
promise = promise.then(() => {
|
||
textWidgetDict.set("MaxLen", 20);
|
||
textWidgetDict.set("Ff", flags);
|
||
|
||
const textWidgetRef = Ref.get(93, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
]);
|
||
|
||
return AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
).then(({ data }) => {
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
|
||
const valid = invalidFieldFlags.length === 0;
|
||
expect(data.comb).toEqual(valid);
|
||
|
||
// Remove the last invalid flag for the next iteration.
|
||
if (!valid) {
|
||
flags -= invalidFieldFlags.pop();
|
||
}
|
||
});
|
||
});
|
||
}
|
||
await promise;
|
||
});
|
||
|
||
it("should render regular text for printing", async function () {
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "test\\print" });
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Helv 5 Tf 1 0 0 1 0 0 Tm" +
|
||
" 2 3.07 Td (test\\\\print) Tj ET Q EMC"
|
||
);
|
||
});
|
||
|
||
it("should render regular text in Japanese for printing", async function () {
|
||
textWidgetDict.get("DR").get("Font").set("Goth", gothRefObj.ref);
|
||
textWidgetDict.set("DA", "/Goth 5 Tf");
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
gothRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value: "こんにちは世界の",
|
||
});
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
const utf16String =
|
||
"\x30\x53\x30\x93\x30\x6b\x30\x61\x30\x6f\x4e\x16\x75\x4c\x30\x6e";
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Goth 5 Tf 1 0 0 1 0 0 Tm" +
|
||
` 2 3.07 Td (${utf16String}) Tj ET Q EMC`
|
||
);
|
||
});
|
||
|
||
it("should render regular text for printing using normal appearance", async function () {
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
|
||
const normalAppearanceStream = new StringStream("0.1 0.2 0.3 rg");
|
||
normalAppearanceStream.dict = normalAppearanceDict;
|
||
|
||
appearanceStatesDict.set("N", normalAppearanceStream);
|
||
textWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
|
||
const { opList } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList.argsArray.length).toEqual(3);
|
||
expect(opList.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList.argsArray[0]).toEqual([
|
||
"271R",
|
||
[0, 0, 32, 10],
|
||
[32, 0, 0, 10, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList.argsArray[1]).toEqual(new Uint8ClampedArray([26, 51, 76]));
|
||
});
|
||
|
||
it("should render auto-sized text for printing", async function () {
|
||
textWidgetDict.set("DA", "/Helv 0 Tf");
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "test (print)" });
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Helv 5.92 Tf 0 g 1 0 0 1 0 0 Tm" +
|
||
" 2 3.07 Td (test \\(print\\)) Tj ET Q EMC"
|
||
);
|
||
});
|
||
|
||
it("should render auto-sized text in Japanese for printing", async function () {
|
||
textWidgetDict.get("DR").get("Font").set("Goth", gothRefObj.ref);
|
||
textWidgetDict.set("DA", "/Goth 0 Tf");
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
gothRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value: "こんにちは世界の",
|
||
});
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
const utf16String =
|
||
"\x30\x53\x30\x93\x30\x6b\x30\x61\x30\x6f\x4e\x16\x75\x4c\x30\x6e";
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Goth 5.92 Tf 0 g 1 0 0 1 0 0 Tm" +
|
||
` 2 3.07 Td (${utf16String}) Tj ET Q EMC`
|
||
);
|
||
});
|
||
|
||
it("should not render a password for printing", async function () {
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.PASSWORD);
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "mypassword" });
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(null);
|
||
});
|
||
|
||
it("should render multiline text for printing", async function () {
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.MULTILINE);
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value:
|
||
"a aa aaa aaaa aaaaa aaaaaa " +
|
||
"pneumonoultramicroscopicsilicovolcanoconiosis",
|
||
});
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Helv 5 Tf 1 0 0 1 0 10 Tm " +
|
||
"2 -6.93 Td (a aa aaa ) Tj\n" +
|
||
"0 -8 Td (aaaa aaaaa ) Tj\n" +
|
||
"0 -8 Td (aaaaaa ) Tj\n" +
|
||
"0 -8 Td (pneumonoultr) Tj\n" +
|
||
"0 -8 Td (amicroscopi) Tj\n" +
|
||
"0 -8 Td (csilicovolca) Tj\n" +
|
||
"0 -8 Td (noconiosis) Tj ET Q EMC"
|
||
);
|
||
});
|
||
|
||
it("should render multiline text in Japanese for printing", async function () {
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.MULTILINE);
|
||
textWidgetDict.get("DR").get("Font").set("Goth", gothRefObj.ref);
|
||
textWidgetDict.set("DA", "/Goth 5 Tf");
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
gothRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value: "こんにちは世界の",
|
||
});
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Goth 5 Tf 1 0 0 1 0 10 Tm " +
|
||
"2 -6.93 Td (\x30\x53\x30\x93\x30\x6b\x30\x61\x30\x6f) Tj\n" +
|
||
"0 -8 Td (\x4e\x16\x75\x4c\x30\x6e) Tj ET Q EMC"
|
||
);
|
||
});
|
||
|
||
it("should render multiline text with various EOL for printing", async function () {
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.MULTILINE);
|
||
textWidgetDict.set("Rect", [0, 0, 128, 10]);
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
const expectedAppearance =
|
||
"/Tx BMC q BT /Helv 5 Tf 1 0 0 1 0 10 Tm " +
|
||
"2 -6.93 Td " +
|
||
"(Lorem ipsum dolor sit amet, consectetur adipiscing elit.) Tj\n" +
|
||
"0 -8 Td " +
|
||
"(Aliquam vitae felis ac lectus bibendum ultricies quis non) Tj\n" +
|
||
"0 -8 Td " +
|
||
"( diam.) Tj\n" +
|
||
"0 -8 Td " +
|
||
"(Morbi id porttitor quam, a iaculis dui.) Tj\n" +
|
||
"0 -8 Td " +
|
||
"(Pellentesque habitant morbi tristique senectus et netus ) Tj\n" +
|
||
"0 -8 Td " +
|
||
"(et malesuada fames ac turpis egestas.) Tj\n" +
|
||
"0 -8 Td () Tj\n" +
|
||
"0 -8 Td () Tj\n" +
|
||
"0 -8 Td " +
|
||
"(Nulla consectetur, ligula in tincidunt placerat, velit ) Tj\n" +
|
||
"0 -8 Td " +
|
||
"(augue consectetur orci, sed mattis libero nunc ut massa.) Tj\n" +
|
||
"0 -8 Td " +
|
||
"(Etiam facilisis tempus interdum.) Tj ET Q EMC";
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value:
|
||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit.\r" +
|
||
"Aliquam vitae felis ac lectus bibendum ultricies quis non diam.\n" +
|
||
"Morbi id porttitor quam, a iaculis dui.\r\n" +
|
||
"Pellentesque habitant morbi tristique senectus et " +
|
||
"netus et malesuada fames ac turpis egestas.\n\r\n\r" +
|
||
"Nulla consectetur, ligula in tincidunt placerat, " +
|
||
"velit augue consectetur orci, sed mattis libero nunc ut massa.\r" +
|
||
"Etiam facilisis tempus interdum.",
|
||
});
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
|
||
expect(appearance).toEqual(expectedAppearance);
|
||
});
|
||
|
||
it("should render comb for printing", async function () {
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.COMB);
|
||
textWidgetDict.set("MaxLen", 4);
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "aa(aa)a\\" });
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Helv 5 Tf 1 0 0 1 2 3.07 Tm" +
|
||
" (a) Tj 8 0 Td (a) Tj 8 0 Td (\\() Tj" +
|
||
" 8 0 Td (a) Tj 8 0 Td (a) Tj" +
|
||
" 8 0 Td (\\)) Tj 8 0 Td (a) Tj" +
|
||
" 8 0 Td (\\\\) Tj ET Q EMC"
|
||
);
|
||
});
|
||
|
||
it("should render comb with Japanese text for printing", async function () {
|
||
textWidgetDict.set("Ff", AnnotationFieldFlag.COMB);
|
||
textWidgetDict.set("MaxLen", 4);
|
||
textWidgetDict.get("DR").get("Font").set("Goth", gothRefObj.ref);
|
||
textWidgetDict.set("DA", "/Goth 5 Tf");
|
||
textWidgetDict.set("Rect", [0, 0, 32, 10]);
|
||
|
||
const textWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
gothRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value: "こんにちは世界の",
|
||
});
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
"/Tx BMC q BT /Goth 5 Tf 1 0 0 1 2 3.07 Tm" +
|
||
" (\x30\x53) Tj 8 0 Td (\x30\x93) Tj 8 0 Td (\x30\x6b) Tj" +
|
||
" 8 0 Td (\x30\x61) Tj 8 0 Td (\x30\x6f) Tj" +
|
||
" 8 0 Td (\x4e\x16) Tj 8 0 Td (\x75\x4c) Tj" +
|
||
" 8 0 Td (\x30\x6e) Tj ET Q EMC"
|
||
);
|
||
});
|
||
|
||
it("should save text", async function () {
|
||
const textWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "hello world" });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data.length).toEqual(2);
|
||
const [oldData, newData] = data;
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(newData.ref).toEqual(Ref.get(2, 0));
|
||
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Tx /DA (/Helv 5 Tf) /DR " +
|
||
"<< /Font << /Helv 314 0 R>>>> /Rect [0 0 32 10] " +
|
||
"/V (hello world) /AP << /N 2 0 R>> /M (date)>>\nendobj\n"
|
||
);
|
||
expect(newData.data).toEqual(
|
||
"2 0 obj\n<< /Subtype /Form /Resources " +
|
||
"<< /Font << /Helv 314 0 R>>>> /BBox [0 0 32 10] /Length 74>> stream\n" +
|
||
"/Tx BMC q BT /Helv 5 Tf 1 0 0 1 0 0 Tm 2 3.07 Td (hello world) Tj " +
|
||
"ET Q EMC\nendstream\nendobj\n"
|
||
);
|
||
});
|
||
|
||
it("should save rotated text", async function () {
|
||
const textWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value: "hello world",
|
||
rotation: 90,
|
||
});
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data.length).toEqual(2);
|
||
const [oldData, newData] = data;
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(newData.ref).toEqual(Ref.get(2, 0));
|
||
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Tx /DA (/Helv 5 Tf) /DR " +
|
||
"<< /Font << /Helv 314 0 R>>>> /Rect [0 0 32 10] " +
|
||
"/V (hello world) /MK << /R 90>> /AP << /N 2 0 R>> /M (date)>>\nendobj\n"
|
||
);
|
||
expect(newData.data).toEqual(
|
||
"2 0 obj\n<< /Subtype /Form /Resources " +
|
||
"<< /Font << /Helv 314 0 R>>>> /BBox [0 0 32 10] /Matrix [0 1 -1 0 32 0] /Length 74>> stream\n" +
|
||
"/Tx BMC q BT /Helv 5 Tf 1 0 0 1 0 0 Tm 2 2.94 Td (hello world) Tj " +
|
||
"ET Q EMC\nendstream\nendobj\n"
|
||
);
|
||
});
|
||
|
||
it("should compress and save text", async function () {
|
||
const textWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
helvRefObj,
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
const value = "a".repeat(256);
|
||
annotationStorage.set(annotation.data.id, { value });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data.length).toEqual(2);
|
||
const [oldData, newData] = data;
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(newData.ref).toEqual(Ref.get(2, 0));
|
||
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Tx /DA (/Helv 5 Tf) /DR " +
|
||
"<< /Font << /Helv 314 0 R>>>> /Rect [0 0 32 10] " +
|
||
`/V (${value}) /AP << /N 2 0 R>> /M (date)>>\nendobj\n`
|
||
);
|
||
|
||
const compressedData = [
|
||
120, 156, 211, 15, 169, 80, 112, 242, 117, 86, 40, 84, 112, 10, 81, 208,
|
||
247, 72, 205, 41, 83, 48, 85, 8, 73, 83, 48, 84, 48, 0, 66, 8, 25, 146,
|
||
171, 96, 164, 96, 172, 103, 96, 174, 16, 146, 162, 160, 145, 56, 194,
|
||
129, 166, 66, 72, 150, 130, 107, 136, 66, 160, 130, 171, 175, 51, 0,
|
||
222, 235, 111, 133,
|
||
];
|
||
const compressedStream = String.fromCharCode(...compressedData);
|
||
expect(newData.data).toEqual(
|
||
"2 0 obj\n<< /Subtype /Form /Resources " +
|
||
"<< /Font << /Helv 314 0 R>>>> /BBox [0 0 32 10] /Filter /FlateDecode /Length 68>> stream\n" +
|
||
`${compressedStream}\nendstream\nendobj\n`
|
||
);
|
||
});
|
||
|
||
it("should get field object for usage in JS sandbox", async function () {
|
||
const textWidgetRef = Ref.get(123, 0);
|
||
const xDictRef = Ref.get(141, 0);
|
||
const dDictRef = Ref.get(262, 0);
|
||
const next0Ref = Ref.get(314, 0);
|
||
const next1Ref = Ref.get(271, 0);
|
||
const next2Ref = Ref.get(577, 0);
|
||
const next00Ref = Ref.get(413, 0);
|
||
const xDict = new Dict();
|
||
const dDict = new Dict();
|
||
const next0Dict = new Dict();
|
||
const next1Dict = new Dict();
|
||
const next2Dict = new Dict();
|
||
const next00Dict = new Dict();
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
{ ref: xDictRef, data: xDict },
|
||
{ ref: dDictRef, data: dDict },
|
||
{ ref: next0Ref, data: next0Dict },
|
||
{ ref: next00Ref, data: next00Dict },
|
||
{ ref: next1Ref, data: next1Dict },
|
||
{ ref: next2Ref, data: next2Dict },
|
||
]);
|
||
|
||
const JS = Name.get("JavaScript");
|
||
const additionalActionsDict = new Dict();
|
||
const eDict = new Dict();
|
||
eDict.set("JS", "hello()");
|
||
eDict.set("S", JS);
|
||
additionalActionsDict.set("E", eDict);
|
||
|
||
// Test the cycle detection here.
|
||
xDict.set("JS", "world()");
|
||
xDict.set("S", JS);
|
||
xDict.set("Next", [next0Ref, next1Ref, next2Ref, xDictRef]);
|
||
|
||
next0Dict.set("JS", "olleh()");
|
||
next0Dict.set("S", JS);
|
||
next0Dict.set("Next", next00Ref);
|
||
|
||
next00Dict.set("JS", "foo()");
|
||
next00Dict.set("S", JS);
|
||
next00Dict.set("Next", next0Ref);
|
||
|
||
next1Dict.set("JS", "dlrow()");
|
||
next1Dict.set("S", JS);
|
||
next1Dict.set("Next", xDictRef);
|
||
|
||
next2Dict.set("JS", "oof()");
|
||
next2Dict.set("S", JS);
|
||
|
||
dDict.set("JS", "bar()");
|
||
dDict.set("S", JS);
|
||
dDict.set("Next", dDictRef);
|
||
additionalActionsDict.set("D", dDictRef);
|
||
|
||
additionalActionsDict.set("X", xDictRef);
|
||
textWidgetDict.set("AA", additionalActionsDict);
|
||
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const fieldObject = await annotation.getFieldObject();
|
||
const actions = fieldObject.actions;
|
||
expect(actions["Mouse Enter"]).toEqual(["hello()"]);
|
||
expect(actions["Mouse Exit"]).toEqual([
|
||
"world()",
|
||
"olleh()",
|
||
"foo()",
|
||
"dlrow()",
|
||
"oof()",
|
||
]);
|
||
expect(actions["Mouse Down"]).toEqual(["bar()"]);
|
||
});
|
||
|
||
it("should save Japanese text", async function () {
|
||
textWidgetDict.get("DR").get("Font").set("Goth", gothRefObj.ref);
|
||
textWidgetDict.set("DA", "/Goth 5 Tf");
|
||
|
||
const textWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: textWidgetRef, data: textWidgetDict },
|
||
gothRefObj,
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
textWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, {
|
||
value: "こんにちは世界の",
|
||
});
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
const utf16String =
|
||
"\x30\x53\x30\x93\x30\x6b\x30\x61\x30\x6f\x4e\x16\x75\x4c\x30\x6e";
|
||
expect(data.length).toEqual(2);
|
||
const [oldData, newData] = data;
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(newData.ref).toEqual(Ref.get(2, 0));
|
||
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Tx /DA (/Goth 5 Tf) /DR " +
|
||
"<< /Font << /Helv 314 0 R /Goth 159 0 R>>>> /Rect [0 0 32 10] " +
|
||
`/V (\xfe\xff${utf16String}) /AP << /N 2 0 R>> /M (date)>>\nendobj\n`
|
||
);
|
||
expect(newData.data).toEqual(
|
||
"2 0 obj\n<< /Subtype /Form /Resources " +
|
||
"<< /Font << /Helv 314 0 R /Goth 159 0 R>>>> /BBox [0 0 32 10] /Length 79>> stream\n" +
|
||
`/Tx BMC q BT /Goth 5 Tf 1 0 0 1 0 0 Tm 2 3.07 Td (${utf16String}) Tj ` +
|
||
"ET Q EMC\nendstream\nendobj\n"
|
||
);
|
||
});
|
||
});
|
||
|
||
describe("ButtonWidgetAnnotation", function () {
|
||
let buttonWidgetDict;
|
||
|
||
beforeEach(function () {
|
||
buttonWidgetDict = new Dict();
|
||
buttonWidgetDict.set("Type", Name.get("Annot"));
|
||
buttonWidgetDict.set("Subtype", Name.get("Widget"));
|
||
buttonWidgetDict.set("FT", Name.get("Btn"));
|
||
});
|
||
|
||
afterEach(function () {
|
||
buttonWidgetDict = null;
|
||
});
|
||
|
||
it("should handle checkboxes with export value", async function () {
|
||
buttonWidgetDict.set("V", Name.get("Checked"));
|
||
buttonWidgetDict.set("DV", Name.get("Off"));
|
||
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
|
||
normalAppearanceDict.set("Off", 0);
|
||
normalAppearanceDict.set("Checked", 1);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.checkBox).toEqual(true);
|
||
expect(data.fieldValue).toEqual("Checked");
|
||
expect(data.defaultFieldValue).toEqual("Off");
|
||
expect(data.radioButton).toEqual(false);
|
||
expect(data.exportValue).toEqual("Checked");
|
||
});
|
||
|
||
it("should handle checkboxes without export value", async function () {
|
||
buttonWidgetDict.set("V", Name.get("Checked"));
|
||
buttonWidgetDict.set("DV", Name.get("Off"));
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.checkBox).toEqual(true);
|
||
expect(data.fieldValue).toEqual("Checked");
|
||
expect(data.defaultFieldValue).toEqual("Off");
|
||
expect(data.radioButton).toEqual(false);
|
||
});
|
||
|
||
it("should handle checkboxes without /Off appearance", async function () {
|
||
buttonWidgetDict.set("V", Name.get("Checked"));
|
||
buttonWidgetDict.set("DV", Name.get("Off"));
|
||
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
|
||
normalAppearanceDict.set("Checked", 1);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.checkBox).toEqual(true);
|
||
expect(data.fieldValue).toEqual("Checked");
|
||
expect(data.defaultFieldValue).toEqual("Off");
|
||
expect(data.radioButton).toEqual(false);
|
||
expect(data.exportValue).toEqual("Checked");
|
||
});
|
||
|
||
it("should render checkbox with fallback font for printing", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
const checkedAppearanceDict = new Dict();
|
||
const uncheckedAppearanceDict = new Dict();
|
||
|
||
const checkedStream = new StringStream("/ 12 Tf (4) Tj");
|
||
checkedStream.dict = checkedAppearanceDict;
|
||
|
||
const uncheckedStream = new StringStream("");
|
||
uncheckedStream.dict = uncheckedAppearanceDict;
|
||
|
||
checkedAppearanceDict.set("BBox", [0, 0, 8, 8]);
|
||
checkedAppearanceDict.set("FormType", 1);
|
||
checkedAppearanceDict.set("Matrix", [1, 0, 0, 1, 0, 0]);
|
||
normalAppearanceDict.set("Checked", checkedStream);
|
||
normalAppearanceDict.set("Off", uncheckedStream);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
const checkboxEvaluator = partialEvaluator.clone({ ignoreErrors: true });
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: true });
|
||
|
||
const { opList } = await annotation.getOperatorList(
|
||
checkboxEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList.argsArray.length).toEqual(5);
|
||
expect(opList.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.dependency,
|
||
OPS.setFont,
|
||
OPS.showText,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList.argsArray[0]).toEqual([
|
||
"124R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList.argsArray[3][0][0].unicode).toEqual("4");
|
||
});
|
||
|
||
it("should render checkboxes for printing", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
const checkedAppearanceDict = new Dict();
|
||
const uncheckedAppearanceDict = new Dict();
|
||
|
||
const checkedStream = new StringStream("0.1 0.2 0.3 rg");
|
||
checkedStream.dict = checkedAppearanceDict;
|
||
|
||
const uncheckedStream = new StringStream("0.3 0.2 0.1 rg");
|
||
uncheckedStream.dict = uncheckedAppearanceDict;
|
||
|
||
checkedAppearanceDict.set("BBox", [0, 0, 8, 8]);
|
||
checkedAppearanceDict.set("FormType", 1);
|
||
checkedAppearanceDict.set("Matrix", [1, 0, 0, 1, 0, 0]);
|
||
normalAppearanceDict.set("Checked", checkedStream);
|
||
normalAppearanceDict.set("Off", uncheckedStream);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: true });
|
||
|
||
const { opList: opList1 } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList1.argsArray.length).toEqual(3);
|
||
expect(opList1.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList1.argsArray[0]).toEqual([
|
||
"124R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList1.argsArray[1]).toEqual(new Uint8ClampedArray([26, 51, 76]));
|
||
|
||
annotationStorage.set(annotation.data.id, { value: false });
|
||
|
||
const { opList: opList2 } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList2.argsArray.length).toEqual(3);
|
||
expect(opList2.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList2.argsArray[0]).toEqual([
|
||
"124R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList2.argsArray[1]).toEqual(new Uint8ClampedArray([76, 51, 26]));
|
||
});
|
||
|
||
it("should render checkboxes for printing twice", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
const checkedAppearanceDict = new Dict();
|
||
const uncheckedAppearanceDict = new Dict();
|
||
|
||
const checkedStream = new StringStream("0.1 0.2 0.3 rg");
|
||
checkedStream.dict = checkedAppearanceDict;
|
||
|
||
const uncheckedStream = new StringStream("0.3 0.2 0.1 rg");
|
||
uncheckedStream.dict = uncheckedAppearanceDict;
|
||
|
||
checkedAppearanceDict.set("BBox", [0, 0, 8, 8]);
|
||
checkedAppearanceDict.set("FormType", 1);
|
||
checkedAppearanceDict.set("Matrix", [1, 0, 0, 1, 0, 0]);
|
||
normalAppearanceDict.set("Checked", checkedStream);
|
||
normalAppearanceDict.set("Off", uncheckedStream);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
buttonWidgetDict.set("AS", Name.get("Off"));
|
||
|
||
const buttonWidgetRef = Ref.get(1249, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
|
||
for (let i = 0; i < 2; i++) {
|
||
annotationStorage.set(annotation.data.id, { value: true });
|
||
|
||
const { opList } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList.argsArray.length).toEqual(3);
|
||
expect(opList.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList.argsArray[0]).toEqual([
|
||
"1249R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList.argsArray[1]).toEqual(
|
||
new Uint8ClampedArray([26, 51, 76])
|
||
);
|
||
}
|
||
});
|
||
|
||
it("should render checkboxes for printing using normal appearance", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
const checkedAppearanceDict = new Dict();
|
||
const uncheckedAppearanceDict = new Dict();
|
||
|
||
const checkedStream = new StringStream("0.1 0.2 0.3 rg");
|
||
checkedStream.dict = checkedAppearanceDict;
|
||
|
||
const uncheckedStream = new StringStream("0.3 0.2 0.1 rg");
|
||
uncheckedStream.dict = uncheckedAppearanceDict;
|
||
|
||
checkedAppearanceDict.set("BBox", [0, 0, 8, 8]);
|
||
checkedAppearanceDict.set("FormType", 1);
|
||
checkedAppearanceDict.set("Matrix", [1, 0, 0, 1, 0, 0]);
|
||
normalAppearanceDict.set("Checked", checkedStream);
|
||
normalAppearanceDict.set("Off", uncheckedStream);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
buttonWidgetDict.set("AS", Name.get("Checked"));
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
|
||
const { opList } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList.argsArray.length).toEqual(3);
|
||
expect(opList.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList.argsArray[0]).toEqual([
|
||
"124R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList.argsArray[1]).toEqual(new Uint8ClampedArray([26, 51, 76]));
|
||
});
|
||
|
||
it("should save checkboxes", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
|
||
normalAppearanceDict.set("Checked", Ref.get(314, 0));
|
||
normalAppearanceDict.set("Off", Ref.get(271, 0));
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
buttonWidgetDict.set("V", Name.get("Off"));
|
||
|
||
const buttonWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: true });
|
||
|
||
const [oldData] = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Btn " +
|
||
"/AP << /N << /Checked 314 0 R /Off 271 0 R>>>> " +
|
||
"/V /Checked /AS /Checked /M (date)>>\nendobj\n"
|
||
);
|
||
|
||
annotationStorage.set(annotation.data.id, { value: false });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data).toEqual(null);
|
||
});
|
||
|
||
it("should save rotated checkboxes", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
|
||
normalAppearanceDict.set("Checked", Ref.get(314, 0));
|
||
normalAppearanceDict.set("Off", Ref.get(271, 0));
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
buttonWidgetDict.set("V", Name.get("Off"));
|
||
|
||
const buttonWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: true, rotation: 180 });
|
||
|
||
const [oldData] = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Btn " +
|
||
"/AP << /N << /Checked 314 0 R /Off 271 0 R>>>> " +
|
||
"/V /Checked /AS /Checked /M (date) /MK << /R 180>>>>\nendobj\n"
|
||
);
|
||
|
||
annotationStorage.set(annotation.data.id, { value: false });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data).toEqual(null);
|
||
});
|
||
|
||
it("should handle radio buttons with a field value", async function () {
|
||
const parentDict = new Dict();
|
||
parentDict.set("V", Name.get("1"));
|
||
|
||
const normalAppearanceStateDict = new Dict();
|
||
normalAppearanceStateDict.set("2", null);
|
||
|
||
const appearanceStatesDict = new Dict();
|
||
appearanceStatesDict.set("N", normalAppearanceStateDict);
|
||
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.RADIO);
|
||
buttonWidgetDict.set("Parent", parentDict);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.checkBox).toEqual(false);
|
||
expect(data.radioButton).toEqual(true);
|
||
expect(data.fieldValue).toEqual("1");
|
||
expect(data.buttonValue).toEqual("2");
|
||
});
|
||
|
||
it("should handle radio buttons with a field value that's not an ASCII string", async function () {
|
||
const parentDict = new Dict();
|
||
parentDict.set("V", Name.get("\x91I=\x91\xf0\x93\xe0\x97e3"));
|
||
|
||
const normalAppearanceStateDict = new Dict();
|
||
normalAppearanceStateDict.set("\x91I=\x91\xf0\x93\xe0\x97e3", null);
|
||
|
||
const appearanceStatesDict = new Dict();
|
||
appearanceStatesDict.set("N", normalAppearanceStateDict);
|
||
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.RADIO);
|
||
buttonWidgetDict.set("Parent", parentDict);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.checkBox).toEqual(false);
|
||
expect(data.radioButton).toEqual(true);
|
||
expect(data.fieldValue).toEqual("‚I=‚ðfiàŠe3");
|
||
expect(data.buttonValue).toEqual("‚I=‚ðfiàŠe3");
|
||
});
|
||
|
||
it("should handle radio buttons without a field value", async function () {
|
||
const normalAppearanceStateDict = new Dict();
|
||
normalAppearanceStateDict.set("2", null);
|
||
|
||
const appearanceStatesDict = new Dict();
|
||
appearanceStatesDict.set("N", normalAppearanceStateDict);
|
||
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.RADIO);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.checkBox).toEqual(false);
|
||
expect(data.radioButton).toEqual(true);
|
||
expect(data.fieldValue).toEqual(null);
|
||
expect(data.buttonValue).toEqual("2");
|
||
});
|
||
|
||
it("should render radio buttons for printing", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
const checkedAppearanceDict = new Dict();
|
||
const uncheckedAppearanceDict = new Dict();
|
||
|
||
const checkedStream = new StringStream("0.1 0.2 0.3 rg");
|
||
checkedStream.dict = checkedAppearanceDict;
|
||
|
||
const uncheckedStream = new StringStream("0.3 0.2 0.1 rg");
|
||
uncheckedStream.dict = uncheckedAppearanceDict;
|
||
|
||
checkedAppearanceDict.set("BBox", [0, 0, 8, 8]);
|
||
checkedAppearanceDict.set("FormType", 1);
|
||
checkedAppearanceDict.set("Matrix", [1, 0, 0, 1, 0, 0]);
|
||
normalAppearanceDict.set("Checked", checkedStream);
|
||
normalAppearanceDict.set("Off", uncheckedStream);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.RADIO);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: true });
|
||
|
||
const { opList: opList1 } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList1.argsArray.length).toEqual(3);
|
||
expect(opList1.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList1.argsArray[0]).toEqual([
|
||
"124R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList1.argsArray[1]).toEqual(new Uint8ClampedArray([26, 51, 76]));
|
||
|
||
annotationStorage.set(annotation.data.id, { value: false });
|
||
|
||
const { opList: opList2 } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList2.argsArray.length).toEqual(3);
|
||
expect(opList2.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList2.argsArray[0]).toEqual([
|
||
"124R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList2.argsArray[1]).toEqual(new Uint8ClampedArray([76, 51, 26]));
|
||
});
|
||
|
||
it("should render radio buttons for printing using normal appearance", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
const checkedAppearanceDict = new Dict();
|
||
const uncheckedAppearanceDict = new Dict();
|
||
|
||
const checkedStream = new StringStream("0.1 0.2 0.3 rg");
|
||
checkedStream.dict = checkedAppearanceDict;
|
||
|
||
const uncheckedStream = new StringStream("0.3 0.2 0.1 rg");
|
||
uncheckedStream.dict = uncheckedAppearanceDict;
|
||
|
||
checkedAppearanceDict.set("BBox", [0, 0, 8, 8]);
|
||
checkedAppearanceDict.set("FormType", 1);
|
||
checkedAppearanceDict.set("Matrix", [1, 0, 0, 1, 0, 0]);
|
||
normalAppearanceDict.set("Checked", checkedStream);
|
||
normalAppearanceDict.set("Off", uncheckedStream);
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.RADIO);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
buttonWidgetDict.set("AS", Name.get("Off"));
|
||
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
|
||
const { opList } = await annotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
annotationStorage
|
||
);
|
||
expect(opList.argsArray.length).toEqual(3);
|
||
expect(opList.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setFillRGBColor,
|
||
OPS.endAnnotation,
|
||
]);
|
||
expect(opList.argsArray[0]).toEqual([
|
||
"124R",
|
||
[0, 0, 0, 0],
|
||
[0, 0, 0, 0, 0, 0],
|
||
[1, 0, 0, 1, 0, 0],
|
||
false,
|
||
]);
|
||
expect(opList.argsArray[1]).toEqual(new Uint8ClampedArray([76, 51, 26]));
|
||
});
|
||
|
||
it("should save radio buttons", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
|
||
normalAppearanceDict.set("Checked", Ref.get(314, 0));
|
||
normalAppearanceDict.set("Off", Ref.get(271, 0));
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.RADIO);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(123, 0);
|
||
const parentRef = Ref.get(456, 0);
|
||
|
||
const parentDict = new Dict();
|
||
parentDict.set("V", Name.get("Off"));
|
||
parentDict.set("Kids", [buttonWidgetRef]);
|
||
buttonWidgetDict.set("Parent", parentRef);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
{ ref: parentRef, data: parentDict },
|
||
]);
|
||
|
||
parentDict.xref = xref;
|
||
buttonWidgetDict.xref = xref;
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: true });
|
||
|
||
let data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data.length).toEqual(2);
|
||
const [radioData, parentData] = data;
|
||
radioData.data = radioData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(radioData.ref).toEqual(Ref.get(123, 0));
|
||
expect(radioData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Btn /Ff 32768 " +
|
||
"/AP << /N << /Checked 314 0 R /Off 271 0 R>>>> " +
|
||
"/Parent 456 0 R /AS /Checked /M (date)>>\nendobj\n"
|
||
);
|
||
expect(parentData.ref).toEqual(Ref.get(456, 0));
|
||
expect(parentData.data).toEqual(
|
||
"456 0 obj\n<< /V /Checked /Kids [123 0 R]>>\nendobj\n"
|
||
);
|
||
|
||
annotationStorage.set(annotation.data.id, { value: false });
|
||
|
||
data = await annotation.save(partialEvaluator, task, annotationStorage);
|
||
expect(data).toEqual(null);
|
||
});
|
||
|
||
it("should save radio buttons without a field value", async function () {
|
||
const appearanceStatesDict = new Dict();
|
||
const normalAppearanceDict = new Dict();
|
||
|
||
normalAppearanceDict.set("Checked", Ref.get(314, 0));
|
||
normalAppearanceDict.set("Off", Ref.get(271, 0));
|
||
appearanceStatesDict.set("N", normalAppearanceDict);
|
||
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.RADIO);
|
||
buttonWidgetDict.set("AP", appearanceStatesDict);
|
||
|
||
const buttonWidgetRef = Ref.get(123, 0);
|
||
const parentRef = Ref.get(456, 0);
|
||
|
||
const parentDict = new Dict();
|
||
parentDict.set("Kids", [buttonWidgetRef]);
|
||
buttonWidgetDict.set("Parent", parentRef);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
{ ref: parentRef, data: parentDict },
|
||
]);
|
||
|
||
parentDict.xref = xref;
|
||
buttonWidgetDict.xref = xref;
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: true });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data.length).toEqual(2);
|
||
const [radioData, parentData] = data;
|
||
radioData.data = radioData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(radioData.ref).toEqual(Ref.get(123, 0));
|
||
expect(radioData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Btn /Ff 32768 " +
|
||
"/AP << /N << /Checked 314 0 R /Off 271 0 R>>>> " +
|
||
"/Parent 456 0 R /AS /Checked /M (date)>>\nendobj\n"
|
||
);
|
||
expect(parentData.ref).toEqual(Ref.get(456, 0));
|
||
expect(parentData.data).toEqual(
|
||
"456 0 obj\n<< /Kids [123 0 R] /V /Checked>>\nendobj\n"
|
||
);
|
||
});
|
||
|
||
it("should save nothing", async function () {
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data).toEqual(null);
|
||
});
|
||
|
||
it("should handle push buttons", async function () {
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.PUSHBUTTON);
|
||
|
||
const actionDict = new Dict();
|
||
actionDict.set("S", Name.get("JavaScript"));
|
||
actionDict.set("JS", "do_something();");
|
||
buttonWidgetDict.set("A", actionDict);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.pushButton).toEqual(true);
|
||
expect(data.actions.Action).toEqual(["do_something();"]);
|
||
});
|
||
|
||
it("should handle push buttons that act as a tooltip only", async function () {
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.PUSHBUTTON);
|
||
buttonWidgetDict.set("TU", "An alternative text");
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.pushButton).toEqual(true);
|
||
expect(data.alternativeText).toEqual("An alternative text");
|
||
});
|
||
|
||
it("should handle URL in A dict in push buttons", async function () {
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.PUSHBUTTON);
|
||
|
||
const actionDict = new Dict();
|
||
actionDict.set("S", Name.get("JavaScript"));
|
||
actionDict.set(
|
||
"JS",
|
||
"app.launchURL('https://developer.mozilla.org/en-US/', true)"
|
||
);
|
||
buttonWidgetDict.set("A", actionDict);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.url).toEqual("https://developer.mozilla.org/en-US/");
|
||
});
|
||
|
||
it("should handle URL in AA dict in push buttons", async function () {
|
||
const buttonWidgetRef = Ref.get(124, 0);
|
||
buttonWidgetDict.set("Ff", AnnotationFieldFlag.PUSHBUTTON);
|
||
|
||
// D stands for MouseDown.
|
||
const dDict = new Dict();
|
||
dDict.set("S", Name.get("JavaScript"));
|
||
dDict.set(
|
||
"JS",
|
||
"app.launchURL('https://developer.mozilla.org/en-US/', true)"
|
||
);
|
||
const actionDict = new Dict();
|
||
actionDict.set("D", dDict);
|
||
buttonWidgetDict.set("AA", actionDict);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: buttonWidgetRef, data: buttonWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
buttonWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.url).toEqual("https://developer.mozilla.org/en-US/");
|
||
});
|
||
});
|
||
|
||
describe("ChoiceWidgetAnnotation", function () {
|
||
let choiceWidgetDict, fontRefObj;
|
||
|
||
beforeEach(function () {
|
||
choiceWidgetDict = new Dict();
|
||
choiceWidgetDict.set("Type", Name.get("Annot"));
|
||
choiceWidgetDict.set("Subtype", Name.get("Widget"));
|
||
choiceWidgetDict.set("FT", Name.get("Ch"));
|
||
|
||
const helvDict = new Dict();
|
||
helvDict.set("BaseFont", Name.get("Helvetica"));
|
||
helvDict.set("Type", Name.get("Font"));
|
||
helvDict.set("Subtype", Name.get("Type1"));
|
||
|
||
const fontRef = Ref.get(314, 0);
|
||
fontRefObj = { ref: fontRef, data: helvDict };
|
||
const resourceDict = new Dict();
|
||
const fontDict = new Dict();
|
||
fontDict.set("Helv", fontRef);
|
||
resourceDict.set("Font", fontDict);
|
||
|
||
choiceWidgetDict.set("DA", "/Helv 5 Tf");
|
||
choiceWidgetDict.set("DR", resourceDict);
|
||
choiceWidgetDict.set("Rect", [0, 0, 32, 10]);
|
||
});
|
||
|
||
afterEach(function () {
|
||
choiceWidgetDict = fontRefObj = null;
|
||
});
|
||
|
||
it("should handle missing option arrays", async function () {
|
||
const choiceWidgetRef = Ref.get(122, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.options).toEqual([]);
|
||
});
|
||
|
||
it("should handle option arrays with array elements", async function () {
|
||
const optionBarRef = Ref.get(20, 0);
|
||
const optionBarStr = "Bar";
|
||
const optionOneRef = Ref.get(10, 0);
|
||
const optionOneArr = ["bar_export", optionBarRef];
|
||
|
||
const options = [["foo_export", "Foo"], optionOneRef];
|
||
const expected = [
|
||
{ exportValue: "foo_export", displayValue: "Foo" },
|
||
{ exportValue: "bar_export", displayValue: "Bar" },
|
||
];
|
||
|
||
choiceWidgetDict.set("Opt", options);
|
||
|
||
const choiceWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
{ ref: optionBarRef, data: optionBarStr },
|
||
{ ref: optionOneRef, data: optionOneArr },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.options).toEqual(expected);
|
||
});
|
||
|
||
it("should handle option arrays with string elements", async function () {
|
||
const optionBarRef = Ref.get(10, 0);
|
||
const optionBarStr = "Bar";
|
||
|
||
const options = ["Foo", optionBarRef];
|
||
const expected = [
|
||
{ exportValue: "Foo", displayValue: "Foo" },
|
||
{ exportValue: "Bar", displayValue: "Bar" },
|
||
];
|
||
|
||
choiceWidgetDict.set("Opt", options);
|
||
|
||
const choiceWidgetRef = Ref.get(981, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
{ ref: optionBarRef, data: optionBarStr },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.options).toEqual(expected);
|
||
});
|
||
|
||
it("should handle inherited option arrays (issue 8094)", async function () {
|
||
const options = [
|
||
["Value1", "Description1"],
|
||
["Value2", "Description2"],
|
||
];
|
||
const expected = [
|
||
{ exportValue: "Value1", displayValue: "Description1" },
|
||
{ exportValue: "Value2", displayValue: "Description2" },
|
||
];
|
||
|
||
const parentDict = new Dict();
|
||
parentDict.set("Opt", options);
|
||
|
||
choiceWidgetDict.set("Parent", parentDict);
|
||
|
||
const choiceWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.options).toEqual(expected);
|
||
});
|
||
|
||
it("should decode form values", async function () {
|
||
const encodedString = "\xFE\xFF\x00F\x00o\x00o";
|
||
const decodedString = "Foo";
|
||
|
||
choiceWidgetDict.set("Opt", [encodedString]);
|
||
choiceWidgetDict.set("V", encodedString);
|
||
choiceWidgetDict.set("DV", Name.get("foo"));
|
||
|
||
const choiceWidgetRef = Ref.get(984, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.fieldValue).toEqual([decodedString]);
|
||
expect(data.defaultFieldValue).toEqual("foo");
|
||
expect(data.options).toEqual([
|
||
{ exportValue: decodedString, displayValue: decodedString },
|
||
]);
|
||
});
|
||
|
||
it("should convert the field value to an array", async function () {
|
||
const inputs = [null, "Foo", ["Foo", "Bar"]];
|
||
const outputs = [[], ["Foo"], ["Foo", "Bar"]];
|
||
|
||
let promise = Promise.resolve();
|
||
for (let i = 0, ii = inputs.length; i < ii; i++) {
|
||
promise = promise.then(() => {
|
||
choiceWidgetDict.set("V", inputs[i]);
|
||
|
||
const choiceWidgetRef = Ref.get(968, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
]);
|
||
|
||
return AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
).then(({ data }) => {
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.fieldValue).toEqual(outputs[i]);
|
||
});
|
||
});
|
||
}
|
||
await promise;
|
||
});
|
||
|
||
it("should handle unknown flags", async function () {
|
||
const choiceWidgetRef = Ref.get(166, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.readOnly).toEqual(false);
|
||
expect(data.hidden).toEqual(false);
|
||
expect(data.combo).toEqual(false);
|
||
expect(data.multiSelect).toEqual(false);
|
||
});
|
||
|
||
it("should not set invalid flags", async function () {
|
||
choiceWidgetDict.set("Ff", "readonly");
|
||
|
||
const choiceWidgetRef = Ref.get(165, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.readOnly).toEqual(false);
|
||
expect(data.hidden).toEqual(false);
|
||
expect(data.combo).toEqual(false);
|
||
expect(data.multiSelect).toEqual(false);
|
||
});
|
||
|
||
it("should set valid flags", async function () {
|
||
choiceWidgetDict.set(
|
||
"Ff",
|
||
AnnotationFieldFlag.READONLY +
|
||
AnnotationFieldFlag.COMBO +
|
||
AnnotationFieldFlag.MULTISELECT
|
||
);
|
||
|
||
const choiceWidgetRef = Ref.get(512, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
||
expect(data.readOnly).toEqual(true);
|
||
expect(data.hidden).toEqual(false);
|
||
expect(data.combo).toEqual(true);
|
||
expect(data.multiSelect).toEqual(true);
|
||
});
|
||
|
||
it("should render choice for printing", async function () {
|
||
const choiceWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
fontRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "a value" });
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
[
|
||
"/Tx BMC q",
|
||
"1 1 32 10 re W n",
|
||
"BT",
|
||
"/Helv 5 Tf",
|
||
"1 0 0 1 0 10 Tm",
|
||
"ET Q EMC",
|
||
].join("\n")
|
||
);
|
||
});
|
||
|
||
it("should render choice with multiple selections but one is visible for printing", async function () {
|
||
choiceWidgetDict.set("Ff", AnnotationFieldFlag.MULTISELECT);
|
||
choiceWidgetDict.set("Opt", [
|
||
["A", "a"],
|
||
["B", "b"],
|
||
["C", "c"],
|
||
["D", "d"],
|
||
]);
|
||
choiceWidgetDict.set("V", ["A"]);
|
||
|
||
const choiceWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
fontRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: ["A", "C"] });
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
[
|
||
"/Tx BMC q",
|
||
"1 1 32 10 re W n",
|
||
"0.600006 0.756866 0.854904 rg",
|
||
"1 3.25 32 6.75 re f",
|
||
"BT",
|
||
"/Helv 5 Tf",
|
||
"1 0 0 1 0 10 Tm",
|
||
"2 -5.88 Td (a) Tj",
|
||
"0 -6.75 Td (b) Tj",
|
||
"ET Q EMC",
|
||
].join("\n")
|
||
);
|
||
});
|
||
|
||
it("should render choice with multiple selections for printing", async function () {
|
||
choiceWidgetDict.set("Ff", AnnotationFieldFlag.MULTISELECT);
|
||
choiceWidgetDict.set("Opt", [
|
||
["A", "a"],
|
||
["B", "b"],
|
||
["C", "c"],
|
||
["D", "d"],
|
||
]);
|
||
choiceWidgetDict.set("V", ["A"]);
|
||
|
||
const choiceWidgetRef = Ref.get(271, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
fontRefObj,
|
||
]);
|
||
const task = new WorkerTask("test print");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: ["B", "C"] });
|
||
|
||
const appearance = await annotation._getAppearance(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
annotationStorage
|
||
);
|
||
expect(appearance).toEqual(
|
||
[
|
||
"/Tx BMC q",
|
||
"1 1 32 10 re W n",
|
||
"0.600006 0.756866 0.854904 rg",
|
||
"1 3.25 32 6.75 re f",
|
||
"1 -3.5 32 6.75 re f",
|
||
"BT",
|
||
"/Helv 5 Tf",
|
||
"1 0 0 1 0 10 Tm",
|
||
"2 -5.88 Td (b) Tj",
|
||
"0 -6.75 Td (c) Tj",
|
||
"ET Q EMC",
|
||
].join("\n")
|
||
);
|
||
});
|
||
|
||
it("should save rotated choice", async function () {
|
||
choiceWidgetDict.set("Opt", ["A", "B", "C"]);
|
||
choiceWidgetDict.set("V", "A");
|
||
|
||
const choiceWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
fontRefObj,
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "C", rotation: 270 });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data.length).toEqual(2);
|
||
const [oldData, newData] = data;
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(newData.ref).toEqual(Ref.get(2, 0));
|
||
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Ch /DA (/Helv 5 Tf) /DR " +
|
||
"<< /Font << /Helv 314 0 R>>>> " +
|
||
"/Rect [0 0 32 10] /Opt [(A) (B) (C)] /V (C) " +
|
||
"/MK << /R 270>> /AP << /N 2 0 R>> /M (date)>>\nendobj\n"
|
||
);
|
||
expect(newData.data).toEqual(
|
||
[
|
||
"2 0 obj",
|
||
"<< /Subtype /Form /Resources << /Font << /Helv 314 0 R>>>> " +
|
||
"/BBox [0 0 32 10] /Matrix [0 -1 1 0 0 10] /Length 170>> stream",
|
||
"/Tx BMC q",
|
||
"1 1 10 32 re W n",
|
||
"0.600006 0.756866 0.854904 rg",
|
||
"1 11.75 10 6.75 re f",
|
||
"BT",
|
||
"/Helv 5 Tf",
|
||
"1 0 0 1 0 32 Tm",
|
||
"2 -5.88 Td (A) Tj",
|
||
"0 -6.75 Td (B) Tj",
|
||
"0 -6.75 Td (C) Tj",
|
||
"ET Q EMC",
|
||
"endstream",
|
||
"endobj\n",
|
||
].join("\n")
|
||
);
|
||
});
|
||
|
||
it("should save choice", async function () {
|
||
choiceWidgetDict.set("Opt", ["A", "B", "C"]);
|
||
choiceWidgetDict.set("V", "A");
|
||
|
||
const choiceWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
fontRefObj,
|
||
]);
|
||
partialEvaluator.xref = xref;
|
||
const task = new WorkerTask("test save");
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: "C" });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
expect(data.length).toEqual(2);
|
||
const [oldData, newData] = data;
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(newData.ref).toEqual(Ref.get(2, 0));
|
||
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Ch /DA (/Helv 5 Tf) /DR " +
|
||
"<< /Font << /Helv 314 0 R>>>> " +
|
||
"/Rect [0 0 32 10] /Opt [(A) (B) (C)] /V (C) " +
|
||
"/AP << /N 2 0 R>> /M (date)>>\nendobj\n"
|
||
);
|
||
expect(newData.data).toEqual(
|
||
[
|
||
"2 0 obj",
|
||
"<< /Subtype /Form /Resources << /Font << /Helv 314 0 R>>>> " +
|
||
"/BBox [0 0 32 10] /Length 133>> stream",
|
||
"/Tx BMC q",
|
||
"1 1 32 10 re W n",
|
||
"0.600006 0.756866 0.854904 rg",
|
||
"1 3.25 32 6.75 re f",
|
||
"BT",
|
||
"/Helv 5 Tf",
|
||
"1 0 0 1 0 10 Tm",
|
||
"2 -5.88 Td (C) Tj",
|
||
"ET Q EMC",
|
||
"endstream",
|
||
"endobj\n",
|
||
].join("\n")
|
||
);
|
||
});
|
||
|
||
it("should save choice with multiple selections", async function () {
|
||
choiceWidgetDict.set("Ff", AnnotationFieldFlag.MULTISELECT);
|
||
choiceWidgetDict.set("Opt", [
|
||
["A", "a"],
|
||
["B", "b"],
|
||
["C", "c"],
|
||
["D", "d"],
|
||
]);
|
||
choiceWidgetDict.set("V", ["A"]);
|
||
|
||
const choiceWidgetRef = Ref.get(123, 0);
|
||
const xref = new XRefMock([
|
||
{ ref: choiceWidgetRef, data: choiceWidgetDict },
|
||
fontRefObj,
|
||
]);
|
||
const task = new WorkerTask("test save");
|
||
partialEvaluator.xref = xref;
|
||
|
||
const annotation = await AnnotationFactory.create(
|
||
xref,
|
||
choiceWidgetRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
const annotationStorage = new Map();
|
||
annotationStorage.set(annotation.data.id, { value: ["B", "C"] });
|
||
|
||
const data = await annotation.save(
|
||
partialEvaluator,
|
||
task,
|
||
annotationStorage
|
||
);
|
||
|
||
expect(data.length).toEqual(2);
|
||
const [oldData, newData] = data;
|
||
expect(oldData.ref).toEqual(Ref.get(123, 0));
|
||
expect(newData.ref).toEqual(Ref.get(2, 0));
|
||
|
||
oldData.data = oldData.data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(oldData.data).toEqual(
|
||
"123 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Widget /FT /Ch /DA (/Helv 5 Tf) /DR " +
|
||
"<< /Font << /Helv 314 0 R>>>> /Rect [0 0 32 10] /Ff 2097152 /Opt " +
|
||
"[[(A) (a)] [(B) (b)] [(C) (c)] [(D) (d)]] /V [(B) (C)] /AP " +
|
||
"<< /N 2 0 R>> /M (date)>>\nendobj\n"
|
||
);
|
||
expect(newData.data).toEqual(
|
||
[
|
||
"2 0 obj",
|
||
"<< /Subtype /Form /Resources << /Font << /Helv 314 0 R>>>> " +
|
||
"/BBox [0 0 32 10] /Length 171>> stream",
|
||
"/Tx BMC q",
|
||
"1 1 32 10 re W n",
|
||
"0.600006 0.756866 0.854904 rg",
|
||
"1 3.25 32 6.75 re f",
|
||
"1 -3.5 32 6.75 re f",
|
||
"BT",
|
||
"/Helv 5 Tf",
|
||
"1 0 0 1 0 10 Tm",
|
||
"2 -5.88 Td (b) Tj",
|
||
"0 -6.75 Td (c) Tj",
|
||
"ET Q EMC",
|
||
"endstream",
|
||
"endobj\n",
|
||
].join("\n")
|
||
);
|
||
});
|
||
});
|
||
|
||
describe("LineAnnotation", function () {
|
||
it("should set the line coordinates", async function () {
|
||
const lineDict = new Dict();
|
||
lineDict.set("Type", Name.get("Annot"));
|
||
lineDict.set("Subtype", Name.get("Line"));
|
||
lineDict.set("L", [1, 2, 3, 4]);
|
||
lineDict.set("LE", ["Square", "Circle"]);
|
||
|
||
const lineRef = Ref.get(122, 0);
|
||
const xref = new XRefMock([{ ref: lineRef, data: lineDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
lineRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINE);
|
||
expect(data.lineCoordinates).toEqual([1, 2, 3, 4]);
|
||
expect(data.lineEndings).toEqual(["None", "None"]);
|
||
});
|
||
|
||
it("should set the line endings", async function () {
|
||
const lineDict = new Dict();
|
||
lineDict.set("Type", Name.get("Annot"));
|
||
lineDict.set("Subtype", Name.get("Line"));
|
||
lineDict.set("L", [1, 2, 3, 4]);
|
||
lineDict.set("LE", [Name.get("Square"), Name.get("Circle")]);
|
||
|
||
const lineRef = Ref.get(122, 0);
|
||
const xref = new XRefMock([{ ref: lineRef, data: lineDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
lineRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.LINE);
|
||
expect(data.lineCoordinates).toEqual([1, 2, 3, 4]);
|
||
expect(data.lineEndings).toEqual(["Square", "Circle"]);
|
||
});
|
||
});
|
||
|
||
describe("FileAttachmentAnnotation", function () {
|
||
it("should correctly parse a file attachment", async function () {
|
||
const fileStream = new StringStream(
|
||
"<<\n" +
|
||
"/Type /EmbeddedFile\n" +
|
||
"/Subtype /text#2Fplain\n" +
|
||
">>\n" +
|
||
"stream\n" +
|
||
"Test attachment" +
|
||
"endstream\n"
|
||
);
|
||
const parser = new Parser({
|
||
lexer: new Lexer(fileStream),
|
||
xref: null,
|
||
allowStreams: true,
|
||
});
|
||
|
||
const fileStreamRef = Ref.get(18, 0);
|
||
const fileStreamDict = parser.getObj();
|
||
|
||
const embeddedFileDict = new Dict();
|
||
embeddedFileDict.set("F", fileStreamRef);
|
||
|
||
const fileSpecRef = Ref.get(19, 0);
|
||
const fileSpecDict = new Dict();
|
||
fileSpecDict.set("Type", Name.get("Filespec"));
|
||
fileSpecDict.set("Desc", "");
|
||
fileSpecDict.set("EF", embeddedFileDict);
|
||
fileSpecDict.set("UF", "Test.txt");
|
||
|
||
const fileAttachmentRef = Ref.get(20, 0);
|
||
const fileAttachmentDict = new Dict();
|
||
fileAttachmentDict.set("Type", Name.get("Annot"));
|
||
fileAttachmentDict.set("Subtype", Name.get("FileAttachment"));
|
||
fileAttachmentDict.set("FS", fileSpecRef);
|
||
fileAttachmentDict.set("T", "Topic");
|
||
fileAttachmentDict.set("Contents", "Test.txt");
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: fileStreamRef, data: fileStreamDict },
|
||
{ ref: fileSpecRef, data: fileSpecDict },
|
||
{ ref: fileAttachmentRef, data: fileAttachmentDict },
|
||
]);
|
||
embeddedFileDict.assignXref(xref);
|
||
fileSpecDict.assignXref(xref);
|
||
fileAttachmentDict.assignXref(xref);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
fileAttachmentRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.FILEATTACHMENT);
|
||
expect(data.file.filename).toEqual("Test.txt");
|
||
expect(data.file.content).toEqual(stringToBytes("Test attachment"));
|
||
});
|
||
});
|
||
|
||
describe("PopupAnnotation", function () {
|
||
it("should inherit properties from its parent", async function () {
|
||
const parentDict = new Dict();
|
||
parentDict.set("Type", Name.get("Annot"));
|
||
parentDict.set("Subtype", Name.get("Text"));
|
||
parentDict.set("M", "D:20190423");
|
||
parentDict.set("C", [0, 0, 1]);
|
||
|
||
const popupDict = new Dict();
|
||
popupDict.set("Type", Name.get("Annot"));
|
||
popupDict.set("Subtype", Name.get("Popup"));
|
||
popupDict.set("Parent", parentDict);
|
||
|
||
const popupRef = Ref.get(13, 0);
|
||
const xref = new XRefMock([{ ref: popupRef, data: popupDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
popupRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.POPUP);
|
||
expect(data.modificationDate).toEqual("D:20190423");
|
||
expect(data.color).toEqual(new Uint8ClampedArray([0, 0, 255]));
|
||
});
|
||
|
||
it("should handle missing parent properties", async function () {
|
||
const parentDict = new Dict();
|
||
parentDict.set("Type", Name.get("Annot"));
|
||
parentDict.set("Subtype", Name.get("Text"));
|
||
|
||
const popupDict = new Dict();
|
||
popupDict.set("Type", Name.get("Annot"));
|
||
popupDict.set("Subtype", Name.get("Popup"));
|
||
popupDict.set("Parent", parentDict);
|
||
|
||
const popupRef = Ref.get(13, 0);
|
||
const xref = new XRefMock([{ ref: popupRef, data: popupDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
popupRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.POPUP);
|
||
expect(data.modificationDate).toEqual(null);
|
||
expect(data.color).toEqual(null);
|
||
});
|
||
|
||
it(
|
||
"should inherit the parent flags when the Popup is not viewable, " +
|
||
"but the parent is (PR 7352)",
|
||
async function () {
|
||
const parentDict = new Dict();
|
||
parentDict.set("Type", Name.get("Annot"));
|
||
parentDict.set("Subtype", Name.get("Text"));
|
||
parentDict.set("F", 28); // viewable
|
||
|
||
const popupDict = new Dict();
|
||
popupDict.set("Type", Name.get("Annot"));
|
||
popupDict.set("Subtype", Name.get("Popup"));
|
||
popupDict.set("F", 25); // not viewable
|
||
popupDict.set("Parent", parentDict);
|
||
|
||
const popupRef = Ref.get(13, 0);
|
||
const xref = new XRefMock([{ ref: popupRef, data: popupDict }]);
|
||
|
||
const { data, viewable } = await AnnotationFactory.create(
|
||
xref,
|
||
popupRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.POPUP);
|
||
// We should not modify the `annotationFlags` returned through
|
||
// e.g., the API.
|
||
expect(data.annotationFlags).toEqual(25);
|
||
// The popup should inherit the `viewable` property of the parent.
|
||
expect(viewable).toEqual(true);
|
||
}
|
||
);
|
||
|
||
it(
|
||
"should correctly inherit Contents from group-master annotation " +
|
||
"if parent has ReplyType == Group",
|
||
async function () {
|
||
const annotationRef = Ref.get(819, 0);
|
||
const annotationDict = new Dict();
|
||
annotationDict.set("Type", Name.get("Annot"));
|
||
annotationDict.set("Subtype", Name.get("Text"));
|
||
annotationDict.set("T", "Correct Title");
|
||
annotationDict.set("Contents", "Correct Text");
|
||
annotationDict.set("M", "D:20190423");
|
||
annotationDict.set("C", [0, 0, 1]);
|
||
|
||
const replyRef = Ref.get(820, 0);
|
||
const replyDict = new Dict();
|
||
replyDict.set("Type", Name.get("Annot"));
|
||
replyDict.set("Subtype", Name.get("Text"));
|
||
replyDict.set("IRT", annotationRef);
|
||
replyDict.set("RT", Name.get("Group"));
|
||
replyDict.set("T", "Reply Title");
|
||
replyDict.set("Contents", "Reply Text");
|
||
replyDict.set("M", "D:20190523");
|
||
replyDict.set("C", [0.4]);
|
||
|
||
const popupRef = Ref.get(821, 0);
|
||
const popupDict = new Dict();
|
||
popupDict.set("Type", Name.get("Annot"));
|
||
popupDict.set("Subtype", Name.get("Popup"));
|
||
popupDict.set("T", "Wrong Title");
|
||
popupDict.set("Contents", "Wrong Text");
|
||
popupDict.set("Parent", replyRef);
|
||
popupDict.set("M", "D:20190623");
|
||
popupDict.set("C", [0.8]);
|
||
replyDict.set("Popup", popupRef);
|
||
|
||
const xref = new XRefMock([
|
||
{ ref: annotationRef, data: annotationDict },
|
||
{ ref: replyRef, data: replyDict },
|
||
{ ref: popupRef, data: popupDict },
|
||
]);
|
||
annotationDict.assignXref(xref);
|
||
popupDict.assignXref(xref);
|
||
replyDict.assignXref(xref);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
popupRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.titleObj).toEqual({ str: "Correct Title", dir: "ltr" });
|
||
expect(data.contentsObj).toEqual({ str: "Correct Text", dir: "ltr" });
|
||
expect(data.modificationDate).toEqual("D:20190423");
|
||
expect(data.color).toEqual(new Uint8ClampedArray([0, 0, 255]));
|
||
}
|
||
);
|
||
});
|
||
|
||
describe("FreeTextAnnotation", () => {
|
||
it("should create a new FreeText annotation", async () => {
|
||
partialEvaluator.xref = new XRefMock();
|
||
const task = new WorkerTask("test FreeText creation");
|
||
const data = await AnnotationFactory.saveNewAnnotations(
|
||
partialEvaluator,
|
||
task,
|
||
[
|
||
{
|
||
annotationType: AnnotationEditorType.FREETEXT,
|
||
rect: [12, 34, 56, 78],
|
||
rotation: 0,
|
||
fontSize: 10,
|
||
color: [0, 0, 0],
|
||
value: "Hello PDF.js World!",
|
||
},
|
||
]
|
||
);
|
||
|
||
const base = data.annotations[0].data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(base).toEqual(
|
||
"2 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /FreeText /CreationDate (date) " +
|
||
"/Rect [12 34 56 78] /DA (/Helv 10 Tf 0 g) /Contents (Hello PDF.js World!) " +
|
||
"/F 4 /Border [0 0 0] /Rotate 0 /AP << /N 3 0 R>>>>\n" +
|
||
"endobj\n"
|
||
);
|
||
|
||
const font = data.dependencies[0].data;
|
||
expect(font).toEqual(
|
||
"1 0 obj\n" +
|
||
"<< /BaseFont /Helvetica /Type /Font /Subtype /Type1 /Encoding " +
|
||
"/WinAnsiEncoding>>\n" +
|
||
"endobj\n"
|
||
);
|
||
|
||
const appearance = data.dependencies[1].data;
|
||
expect(appearance).toEqual(
|
||
"3 0 obj\n" +
|
||
"<< /FormType 1 /Subtype /Form /Type /XObject /BBox [0 0 44 44] " +
|
||
"/Resources << /Font << /Helv 1 0 R>>>> /Length 101>> stream\n" +
|
||
"q\n" +
|
||
"0 0 44 44 re W n\n" +
|
||
"BT\n" +
|
||
"1 0 0 1 0 47.5 Tm 0 Tc 0 g\n" +
|
||
"/Helv 10 Tf\n" +
|
||
"0 -13.5 Td (Hello PDF.js World!) Tj\n" +
|
||
"ET\n" +
|
||
"Q\n" +
|
||
"endstream\n" +
|
||
"endobj\n"
|
||
);
|
||
});
|
||
|
||
it("should render an added FreeText annotation for printing", async function () {
|
||
partialEvaluator.xref = new XRefMock();
|
||
const task = new WorkerTask("test FreeText printing");
|
||
const freetextAnnotation = (
|
||
await AnnotationFactory.printNewAnnotations(partialEvaluator, task, [
|
||
{
|
||
annotationType: AnnotationEditorType.FREETEXT,
|
||
rect: [12, 34, 56, 78],
|
||
rotation: 0,
|
||
fontSize: 10,
|
||
color: [0, 0, 0],
|
||
value: "A",
|
||
},
|
||
])
|
||
)[0];
|
||
|
||
const { opList } = await freetextAnnotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
null
|
||
);
|
||
|
||
expect(opList.fnArray.length).toEqual(16);
|
||
expect(opList.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.save,
|
||
OPS.constructPath,
|
||
OPS.clip,
|
||
OPS.endPath,
|
||
OPS.beginText,
|
||
OPS.setTextMatrix,
|
||
OPS.setCharSpacing,
|
||
OPS.setFillRGBColor,
|
||
OPS.dependency,
|
||
OPS.setFont,
|
||
OPS.moveText,
|
||
OPS.showText,
|
||
OPS.endText,
|
||
OPS.restore,
|
||
OPS.endAnnotation,
|
||
]);
|
||
});
|
||
|
||
it("should extract the text from a FreeText annotation", async function () {
|
||
partialEvaluator.xref = new XRefMock();
|
||
const task = new WorkerTask("test FreeText text extraction");
|
||
const freetextAnnotation = (
|
||
await AnnotationFactory.printNewAnnotations(partialEvaluator, task, [
|
||
{
|
||
annotationType: AnnotationEditorType.FREETEXT,
|
||
rect: [12, 34, 56, 78],
|
||
rotation: 0,
|
||
fontSize: 10,
|
||
color: [0, 0, 0],
|
||
value: "Hello PDF.js\nWorld !",
|
||
},
|
||
])
|
||
)[0];
|
||
|
||
await freetextAnnotation.extractTextContent(partialEvaluator, task, [
|
||
-Infinity,
|
||
-Infinity,
|
||
Infinity,
|
||
Infinity,
|
||
]);
|
||
|
||
expect(freetextAnnotation.data.textContent).toEqual([
|
||
"Hello PDF.js",
|
||
"World !",
|
||
]);
|
||
});
|
||
});
|
||
|
||
describe("InkAnnotation", function () {
|
||
it("should handle a single ink list", async function () {
|
||
const inkDict = new Dict();
|
||
inkDict.set("Type", Name.get("Annot"));
|
||
inkDict.set("Subtype", Name.get("Ink"));
|
||
inkDict.set("InkList", [[1, 1, 1, 2, 2, 2, 3, 3]]);
|
||
|
||
const inkRef = Ref.get(142, 0);
|
||
const xref = new XRefMock([{ ref: inkRef, data: inkDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
inkRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.INK);
|
||
expect(data.inkLists.length).toEqual(1);
|
||
expect(data.inkLists[0]).toEqual([
|
||
{ x: 1, y: 1 },
|
||
{ x: 1, y: 2 },
|
||
{ x: 2, y: 2 },
|
||
{ x: 3, y: 3 },
|
||
]);
|
||
});
|
||
|
||
it("should handle multiple ink lists", async function () {
|
||
const inkDict = new Dict();
|
||
inkDict.set("Type", Name.get("Annot"));
|
||
inkDict.set("Subtype", Name.get("Ink"));
|
||
inkDict.set("InkList", [
|
||
[1, 1, 1, 2],
|
||
[3, 3, 4, 5],
|
||
]);
|
||
|
||
const inkRef = Ref.get(143, 0);
|
||
const xref = new XRefMock([{ ref: inkRef, data: inkDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
inkRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.INK);
|
||
expect(data.inkLists.length).toEqual(2);
|
||
expect(data.inkLists[0]).toEqual([
|
||
{ x: 1, y: 1 },
|
||
{ x: 1, y: 2 },
|
||
]);
|
||
expect(data.inkLists[1]).toEqual([
|
||
{ x: 3, y: 3 },
|
||
{ x: 4, y: 5 },
|
||
]);
|
||
});
|
||
|
||
it("should create a new Ink annotation", async function () {
|
||
partialEvaluator.xref = new XRefMock();
|
||
const task = new WorkerTask("test Ink creation");
|
||
const data = await AnnotationFactory.saveNewAnnotations(
|
||
partialEvaluator,
|
||
task,
|
||
[
|
||
{
|
||
annotationType: AnnotationEditorType.INK,
|
||
rect: [12, 34, 56, 78],
|
||
rotation: 0,
|
||
thickness: 1,
|
||
opacity: 1,
|
||
color: [0, 0, 0],
|
||
paths: [
|
||
{
|
||
bezier: [
|
||
10, 11, 12, 13, 14, 15, 16, 17, 22, 23, 24, 25, 26, 27,
|
||
],
|
||
points: [1, 2, 3, 4, 5, 6, 7, 8],
|
||
},
|
||
{
|
||
bezier: [
|
||
910, 911, 912, 913, 914, 915, 916, 917, 922, 923, 924, 925,
|
||
926, 927,
|
||
],
|
||
points: [91, 92, 93, 94, 95, 96, 97, 98],
|
||
},
|
||
],
|
||
},
|
||
]
|
||
);
|
||
|
||
const base = data.annotations[0].data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(base).toEqual(
|
||
"1 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Ink /CreationDate (date) /Rect [12 34 56 78] " +
|
||
"/InkList [[1 2 3 4 5 6 7 8] [91 92 93 94 95 96 97 98]] /F 4 /Border [0 0 0] " +
|
||
"/Rotate 0 /AP << /N 2 0 R>>>>\n" +
|
||
"endobj\n"
|
||
);
|
||
|
||
const appearance = data.dependencies[0].data;
|
||
expect(appearance).toEqual(
|
||
"2 0 obj\n" +
|
||
"<< /FormType 1 /Subtype /Form /Type /XObject /BBox [0 0 44 44] /Length 129>> stream\n" +
|
||
"1 w 1 J 1 j\n" +
|
||
"0 G\n" +
|
||
"10 11 m\n" +
|
||
"12 13 14 15 16 17 c\n" +
|
||
"22 23 24 25 26 27 c\n" +
|
||
"S\n" +
|
||
"910 911 m\n" +
|
||
"912 913 914 915 916 917 c\n" +
|
||
"922 923 924 925 926 927 c\n" +
|
||
"S\n" +
|
||
"endstream\n" +
|
||
"endobj\n"
|
||
);
|
||
});
|
||
|
||
it("should create a new Ink annotation with some transparency", async function () {
|
||
partialEvaluator.xref = new XRefMock();
|
||
const task = new WorkerTask("test Ink creation");
|
||
const data = await AnnotationFactory.saveNewAnnotations(
|
||
partialEvaluator,
|
||
task,
|
||
[
|
||
{
|
||
annotationType: AnnotationEditorType.INK,
|
||
rect: [12, 34, 56, 78],
|
||
rotation: 0,
|
||
thickness: 1,
|
||
opacity: 0.12,
|
||
color: [0, 0, 0],
|
||
paths: [
|
||
{
|
||
bezier: [
|
||
10, 11, 12, 13, 14, 15, 16, 17, 22, 23, 24, 25, 26, 27,
|
||
],
|
||
points: [1, 2, 3, 4, 5, 6, 7, 8],
|
||
},
|
||
{
|
||
bezier: [
|
||
910, 911, 912, 913, 914, 915, 916, 917, 922, 923, 924, 925,
|
||
926, 927,
|
||
],
|
||
points: [91, 92, 93, 94, 95, 96, 97, 98],
|
||
},
|
||
],
|
||
},
|
||
]
|
||
);
|
||
|
||
const base = data.annotations[0].data.replace(/\(D:\d+\)/, "(date)");
|
||
expect(base).toEqual(
|
||
"1 0 obj\n" +
|
||
"<< /Type /Annot /Subtype /Ink /CreationDate (date) /Rect [12 34 56 78] " +
|
||
"/InkList [[1 2 3 4 5 6 7 8] [91 92 93 94 95 96 97 98]] /F 4 /Border [0 0 0] " +
|
||
"/Rotate 0 /AP << /N 2 0 R>>>>\n" +
|
||
"endobj\n"
|
||
);
|
||
|
||
const appearance = data.dependencies[0].data;
|
||
expect(appearance).toEqual(
|
||
"2 0 obj\n" +
|
||
"<< /FormType 1 /Subtype /Form /Type /XObject /BBox [0 0 44 44] /Length 136 /Resources " +
|
||
"<< /ExtGState << /R0 << /CA 0.12 /Type /ExtGState>>>>>>>> stream\n" +
|
||
"1 w 1 J 1 j\n" +
|
||
"0 G\n" +
|
||
"/R0 gs\n" +
|
||
"10 11 m\n" +
|
||
"12 13 14 15 16 17 c\n" +
|
||
"22 23 24 25 26 27 c\n" +
|
||
"S\n" +
|
||
"910 911 m\n" +
|
||
"912 913 914 915 916 917 c\n" +
|
||
"922 923 924 925 926 927 c\n" +
|
||
"S\n" +
|
||
"endstream\n" +
|
||
"endobj\n"
|
||
);
|
||
});
|
||
|
||
it("should render an added Ink annotation for printing", async function () {
|
||
partialEvaluator.xref = new XRefMock();
|
||
const task = new WorkerTask("test Ink printing");
|
||
const inkAnnotation = (
|
||
await AnnotationFactory.printNewAnnotations(partialEvaluator, task, [
|
||
{
|
||
annotationType: AnnotationEditorType.INK,
|
||
rect: [12, 34, 56, 78],
|
||
rotation: 0,
|
||
thickness: 3,
|
||
opacity: 1,
|
||
color: [0, 255, 0],
|
||
paths: [
|
||
{
|
||
bezier: [1, 2, 3, 4, 5, 6, 7, 8],
|
||
// Useless in the printing case.
|
||
points: [1, 2, 3, 4, 5, 6, 7, 8],
|
||
},
|
||
],
|
||
},
|
||
])
|
||
)[0];
|
||
|
||
const { opList } = await inkAnnotation.getOperatorList(
|
||
partialEvaluator,
|
||
task,
|
||
RenderingIntentFlag.PRINT,
|
||
false,
|
||
null
|
||
);
|
||
|
||
expect(opList.argsArray.length).toEqual(8);
|
||
expect(opList.fnArray).toEqual([
|
||
OPS.beginAnnotation,
|
||
OPS.setLineWidth,
|
||
OPS.setLineCap,
|
||
OPS.setLineJoin,
|
||
OPS.setStrokeRGBColor,
|
||
OPS.constructPath,
|
||
OPS.stroke,
|
||
OPS.endAnnotation,
|
||
]);
|
||
|
||
// Linewidth.
|
||
expect(opList.argsArray[1]).toEqual([3]);
|
||
// LineCap.
|
||
expect(opList.argsArray[2]).toEqual([1]);
|
||
// LineJoin.
|
||
expect(opList.argsArray[3]).toEqual([1]);
|
||
// Color.
|
||
expect(opList.argsArray[4]).toEqual(new Uint8ClampedArray([0, 255, 0]));
|
||
// Path.
|
||
expect(opList.argsArray[5][0]).toEqual([OPS.moveTo, OPS.curveTo]);
|
||
expect(opList.argsArray[5][1]).toEqual([1, 2, 3, 4, 5, 6, 7, 8]);
|
||
// Min-max.
|
||
expect(opList.argsArray[5][2]).toEqual([1, 1, 2, 2]);
|
||
});
|
||
});
|
||
|
||
describe("HightlightAnnotation", function () {
|
||
it("should set quadpoints to null if not defined", async function () {
|
||
const highlightDict = new Dict();
|
||
highlightDict.set("Type", Name.get("Annot"));
|
||
highlightDict.set("Subtype", Name.get("Highlight"));
|
||
|
||
const highlightRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: highlightRef, data: highlightDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
highlightRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.HIGHLIGHT);
|
||
expect(data.quadPoints).toEqual(null);
|
||
});
|
||
|
||
it("should set quadpoints if defined", async function () {
|
||
const highlightDict = new Dict();
|
||
highlightDict.set("Type", Name.get("Annot"));
|
||
highlightDict.set("Subtype", Name.get("Highlight"));
|
||
highlightDict.set("Rect", [10, 10, 20, 20]);
|
||
highlightDict.set("QuadPoints", [10, 20, 20, 20, 10, 10, 20, 10]);
|
||
|
||
const highlightRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: highlightRef, data: highlightDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
highlightRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.HIGHLIGHT);
|
||
expect(data.quadPoints).toEqual([
|
||
[
|
||
{ x: 10, y: 20 },
|
||
{ x: 20, y: 20 },
|
||
{ x: 10, y: 10 },
|
||
{ x: 20, y: 10 },
|
||
],
|
||
]);
|
||
});
|
||
|
||
it("should set quadpoints to null when empty", async function () {
|
||
const highlightDict = new Dict();
|
||
highlightDict.set("Type", Name.get("Annot"));
|
||
highlightDict.set("Subtype", Name.get("Highlight"));
|
||
highlightDict.set("Rect", [10, 10, 20, 20]);
|
||
highlightDict.set("QuadPoints", []);
|
||
|
||
const highlightRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: highlightRef, data: highlightDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
highlightRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.HIGHLIGHT);
|
||
expect(data.quadPoints).toEqual(null);
|
||
});
|
||
});
|
||
|
||
describe("UnderlineAnnotation", function () {
|
||
it("should set quadpoints to null if not defined", async function () {
|
||
const underlineDict = new Dict();
|
||
underlineDict.set("Type", Name.get("Annot"));
|
||
underlineDict.set("Subtype", Name.get("Underline"));
|
||
|
||
const underlineRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: underlineRef, data: underlineDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
underlineRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.UNDERLINE);
|
||
expect(data.quadPoints).toEqual(null);
|
||
});
|
||
|
||
it("should set quadpoints if defined", async function () {
|
||
const underlineDict = new Dict();
|
||
underlineDict.set("Type", Name.get("Annot"));
|
||
underlineDict.set("Subtype", Name.get("Underline"));
|
||
underlineDict.set("Rect", [10, 10, 20, 20]);
|
||
underlineDict.set("QuadPoints", [10, 20, 20, 20, 10, 10, 20, 10]);
|
||
|
||
const underlineRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: underlineRef, data: underlineDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
underlineRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.UNDERLINE);
|
||
expect(data.quadPoints).toEqual([
|
||
[
|
||
{ x: 10, y: 20 },
|
||
{ x: 20, y: 20 },
|
||
{ x: 10, y: 10 },
|
||
{ x: 20, y: 10 },
|
||
],
|
||
]);
|
||
});
|
||
});
|
||
|
||
describe("SquigglyAnnotation", function () {
|
||
it("should set quadpoints to null if not defined", async function () {
|
||
const squigglyDict = new Dict();
|
||
squigglyDict.set("Type", Name.get("Annot"));
|
||
squigglyDict.set("Subtype", Name.get("Squiggly"));
|
||
|
||
const squigglyRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: squigglyRef, data: squigglyDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
squigglyRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.SQUIGGLY);
|
||
expect(data.quadPoints).toEqual(null);
|
||
});
|
||
|
||
it("should set quadpoints if defined", async function () {
|
||
const squigglyDict = new Dict();
|
||
squigglyDict.set("Type", Name.get("Annot"));
|
||
squigglyDict.set("Subtype", Name.get("Squiggly"));
|
||
squigglyDict.set("Rect", [10, 10, 20, 20]);
|
||
squigglyDict.set("QuadPoints", [10, 20, 20, 20, 10, 10, 20, 10]);
|
||
|
||
const squigglyRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: squigglyRef, data: squigglyDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
squigglyRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.SQUIGGLY);
|
||
expect(data.quadPoints).toEqual([
|
||
[
|
||
{ x: 10, y: 20 },
|
||
{ x: 20, y: 20 },
|
||
{ x: 10, y: 10 },
|
||
{ x: 20, y: 10 },
|
||
],
|
||
]);
|
||
});
|
||
});
|
||
|
||
describe("StrikeOutAnnotation", function () {
|
||
it("should set quadpoints to null if not defined", async function () {
|
||
const strikeOutDict = new Dict();
|
||
strikeOutDict.set("Type", Name.get("Annot"));
|
||
strikeOutDict.set("Subtype", Name.get("StrikeOut"));
|
||
|
||
const strikeOutRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: strikeOutRef, data: strikeOutDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
strikeOutRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.STRIKEOUT);
|
||
expect(data.quadPoints).toEqual(null);
|
||
});
|
||
|
||
it("should set quadpoints if defined", async function () {
|
||
const strikeOutDict = new Dict();
|
||
strikeOutDict.set("Type", Name.get("Annot"));
|
||
strikeOutDict.set("Subtype", Name.get("StrikeOut"));
|
||
strikeOutDict.set("Rect", [10, 10, 20, 20]);
|
||
strikeOutDict.set("QuadPoints", [10, 20, 20, 20, 10, 10, 20, 10]);
|
||
|
||
const strikeOutRef = Ref.get(121, 0);
|
||
const xref = new XRefMock([{ ref: strikeOutRef, data: strikeOutDict }]);
|
||
|
||
const { data } = await AnnotationFactory.create(
|
||
xref,
|
||
strikeOutRef,
|
||
pdfManagerMock,
|
||
idFactoryMock
|
||
);
|
||
expect(data.annotationType).toEqual(AnnotationType.STRIKEOUT);
|
||
expect(data.quadPoints).toEqual([
|
||
[
|
||
{ x: 10, y: 20 },
|
||
{ x: 20, y: 20 },
|
||
{ x: 10, y: 10 },
|
||
{ x: 20, y: 10 },
|
||
],
|
||
]);
|
||
});
|
||
});
|
||
});
|