| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086 | /* 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. */'use strict';var corePrimitives = require('../../core/primitives.js');var coreAnnotation = require('../../core/annotation.js');var coreStream = require('../../core/stream.js');var coreParser = require('../../core/parser.js');var sharedUtil = require('../../shared/util.js');var Annotation = coreAnnotation.Annotation;var AnnotationBorderStyle = coreAnnotation.AnnotationBorderStyle;var AnnotationFactory = coreAnnotation.AnnotationFactory;var Lexer = coreParser.Lexer;var Parser = coreParser.Parser;var isRef = corePrimitives.isRef;var Dict = corePrimitives.Dict;var Name = corePrimitives.Name;var Ref = corePrimitives.Ref;var StringStream = coreStream.StringStream;var AnnotationType = sharedUtil.AnnotationType;var AnnotationFlag = sharedUtil.AnnotationFlag;var AnnotationBorderStyleType = sharedUtil.AnnotationBorderStyleType;var AnnotationFieldFlag = sharedUtil.AnnotationFieldFlag;var stringToBytes = sharedUtil.stringToBytes;var stringToUTF8String = sharedUtil.stringToUTF8String;describe('annotation', function () {  function XRefMock(array) {    this.map = Object.create(null);    for (var elem in array) {      var obj = array[elem];      var ref = obj.ref,          data = obj.data;      this.map[ref.toString()] = data;    }  }  XRefMock.prototype = {    fetch: function (ref) {      return this.map[ref.toString()];    },    fetchIfRef: function (obj) {      if (!isRef(obj)) {        return obj;      }      return this.fetch(obj);    }  };  function PDFManagerMock(params) {    this.docBaseUrl = params.docBaseUrl || null;  }  PDFManagerMock.prototype = {};  function IdFactoryMock(params) {    var uniquePrefix = params.prefix || 'p0_';    var idCounters = { obj: params.startObjId || 0 };    return {      createObjId: function () {        return uniquePrefix + ++idCounters.obj;      }    };  }  IdFactoryMock.prototype = {};  var annotationFactory, pdfManagerMock, idFactoryMock;  beforeAll(function (done) {    annotationFactory = new AnnotationFactory();    pdfManagerMock = new PDFManagerMock({ docBaseUrl: null });    idFactoryMock = new IdFactoryMock({});    done();  });  afterAll(function () {    annotationFactory = null;    pdfManagerMock = null;    idFactoryMock = null;  });  describe('AnnotationFactory', function () {    it('should get id for annotation', function () {      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      var annotationRef = new Ref(10, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.LINK);      expect(data.id).toEqual('10R');    });    it('should handle, and get fallback id\'s for, annotations that are not ' + 'indirect objects (issue 7569)', function () {      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      var xref = new XRefMock();      var idFactory = new IdFactoryMock({        prefix: 'p0_',        startObjId: 0      });      var annotation1 = annotationFactory.create(xref, annotationDict, pdfManagerMock, idFactory);      var annotation2 = annotationFactory.create(xref, annotationDict, pdfManagerMock, idFactory);      var data1 = annotation1.data,          data2 = annotation2.data;      expect(data1.annotationType).toEqual(AnnotationType.LINK);      expect(data2.annotationType).toEqual(AnnotationType.LINK);      expect(data1.id).toEqual('annot_p0_1');      expect(data2.id).toEqual('annot_p0_2');    });    it('should handle missing /Subtype', function () {      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      var annotationRef = new Ref(1, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toBeUndefined();    });  });  describe('Annotation', function () {    var dict, ref;    beforeAll(function (done) {      dict = new Dict();      ref = new Ref(1, 0);      done();    });    afterAll(function () {      dict = ref = null;    });    it('should set and get flags', function () {      var annotation = new Annotation({        dict: dict,        ref: 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);    });    it('should be viewable and not printable by default', function () {      var annotation = new Annotation({        dict: dict,        ref: ref      });      expect(annotation.viewable).toEqual(true);      expect(annotation.printable).toEqual(false);    });    it('should set and get a valid rectangle', function () {      var annotation = new Annotation({        dict: dict,        ref: 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 () {      var annotation = new Annotation({        dict: dict,        ref: 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 () {      var annotation = new Annotation({        dict: dict,        ref: ref      });      annotation.setColor('red');      expect(annotation.color).toEqual(new Uint8Array([0, 0, 0]));    });    it('should set and get a transparent color', function () {      var annotation = new Annotation({        dict: dict,        ref: ref      });      annotation.setColor([]);      expect(annotation.color).toEqual(null);    });    it('should set and get a grayscale color', function () {      var annotation = new Annotation({        dict: dict,        ref: ref      });      annotation.setColor([0.4]);      expect(annotation.color).toEqual(new Uint8Array([102, 102, 102]));    });    it('should set and get an RGB color', function () {      var annotation = new Annotation({        dict: dict,        ref: ref      });      annotation.setColor([0, 0, 1]);      expect(annotation.color).toEqual(new Uint8Array([0, 0, 255]));    });    it('should set and get a CMYK color', function () {      var annotation = new Annotation({        dict: dict,        ref: ref      });      annotation.setColor([0.1, 0.92, 0.84, 0.02]);      expect(annotation.color).toEqual(new Uint8Array([233, 59, 47]));    });    it('should not set and get an invalid color', function () {      var annotation = new Annotation({        dict: dict,        ref: ref      });      annotation.setColor([0.4, 0.6]);      expect(annotation.color).toEqual(new Uint8Array([0, 0, 0]));    });  });  describe('AnnotationBorderStyle', function () {    it('should set and get a valid width', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setWidth(3);      expect(borderStyle.width).toEqual(3);    });    it('should not set and get an invalid width', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setWidth('three');      expect(borderStyle.width).toEqual(1);    });    it('should set and get a valid style', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setStyle(Name.get('D'));      expect(borderStyle.style).toEqual(AnnotationBorderStyleType.DASHED);    });    it('should not set and get an invalid style', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setStyle('Dashed');      expect(borderStyle.style).toEqual(AnnotationBorderStyleType.SOLID);    });    it('should set and get a valid dash array', function () {      var 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 () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setDashArray([0, 0]);      expect(borderStyle.dashArray).toEqual([3]);    });    it('should set and get a valid horizontal corner radius', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setHorizontalCornerRadius(3);      expect(borderStyle.horizontalCornerRadius).toEqual(3);    });    it('should not set and get an invalid horizontal corner radius', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setHorizontalCornerRadius('three');      expect(borderStyle.horizontalCornerRadius).toEqual(0);    });    it('should set and get a valid vertical corner radius', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setVerticalCornerRadius(3);      expect(borderStyle.verticalCornerRadius).toEqual(3);    });    it('should not set and get an invalid horizontal corner radius', function () {      var borderStyle = new AnnotationBorderStyle();      borderStyle.setVerticalCornerRadius('three');      expect(borderStyle.verticalCornerRadius).toEqual(0);    });  });  describe('LinkAnnotation', function () {    it('should correctly parse a URI action', function () {      var 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');      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(820, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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', function () {      var actionDict = new Dict();      actionDict.set('Type', Name.get('Action'));      actionDict.set('S', Name.get('URI'));      actionDict.set('URI', 'www.hmrc.gov.uk');      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(353, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.LINK);      expect(data.url).toEqual('http://www.hmrc.gov.uk/');      expect(data.unsafeUrl).toEqual('http://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)', function () {      var actionStream = new StringStream('<<\n' + '/Type /Action\n' + '/S /URI\n' + '/URI (http://www.example.com/\\303\\274\\303\\266\\303\\244)\n' + '>>\n');      var lexer = new Lexer(actionStream);      var parser = new Parser(lexer);      var actionDict = parser.getObj();      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(8, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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(stringToUTF8String('http://www.example.com/\xC3\xBC\xC3\xB6\xC3\xA4'));      expect(data.dest).toBeUndefined();    });    it('should correctly parse a GoTo action', function () {      var actionDict = new Dict();      actionDict.set('Type', Name.get('Action'));      actionDict.set('S', Name.get('GoTo'));      actionDict.set('D', 'page.157');      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(798, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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', function () {      var 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);      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(489, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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', function () {      var 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');      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(489, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var pdfManager = new PDFManagerMock({ docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf' });      var annotation = annotationFactory.create(xref, annotationRef, pdfManager, idFactoryMock);      var data = annotation.data;      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', function () {      var 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');      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(495, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.LINK);      expect(data.url).toEqual('http://www.example.com/test.pdf#nameddest=15');      expect(data.unsafeUrl).toEqual('http://www.example.com/test.pdf#nameddest=15');      expect(data.dest).toBeUndefined();      expect(data.newWindow).toBeFalsy();    });    it('should correctly parse a GoToR action, with explicit destination array', function () {      var 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]);      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(489, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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', function () {      var 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');      var actionDict = new Dict();      actionDict.set('Type', Name.get('Action'));      actionDict.set('S', Name.get('Launch'));      actionDict.set('F', fileSpecDict);      actionDict.set('NewWindow', true);      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(88, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var pdfManager = new PDFManagerMock({ docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf' });      var annotation = annotationFactory.create(xref, annotationRef, pdfManager, idFactoryMock);      var data = annotation.data;      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', function () {      function checkJsAction(params) {        var jsEntry = params.jsEntry;        var expectedUrl = params.expectedUrl;        var expectedUnsafeUrl = params.expectedUnsafeUrl;        var expectedNewWindow = params.expectedNewWindow;        var actionDict = new Dict();        actionDict.set('Type', Name.get('Action'));        actionDict.set('S', Name.get('JavaScript'));        actionDict.set('JS', jsEntry);        var annotationDict = new Dict();        annotationDict.set('Type', Name.get('Annot'));        annotationDict.set('Subtype', Name.get('Link'));        annotationDict.set('A', actionDict);        var annotationRef = new Ref(46, 0);        var xref = new XRefMock([{          ref: annotationRef,          data: annotationDict        }]);        var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);        var data = annotation.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);      }      checkJsAction({        jsEntry: 'function someFun() { return "qwerty"; } someFun();',        expectedUrl: undefined,        expectedUnsafeUrl: undefined,        expectedNewWindow: undefined      });      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: undefined      });      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      });    });    it('should correctly parse a Named action', function () {      var actionDict = new Dict();      actionDict.set('Type', Name.get('Action'));      actionDict.set('S', Name.get('Named'));      actionDict.set('N', Name.get('GoToPage'));      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('A', actionDict);      var annotationRef = new Ref(12, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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', function () {      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('Dest', Name.get('LI0'));      var annotationRef = new Ref(583, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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', function () {      var annotationDict = new Dict();      annotationDict.set('Type', Name.get('Annot'));      annotationDict.set('Subtype', Name.get('Link'));      annotationDict.set('Dest', [new Ref(17, 0), Name.get('XYZ'), 0, 841.89, null]);      var annotationRef = new Ref(10, 0);      var xref = new XRefMock([{        ref: annotationRef,        data: annotationDict      }]);      var annotation = annotationFactory.create(xref, annotationRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.LINK);      expect(data.url).toBeUndefined();      expect(data.unsafeUrl).toBeUndefined();      expect(data.dest).toEqual([{        num: 17,        gen: 0      }, { name: 'XYZ' }, 0, 841.89, null]);    });  });  describe('WidgetAnnotation', function () {    var widgetDict;    beforeEach(function (done) {      widgetDict = new Dict();      widgetDict.set('Type', Name.get('Annot'));      widgetDict.set('Subtype', Name.get('Widget'));      done();    });    afterEach(function () {      widgetDict = null;    });    it('should handle unknown field names', function () {      var widgetRef = new Ref(20, 0);      var xref = new XRefMock([{        ref: widgetRef,        data: widgetDict      }]);      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.fieldName).toEqual('');    });    it('should construct the field name when there are no ancestors', function () {      widgetDict.set('T', 'foo');      var widgetRef = new Ref(21, 0);      var xref = new XRefMock([{        ref: widgetRef,        data: widgetDict      }]);      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.fieldName).toEqual('foo');    });    it('should construct the field name when there are ancestors', function () {      var firstParent = new Dict();      firstParent.set('T', 'foo');      var secondParent = new Dict();      secondParent.set('Parent', firstParent);      secondParent.set('T', 'bar');      widgetDict.set('Parent', secondParent);      widgetDict.set('T', 'baz');      var widgetRef = new Ref(22, 0);      var xref = new XRefMock([{        ref: widgetRef,        data: widgetDict      }]);      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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)', function () {      var parentDict = new Dict();      parentDict.set('Parent', null);      parentDict.set('T', 'foo');      widgetDict.set('Parent', parentDict);      widgetDict.set('T', 'bar');      var widgetRef = new Ref(22, 0);      var xref = new XRefMock([{        ref: widgetRef,        data: widgetDict      }]);      var annotation = annotationFactory.create(xref, widgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.fieldName).toEqual('foo.bar');    });  });  describe('TextWidgetAnnotation', function () {    var textWidgetDict;    beforeEach(function (done) {      textWidgetDict = new Dict();      textWidgetDict.set('Type', Name.get('Annot'));      textWidgetDict.set('Subtype', Name.get('Widget'));      textWidgetDict.set('FT', Name.get('Tx'));      done();    });    afterEach(function () {      textWidgetDict = null;    });    it('should handle unknown text alignment, maximum length and flags', function () {      var textWidgetRef = new Ref(124, 0);      var xref = new XRefMock([{        ref: textWidgetRef,        data: textWidgetDict      }]);      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.textAlignment).toEqual(null);      expect(data.maxLen).toEqual(null);      expect(data.readOnly).toEqual(false);      expect(data.multiLine).toEqual(false);      expect(data.comb).toEqual(false);    });    it('should not set invalid text alignment, maximum length and flags', function () {      textWidgetDict.set('Q', 'center');      textWidgetDict.set('MaxLen', 'five');      textWidgetDict.set('Ff', 'readonly');      var textWidgetRef = new Ref(43, 0);      var xref = new XRefMock([{        ref: textWidgetRef,        data: textWidgetDict      }]);      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.textAlignment).toEqual(null);      expect(data.maxLen).toEqual(null);      expect(data.readOnly).toEqual(false);      expect(data.multiLine).toEqual(false);      expect(data.comb).toEqual(false);    });    it('should set valid text alignment, maximum length and flags', function () {      textWidgetDict.set('Q', 1);      textWidgetDict.set('MaxLen', 20);      textWidgetDict.set('Ff', AnnotationFieldFlag.READONLY + AnnotationFieldFlag.MULTILINE);      var textWidgetRef = new Ref(84, 0);      var xref = new XRefMock([{        ref: textWidgetRef,        data: textWidgetDict      }]);      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.textAlignment).toEqual(1);      expect(data.maxLen).toEqual(20);      expect(data.readOnly).toEqual(true);      expect(data.multiLine).toEqual(true);    });    it('should reject comb fields without a maximum length', function () {      textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);      var textWidgetRef = new Ref(46, 0);      var xref = new XRefMock([{        ref: textWidgetRef,        data: textWidgetDict      }]);      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.comb).toEqual(false);    });    it('should accept comb fields with a maximum length', function () {      textWidgetDict.set('MaxLen', 20);      textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);      var textWidgetRef = new Ref(46, 0);      var xref = new XRefMock([{        ref: textWidgetRef,        data: textWidgetDict      }]);      var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.comb).toEqual(true);    });    it('should only accept comb fields when the flags are valid', function () {      var invalidFieldFlags = [AnnotationFieldFlag.MULTILINE, AnnotationFieldFlag.PASSWORD, AnnotationFieldFlag.FILESELECT];      var flags = AnnotationFieldFlag.COMB + AnnotationFieldFlag.MULTILINE + AnnotationFieldFlag.PASSWORD + AnnotationFieldFlag.FILESELECT;      for (var i = 0, ii = invalidFieldFlags.length; i <= ii; i++) {        textWidgetDict.set('MaxLen', 20);        textWidgetDict.set('Ff', flags);        var textWidgetRef = new Ref(93, 0);        var xref = new XRefMock([{          ref: textWidgetRef,          data: textWidgetDict        }]);        var annotation = annotationFactory.create(xref, textWidgetRef, pdfManagerMock, idFactoryMock);        var data = annotation.data;        expect(data.annotationType).toEqual(AnnotationType.WIDGET);        var valid = invalidFieldFlags.length === 0;        expect(data.comb).toEqual(valid);        if (!valid) {          flags -= invalidFieldFlags.splice(-1, 1);        }      }    });  });  describe('ButtonWidgetAnnotation', function () {    var buttonWidgetDict;    beforeEach(function (done) {      buttonWidgetDict = new Dict();      buttonWidgetDict.set('Type', Name.get('Annot'));      buttonWidgetDict.set('Subtype', Name.get('Widget'));      buttonWidgetDict.set('FT', Name.get('Btn'));      done();    });    afterEach(function () {      buttonWidgetDict = null;    });    it('should handle checkboxes', function () {      buttonWidgetDict.set('V', Name.get('1'));      var buttonWidgetRef = new Ref(124, 0);      var xref = new XRefMock([{        ref: buttonWidgetRef,        data: buttonWidgetDict      }]);      var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.checkBox).toEqual(true);      expect(data.fieldValue).toEqual('1');      expect(data.radioButton).toEqual(false);    });    it('should handle radio buttons with a field value', function () {      var parentDict = new Dict();      parentDict.set('V', Name.get('1'));      var normalAppearanceStateDict = new Dict();      normalAppearanceStateDict.set('2', null);      var appearanceStatesDict = new Dict();      appearanceStatesDict.set('N', normalAppearanceStateDict);      buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);      buttonWidgetDict.set('Parent', parentDict);      buttonWidgetDict.set('AP', appearanceStatesDict);      var buttonWidgetRef = new Ref(124, 0);      var xref = new XRefMock([{        ref: buttonWidgetRef,        data: buttonWidgetDict      }]);      var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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 without a field value', function () {      var normalAppearanceStateDict = new Dict();      normalAppearanceStateDict.set('2', null);      var appearanceStatesDict = new Dict();      appearanceStatesDict.set('N', normalAppearanceStateDict);      buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);      buttonWidgetDict.set('AP', appearanceStatesDict);      var buttonWidgetRef = new Ref(124, 0);      var xref = new XRefMock([{        ref: buttonWidgetRef,        data: buttonWidgetDict      }]);      var annotation = annotationFactory.create(xref, buttonWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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');    });  });  describe('ChoiceWidgetAnnotation', function () {    var choiceWidgetDict;    beforeEach(function (done) {      choiceWidgetDict = new Dict();      choiceWidgetDict.set('Type', Name.get('Annot'));      choiceWidgetDict.set('Subtype', Name.get('Widget'));      choiceWidgetDict.set('FT', Name.get('Ch'));      done();    });    afterEach(function () {      choiceWidgetDict = null;    });    it('should handle missing option arrays', function () {      var choiceWidgetRef = new Ref(122, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.options).toEqual([]);    });    it('should handle option arrays with array elements', function () {      var optionBarRef = new Ref(20, 0);      var optionBarStr = 'Bar';      var optionOneRef = new Ref(10, 0);      var optionOneArr = ['bar_export', optionBarRef];      var options = [['foo_export', 'Foo'], optionOneRef];      var expected = [{        exportValue: 'foo_export',        displayValue: 'Foo'      }, {        exportValue: 'bar_export',        displayValue: 'Bar'      }];      choiceWidgetDict.set('Opt', options);      var choiceWidgetRef = new Ref(123, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }, {        ref: optionBarRef,        data: optionBarStr      }, {        ref: optionOneRef,        data: optionOneArr      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.options).toEqual(expected);    });    it('should handle option arrays with string elements', function () {      var optionBarRef = new Ref(10, 0);      var optionBarStr = 'Bar';      var options = ['Foo', optionBarRef];      var expected = [{        exportValue: 'Foo',        displayValue: 'Foo'      }, {        exportValue: 'Bar',        displayValue: 'Bar'      }];      choiceWidgetDict.set('Opt', options);      var choiceWidgetRef = new Ref(981, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }, {        ref: optionBarRef,        data: optionBarStr      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.options).toEqual(expected);    });    it('should handle inherited option arrays (issue 8094)', function () {      var options = [['Value1', 'Description1'], ['Value2', 'Description2']];      var expected = [{        exportValue: 'Value1',        displayValue: 'Description1'      }, {        exportValue: 'Value2',        displayValue: 'Description2'      }];      var parentDict = new Dict();      parentDict.set('Opt', options);      choiceWidgetDict.set('Parent', parentDict);      var choiceWidgetRef = new Ref(123, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.options).toEqual(expected);    });    it('should handle array field values', function () {      var fieldValue = ['Foo', 'Bar'];      choiceWidgetDict.set('V', fieldValue);      var choiceWidgetRef = new Ref(968, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.fieldValue).toEqual(fieldValue);    });    it('should handle string field values', function () {      var fieldValue = 'Foo';      choiceWidgetDict.set('V', fieldValue);      var choiceWidgetRef = new Ref(978, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.fieldValue).toEqual([fieldValue]);    });    it('should handle unknown flags', function () {      var choiceWidgetRef = new Ref(166, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.readOnly).toEqual(false);      expect(data.combo).toEqual(false);      expect(data.multiSelect).toEqual(false);    });    it('should not set invalid flags', function () {      choiceWidgetDict.set('Ff', 'readonly');      var choiceWidgetRef = new Ref(165, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.readOnly).toEqual(false);      expect(data.combo).toEqual(false);      expect(data.multiSelect).toEqual(false);    });    it('should set valid flags', function () {      choiceWidgetDict.set('Ff', AnnotationFieldFlag.READONLY + AnnotationFieldFlag.COMBO + AnnotationFieldFlag.MULTISELECT);      var choiceWidgetRef = new Ref(512, 0);      var xref = new XRefMock([{        ref: choiceWidgetRef,        data: choiceWidgetDict      }]);      var annotation = annotationFactory.create(xref, choiceWidgetRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.WIDGET);      expect(data.readOnly).toEqual(true);      expect(data.combo).toEqual(true);      expect(data.multiSelect).toEqual(true);    });  });  describe('LineAnnotation', function () {    it('should set the line coordinates', function () {      var lineDict = new Dict();      lineDict.set('Type', Name.get('Annot'));      lineDict.set('Subtype', Name.get('Line'));      lineDict.set('L', [1, 2, 3, 4]);      var lineRef = new Ref(122, 0);      var xref = new XRefMock([{        ref: lineRef,        data: lineDict      }]);      var annotation = annotationFactory.create(xref, lineRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.LINE);      expect(data.lineCoordinates).toEqual([1, 2, 3, 4]);    });  });  describe('FileAttachmentAnnotation', function () {    it('should correctly parse a file attachment', function () {      var fileStream = new StringStream('<<\n' + '/Type /EmbeddedFile\n' + '/Subtype /text#2Fplain\n' + '>>\n' + 'stream\n' + 'Test attachment' + 'endstream\n');      var lexer = new Lexer(fileStream);      var parser = new Parser(lexer, true);      var fileStreamRef = new Ref(18, 0);      var fileStreamDict = parser.getObj();      var embeddedFileDict = new Dict();      embeddedFileDict.set('F', fileStreamRef);      var fileSpecRef = new Ref(19, 0);      var fileSpecDict = new Dict();      fileSpecDict.set('Type', Name.get('Filespec'));      fileSpecDict.set('Desc', '');      fileSpecDict.set('EF', embeddedFileDict);      fileSpecDict.set('UF', 'Test.txt');      var fileAttachmentRef = new Ref(20, 0);      var 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');      var xref = new XRefMock([{        ref: fileStreamRef,        data: fileStreamDict      }, {        ref: fileSpecRef,        data: fileSpecDict      }, {        ref: fileAttachmentRef,        data: fileAttachmentDict      }]);      embeddedFileDict.assignXref(xref);      fileSpecDict.assignXref(xref);      fileAttachmentDict.assignXref(xref);      var annotation = annotationFactory.create(xref, fileAttachmentRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      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 the parent flags when the Popup is not viewable, ' + 'but the parent is (PR 7352)', function () {      var parentDict = new Dict();      parentDict.set('Type', Name.get('Annot'));      parentDict.set('Subtype', Name.get('Text'));      parentDict.set('F', 28);      var popupDict = new Dict();      popupDict.set('Type', Name.get('Annot'));      popupDict.set('Subtype', Name.get('Popup'));      popupDict.set('F', 25);      popupDict.set('Parent', parentDict);      var popupRef = new Ref(13, 0);      var xref = new XRefMock([{        ref: popupRef,        data: popupDict      }]);      var annotation = annotationFactory.create(xref, popupRef, pdfManagerMock, idFactoryMock);      var data = annotation.data;      expect(data.annotationType).toEqual(AnnotationType.POPUP);      expect(data.annotationFlags).toEqual(25);      expect(annotation.viewable).toEqual(true);    });  });});
 |