2017-01-10 01:40:57 +09:00
|
|
|
/* 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.
|
|
|
|
*/
|
2017-04-17 05:30:27 +09:00
|
|
|
|
|
|
|
import {
|
|
|
|
Annotation, AnnotationBorderStyle, AnnotationFactory
|
|
|
|
} from '../../src/core/annotation';
|
|
|
|
import {
|
|
|
|
AnnotationBorderStyleType, AnnotationFieldFlag, AnnotationFlag,
|
|
|
|
AnnotationType, stringToBytes, stringToUTF8String
|
|
|
|
} from '../../src/shared/util';
|
2017-07-29 07:35:10 +09:00
|
|
|
import { Dict, Name, Ref } from '../../src/core/primitives';
|
2017-04-17 05:30:27 +09:00
|
|
|
import { Lexer, Parser } from '../../src/core/parser';
|
|
|
|
import { StringStream } from '../../src/core/stream';
|
2017-07-29 07:35:10 +09:00
|
|
|
import { XRefMock } from './test_utils';
|
2017-01-10 01:40:57 +09:00
|
|
|
|
2017-01-13 03:00:53 +09:00
|
|
|
describe('annotation', function() {
|
2018-03-21 09:43:40 +09:00
|
|
|
class PDFManagerMock {
|
|
|
|
constructor(params) {
|
|
|
|
this.docBaseUrl = params.docBaseUrl || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-10-01 19:05:07 +09:00
|
|
|
}
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
class IdFactoryMock {
|
|
|
|
constructor(params) {
|
|
|
|
this.uniquePrefix = params.prefix || 'p0_';
|
|
|
|
this.idCounters = {
|
|
|
|
obj: params.startObjId || 0,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
createObjId() {
|
|
|
|
return this.uniquePrefix + (++this.idCounters.obj);
|
|
|
|
}
|
2017-01-09 00:51:30 +09:00
|
|
|
}
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
let pdfManagerMock, idFactoryMock;
|
2016-05-25 00:35:45 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
beforeAll(function(done) {
|
2016-10-01 19:05:07 +09:00
|
|
|
pdfManagerMock = new PDFManagerMock({
|
|
|
|
docBaseUrl: null,
|
|
|
|
});
|
2017-01-09 00:51:30 +09:00
|
|
|
idFactoryMock = new IdFactoryMock({ });
|
2016-05-25 00:35:45 +09:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
afterAll(function() {
|
2016-10-01 19:05:07 +09:00
|
|
|
pdfManagerMock = null;
|
2017-01-09 00:51:30 +09:00
|
|
|
idFactoryMock = null;
|
2016-05-25 00:35:45 +09:00
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
describe('AnnotationFactory', function() {
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should get id for annotation', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-08-26 23:01:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(10, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.id).toEqual('10R');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-08-26 23:01:25 +09:00
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
it('should handle, and get fallback IDs for, annotations that are not ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'indirect objects (issue 7569)', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-08-26 23:01:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const xref = new XRefMock();
|
|
|
|
const idFactory = new IdFactoryMock({
|
2017-01-09 00:51:30 +09:00
|
|
|
prefix: 'p0_',
|
|
|
|
startObjId: 0,
|
|
|
|
});
|
2016-08-26 23:01:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
const annotation1 = AnnotationFactory.create(xref, annotationDict,
|
|
|
|
pdfManagerMock, idFactory).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.id).toEqual('annot_p0_1');
|
|
|
|
});
|
2016-08-26 23:01:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
const annotation2 = AnnotationFactory.create(xref, annotationDict,
|
|
|
|
pdfManagerMock, idFactory).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.id).toEqual('annot_p0_2');
|
|
|
|
});
|
|
|
|
|
|
|
|
Promise.all([annotation1, annotation2]).then(done, done.fail);
|
2016-08-26 23:01:25 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle missing /Subtype', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-07-24 21:32:48 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(1, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2016-07-24 21:32:48 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toBeUndefined();
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-07-24 21:32:48 +09:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-07-15 23:59:25 +09:00
|
|
|
describe('Annotation', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
let dict, ref;
|
2016-07-24 21:32:48 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
beforeAll(function(done) {
|
2016-07-24 21:32:48 +09:00
|
|
|
dict = new Dict();
|
|
|
|
ref = new Ref(1, 0);
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
afterAll(function() {
|
2016-07-24 21:32:48 +09:00
|
|
|
dict = ref = null;
|
|
|
|
});
|
|
|
|
|
2015-11-22 07:25:17 +09:00
|
|
|
it('should set and get flags', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-11-22 07:25:17 +09:00
|
|
|
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() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-11-22 07:25:17 +09:00
|
|
|
|
|
|
|
expect(annotation.viewable).toEqual(true);
|
|
|
|
expect(annotation.printable).toEqual(false);
|
|
|
|
});
|
|
|
|
|
2015-07-21 05:01:47 +09:00
|
|
|
it('should set and get a valid rectangle', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-21 05:01:47 +09:00
|
|
|
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() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-21 05:01:47 +09:00
|
|
|
annotation.setRectangle([117, 694, 164.298]);
|
|
|
|
|
|
|
|
expect(annotation.rectangle).toEqual([0, 0, 0, 0]);
|
|
|
|
});
|
|
|
|
|
2015-07-15 23:59:25 +09:00
|
|
|
it('should reject a color if it is not an array', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-15 23:59:25 +09:00
|
|
|
annotation.setColor('red');
|
|
|
|
|
2018-06-12 00:25:40 +09:00
|
|
|
expect(annotation.color).toEqual(new Uint8ClampedArray([0, 0, 0]));
|
2015-07-15 23:59:25 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should set and get a transparent color', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-15 23:59:25 +09:00
|
|
|
annotation.setColor([]);
|
|
|
|
|
|
|
|
expect(annotation.color).toEqual(null);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set and get a grayscale color', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-15 23:59:25 +09:00
|
|
|
annotation.setColor([0.4]);
|
|
|
|
|
2018-06-12 00:25:40 +09:00
|
|
|
expect(annotation.color).toEqual(new Uint8ClampedArray([102, 102, 102]));
|
2015-07-15 23:59:25 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should set and get an RGB color', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-15 23:59:25 +09:00
|
|
|
annotation.setColor([0, 0, 1]);
|
|
|
|
|
2018-06-12 00:25:40 +09:00
|
|
|
expect(annotation.color).toEqual(new Uint8ClampedArray([0, 0, 255]));
|
2015-07-15 23:59:25 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should set and get a CMYK color', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-15 23:59:25 +09:00
|
|
|
annotation.setColor([0.1, 0.92, 0.84, 0.02]);
|
|
|
|
|
2018-06-12 00:25:40 +09:00
|
|
|
expect(annotation.color).toEqual(new Uint8ClampedArray([234, 59, 48]));
|
2015-07-15 23:59:25 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should not set and get an invalid color', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotation = new Annotation({ dict, ref, });
|
2015-07-15 23:59:25 +09:00
|
|
|
annotation.setColor([0.4, 0.6]);
|
|
|
|
|
2018-06-12 00:25:40 +09:00
|
|
|
expect(annotation.color).toEqual(new Uint8ClampedArray([0, 0, 0]));
|
2015-07-15 23:59:25 +09:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2014-12-26 05:04:01 +09:00
|
|
|
describe('AnnotationBorderStyle', function() {
|
|
|
|
it('should set and get a valid width', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setWidth(3);
|
|
|
|
|
|
|
|
expect(borderStyle.width).toEqual(3);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not set and get an invalid width', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setWidth('three');
|
|
|
|
|
|
|
|
expect(borderStyle.width).toEqual(1);
|
|
|
|
});
|
|
|
|
|
2019-01-04 18:33:15 +09:00
|
|
|
it('should set the width to zero, when the input is a `Name` (issue 10385)',
|
2018-12-31 20:21:28 +09:00
|
|
|
function() {
|
2019-01-04 18:33:15 +09:00
|
|
|
const borderStyleZero = new AnnotationBorderStyle();
|
|
|
|
borderStyleZero.setWidth(Name.get('0'));
|
|
|
|
const borderStyleFive = new AnnotationBorderStyle();
|
|
|
|
borderStyleFive.setWidth(Name.get('5'));
|
2018-12-31 20:21:28 +09:00
|
|
|
|
2019-01-04 18:33:15 +09:00
|
|
|
expect(borderStyleZero.width).toEqual(0);
|
|
|
|
expect(borderStyleFive.width).toEqual(0);
|
2018-12-31 20:21:28 +09:00
|
|
|
});
|
|
|
|
|
2014-12-26 05:04:01 +09:00
|
|
|
it('should set and get a valid style', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2015-07-18 04:51:24 +09:00
|
|
|
borderStyle.setStyle(Name.get('D'));
|
2014-12-26 05:04:01 +09:00
|
|
|
|
|
|
|
expect(borderStyle.style).toEqual(AnnotationBorderStyleType.DASHED);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not set and get an invalid style', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setStyle('Dashed');
|
|
|
|
|
|
|
|
expect(borderStyle.style).toEqual(AnnotationBorderStyleType.SOLID);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set and get a valid dash array', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setDashArray([1, 2, 3]);
|
|
|
|
|
|
|
|
expect(borderStyle.dashArray).toEqual([1, 2, 3]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not set and get an invalid dash array', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setDashArray([0, 0]);
|
|
|
|
|
|
|
|
expect(borderStyle.dashArray).toEqual([3]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set and get a valid horizontal corner radius', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setHorizontalCornerRadius(3);
|
|
|
|
|
|
|
|
expect(borderStyle.horizontalCornerRadius).toEqual(3);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not set and get an invalid horizontal corner radius',
|
|
|
|
function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setHorizontalCornerRadius('three');
|
|
|
|
|
|
|
|
expect(borderStyle.horizontalCornerRadius).toEqual(0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set and get a valid vertical corner radius', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setVerticalCornerRadius(3);
|
|
|
|
|
|
|
|
expect(borderStyle.verticalCornerRadius).toEqual(3);
|
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
it('should not set and get an invalid vertical corner radius', function() {
|
|
|
|
const borderStyle = new AnnotationBorderStyle();
|
2014-12-26 05:04:01 +09:00
|
|
|
borderStyle.setVerticalCornerRadius('three');
|
|
|
|
|
|
|
|
expect(borderStyle.verticalCornerRadius).toEqual(0);
|
|
|
|
});
|
|
|
|
});
|
2016-02-15 05:27:53 +09:00
|
|
|
|
2016-03-26 18:00:25 +09:00
|
|
|
describe('LinkAnnotation', function() {
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should correctly parse a URI action', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('URI'));
|
|
|
|
actionDict.set('URI', 'http://www.ctan.org/tex-archive/info/lshort');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(820, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2016-03-26 18:00:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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();
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-03-26 18:00:25 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should correctly parse a URI action, where the URI entry ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'is missing a protocol', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('URI'));
|
|
|
|
actionDict.set('URI', 'www.hmrc.gov.uk');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(353, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2016-03-26 18:00:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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();
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-03-26 18:00:25 +09:00
|
|
|
});
|
|
|
|
|
2016-09-30 18:44:24 +09:00
|
|
|
it('should correctly parse a URI action, where the URI entry ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'has an incorrect encoding (bug 1122280)', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionStream = new StringStream(
|
2016-09-30 18:44:24 +09:00
|
|
|
'<<\n' +
|
|
|
|
'/Type /Action\n' +
|
|
|
|
'/S /URI\n' +
|
|
|
|
'/URI (http://www.example.com/\\303\\274\\303\\266\\303\\244)\n' +
|
|
|
|
'>>\n'
|
|
|
|
);
|
2018-08-12 01:50:31 +09:00
|
|
|
const lexer = new Lexer(actionStream);
|
|
|
|
const parser = new Parser(lexer);
|
|
|
|
const actionDict = parser.getObj();
|
2016-09-30 18:44:24 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-09-30 18:44:24 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(8, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-30 18:44:24 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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();
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-30 18:44:24 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should correctly parse a GoTo action', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('GoTo'));
|
|
|
|
actionDict.set('D', 'page.157');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(798, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2016-03-26 18:00:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.url).toBeUndefined();
|
|
|
|
expect(data.unsafeUrl).toBeUndefined();
|
|
|
|
expect(data.dest).toEqual('page.157');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-03-26 18:00:25 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should correctly parse a GoToR action, where the FileSpec entry ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'is a string containing a relative URL', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('GoToR'));
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
actionDict.set('F', '../../0013/001346/134685E.pdf');
|
|
|
|
actionDict.set('D', '4.3');
|
|
|
|
actionDict.set('NewWindow', true);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(489, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
});
|
|
|
|
|
2016-10-01 19:05:07 +09:00
|
|
|
it('should correctly parse a GoToR action, containing a relative URL, ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'with the "docBaseUrl" parameter specified', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-10-01 19:05:07 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('GoToR'));
|
|
|
|
actionDict.set('F', '../../0013/001346/134685E.pdf');
|
|
|
|
actionDict.set('D', '4.3');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-10-01 19:05:07 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(489, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-10-01 19:05:07 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2018-08-12 01:50:31 +09:00
|
|
|
const pdfManager = new PDFManagerMock({
|
2016-10-01 19:05:07 +09:00
|
|
|
docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf',
|
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManager,
|
|
|
|
idFactoryMock).then(({ 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();
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-10-01 19:05:07 +09:00
|
|
|
});
|
|
|
|
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
it('should correctly parse a GoToR action, with named destination',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('GoToR'));
|
|
|
|
actionDict.set('F', 'http://www.example.com/test.pdf');
|
2016-03-26 18:00:25 +09:00
|
|
|
actionDict.set('D', '15');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(495, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
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();
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should correctly parse a GoToR action, with explicit destination array',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
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]);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
[api-minor] Let `LinkAnnotation`/`PDFLinkService_getDestinationHash` return a stringified version of the destination array for explicit destinations
Currently for explicit destinations, compared to named destinations, we manually try to build a hash that often times is a quite poor representation of the *actual* destination. (Currently this only, kind of, works for `\XYZ` destinations.)
For PDF files using explicit destinations, this can make it difficult/impossible to obtain a link to a specific section of the document through the URL.
Note that in practice most PDF files, especially newer ones, use named destinations and these are thus unnaffected by this patch.
This patch also fixes an existing issue in `PDFLinkService_getDestinationHash`, where a named destination consisting of only a number would not be handled correctly.
With the added, and already existing, type checks in place for destinations, I really don't think that this patch exposes any "sensitive" internal destination code not already accessible through normal hash parameters.
*Please note:* Just trying to improve the algorithm that generates the hash is unfortunately not possible in general, since there are a number of cases where it will simply never work well.
- First of all, note that `getDestinationHash` currently relies on the `_pagesRefCache`, hence it's possible that the hash returned is empty during e.g. ranged/streamed loading of a PDF file.
- Second of all, the currently computed hash is actually dependent on the document rotation. With named destinations, the fetched internal destination array is rotational invariant (as it should be), but this will not hold in general for the hash. We can easily avoid this issue by using a stringified destination array.
- Third of all, note that according to the PDF specification[1], `GoToR` destinations may actually contain explicit destination arrays. Since we cannot really construct a hash in `annotation.js`, we currently have no good way to support those. Even though this case seems *very* rare in practice (I've not actually seen such a PDF file), it's in the specification, and this patch allows us to support that for "free".
---
[1] http://www.adobe.com/content/dam/Adobe/en/devnet/acrobat/pdfs/PDF32000_2008.pdf#G11.1951685
2016-05-15 19:12:47 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(489, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2016-03-26 18:00:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.url).toEqual(new URL('http://www.example.com/test.pdf#' +
|
2016-09-30 23:32:22 +09:00
|
|
|
'[14,{"name":"XYZ"},null,298.043,null]').href);
|
2018-03-21 09:43:40 +09:00
|
|
|
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();
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-03-26 18:00:25 +09:00
|
|
|
});
|
2016-10-21 20:29:15 +09:00
|
|
|
|
|
|
|
it('should correctly parse a Launch action, where the FileSpec dict ' +
|
|
|
|
'contains a relative URL, with the "docBaseUrl" parameter specified',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const fileSpecDict = new Dict();
|
2016-10-21 20:29:15 +09:00
|
|
|
fileSpecDict.set('Type', Name.get('FileSpec'));
|
|
|
|
fileSpecDict.set('F', 'Part II/Part II.pdf');
|
|
|
|
fileSpecDict.set('UF', 'Part II/Part II.pdf');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-10-21 20:29:15 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('Launch'));
|
|
|
|
actionDict.set('F', fileSpecDict);
|
|
|
|
actionDict.set('NewWindow', true);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-10-21 20:29:15 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(88, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-10-21 20:29:15 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2018-08-12 01:50:31 +09:00
|
|
|
const pdfManager = new PDFManagerMock({
|
2016-10-21 20:29:15 +09:00
|
|
|
docBaseUrl: 'http://www.example.com/test/pdfs/qwerty.pdf',
|
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManager,
|
|
|
|
idFactoryMock).then(({ 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);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-10-21 20:29:15 +09:00
|
|
|
});
|
2016-03-26 18:00:25 +09:00
|
|
|
|
2016-11-09 00:38:22 +09:00
|
|
|
it('should recover valid URLs from JavaScript actions having certain ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'white-listed formats', function(done) {
|
2016-11-09 00:38:22 +09:00
|
|
|
function checkJsAction(params) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const jsEntry = params.jsEntry;
|
|
|
|
const expectedUrl = params.expectedUrl;
|
|
|
|
const expectedUnsafeUrl = params.expectedUnsafeUrl;
|
|
|
|
const expectedNewWindow = params.expectedNewWindow;
|
2016-11-09 00:38:22 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-11-09 00:38:22 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('JavaScript'));
|
|
|
|
actionDict.set('JS', jsEntry);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-11-09 00:38:22 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(46, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-11-09 00:38:22 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
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);
|
|
|
|
});
|
2016-11-09 00:38:22 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check that we reject a 'JS' entry containing arbitrary JavaScript.
|
2018-03-21 09:43:40 +09:00
|
|
|
const annotation1 = checkJsAction({
|
2016-11-09 00:38:22 +09:00
|
|
|
jsEntry: 'function someFun() { return "qwerty"; } someFun();',
|
|
|
|
expectedUrl: undefined,
|
|
|
|
expectedUnsafeUrl: undefined,
|
2017-02-27 20:08:41 +09:00
|
|
|
expectedNewWindow: undefined,
|
2016-11-09 00:38:22 +09:00
|
|
|
});
|
2018-03-21 09:43:40 +09:00
|
|
|
|
2016-11-09 00:38:22 +09:00
|
|
|
// Check that we accept a white-listed {string} 'JS' entry.
|
2018-03-21 09:43:40 +09:00
|
|
|
const annotation2 = checkJsAction({
|
2016-11-09 00:38:22 +09:00
|
|
|
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',
|
2017-02-27 20:08:41 +09:00
|
|
|
expectedNewWindow: undefined,
|
2016-11-09 00:38:22 +09:00
|
|
|
});
|
2018-03-21 09:43:40 +09:00
|
|
|
|
2016-11-09 00:38:22 +09:00
|
|
|
// Check that we accept a white-listed {Stream} 'JS' entry.
|
2018-03-21 09:43:40 +09:00
|
|
|
const annotation3 = checkJsAction({
|
2016-11-09 00:38:22 +09:00
|
|
|
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',
|
2017-02-27 20:08:41 +09:00
|
|
|
expectedNewWindow: true,
|
2016-11-09 00:38:22 +09:00
|
|
|
});
|
2018-03-21 09:43:40 +09:00
|
|
|
|
|
|
|
Promise.all([annotation1, annotation2, annotation3]).then(done,
|
|
|
|
done.fail);
|
2016-11-09 00:38:22 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should correctly parse a Named action', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const actionDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
actionDict.set('Type', Name.get('Action'));
|
|
|
|
actionDict.set('S', Name.get('Named'));
|
|
|
|
actionDict.set('N', Name.get('GoToPage'));
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('A', actionDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(12, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2016-03-26 18:00:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.url).toBeUndefined();
|
|
|
|
expect(data.unsafeUrl).toBeUndefined();
|
|
|
|
expect(data.action).toEqual('GoToPage');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-03-26 18:00:25 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should correctly parse a simple Dest', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-03-26 18:00:25 +09:00
|
|
|
annotationDict.set('Type', Name.get('Annot'));
|
|
|
|
annotationDict.set('Subtype', Name.get('Link'));
|
|
|
|
annotationDict.set('Dest', Name.get('LI0'));
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(583, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
2016-03-26 18:00:25 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.url).toBeUndefined();
|
|
|
|
expect(data.unsafeUrl).toBeUndefined();
|
|
|
|
expect(data.dest).toEqual('LI0');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-03-26 18:00:25 +09:00
|
|
|
});
|
2016-09-30 23:32:22 +09:00
|
|
|
|
|
|
|
it('should correctly parse a simple Dest, with explicit destination array',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationDict = new Dict();
|
2016-09-30 23:32:22 +09:00
|
|
|
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]);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const annotationRef = new Ref(10, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-30 23:32:22 +09:00
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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]);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-30 23:32:22 +09:00
|
|
|
});
|
2017-08-25 20:40:50 +09:00
|
|
|
|
|
|
|
it('should correctly parse a Dest, which violates the specification ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'by containing a dictionary', function(done) {
|
2017-08-25 20:40:50 +09:00
|
|
|
let destDict = new Dict();
|
|
|
|
destDict.set('Type', Name.get('Action'));
|
|
|
|
destDict.set('S', Name.get('GoTo'));
|
|
|
|
destDict.set('D', 'page.157');
|
|
|
|
|
|
|
|
let 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);
|
|
|
|
|
|
|
|
let annotationRef = new Ref(798, 0);
|
|
|
|
let xref = new XRefMock([
|
|
|
|
{ ref: annotationRef, data: annotationDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, annotationRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINK);
|
|
|
|
expect(data.url).toBeUndefined();
|
|
|
|
expect(data.unsafeUrl).toBeUndefined();
|
|
|
|
expect(data.dest).toEqual('page.157');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2017-08-25 20:40:50 +09:00
|
|
|
});
|
2016-03-26 18:00:25 +09:00
|
|
|
});
|
|
|
|
|
2016-11-02 08:07:31 +09:00
|
|
|
describe('WidgetAnnotation', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
let widgetDict;
|
2016-11-02 08:07:31 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
beforeEach(function(done) {
|
2016-11-02 08:07:31 +09:00
|
|
|
widgetDict = new Dict();
|
|
|
|
widgetDict.set('Type', Name.get('Annot'));
|
|
|
|
widgetDict.set('Subtype', Name.get('Widget'));
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
afterEach(function() {
|
2016-11-02 08:07:31 +09:00
|
|
|
widgetDict = null;
|
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle unknown field names', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const widgetRef = new Ref(20, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-11-02 08:07:31 +09:00
|
|
|
{ ref: widgetRef, data: widgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, widgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.fieldName).toEqual('');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-11-02 08:07:31 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should construct the field name when there are no ancestors',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2016-11-02 08:07:31 +09:00
|
|
|
widgetDict.set('T', 'foo');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const widgetRef = new Ref(21, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-11-02 08:07:31 +09:00
|
|
|
{ ref: widgetRef, data: widgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, widgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.fieldName).toEqual('foo');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-11-02 08:07:31 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should construct the field name when there are ancestors',
|
|
|
|
function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const firstParent = new Dict();
|
2016-11-02 08:07:31 +09:00
|
|
|
firstParent.set('T', 'foo');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const secondParent = new Dict();
|
2016-11-02 08:07:31 +09:00
|
|
|
secondParent.set('Parent', firstParent);
|
|
|
|
secondParent.set('T', 'bar');
|
|
|
|
|
|
|
|
widgetDict.set('Parent', secondParent);
|
|
|
|
widgetDict.set('T', 'baz');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const widgetRef = new Ref(22, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-11-02 08:07:31 +09:00
|
|
|
{ ref: widgetRef, data: widgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, widgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.fieldName).toEqual('foo.bar.baz');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-11-02 08:07:31 +09:00
|
|
|
});
|
2017-03-10 07:51:52 +09:00
|
|
|
|
|
|
|
it('should construct the field name if a parent is not a dictionary ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'(issue 8143)', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const parentDict = new Dict();
|
2017-03-10 07:51:52 +09:00
|
|
|
parentDict.set('Parent', null);
|
|
|
|
parentDict.set('T', 'foo');
|
|
|
|
|
|
|
|
widgetDict.set('Parent', parentDict);
|
|
|
|
widgetDict.set('T', 'bar');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const widgetRef = new Ref(22, 0);
|
|
|
|
const xref = new XRefMock([
|
2017-03-10 07:51:52 +09:00
|
|
|
{ ref: widgetRef, data: widgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, widgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.fieldName).toEqual('foo.bar');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2017-03-10 07:51:52 +09:00
|
|
|
});
|
2016-11-02 08:07:31 +09:00
|
|
|
});
|
|
|
|
|
2016-09-13 21:57:11 +09:00
|
|
|
describe('TextWidgetAnnotation', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
let textWidgetDict;
|
2016-09-13 21:57:11 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
beforeEach(function(done) {
|
2016-09-13 21:57:11 +09:00
|
|
|
textWidgetDict = new Dict();
|
|
|
|
textWidgetDict.set('Type', Name.get('Annot'));
|
|
|
|
textWidgetDict.set('Subtype', Name.get('Widget'));
|
|
|
|
textWidgetDict.set('FT', Name.get('Tx'));
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
afterEach(function() {
|
2016-09-13 21:57:11 +09:00
|
|
|
textWidgetDict = null;
|
|
|
|
});
|
|
|
|
|
2016-09-15 04:51:21 +09:00
|
|
|
it('should handle unknown text alignment, maximum length and flags',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const textWidgetRef = new Ref(124, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-13 21:57:11 +09:00
|
|
|
{ ref: textWidgetRef, data: textWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, textWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-13 21:57:11 +09:00
|
|
|
});
|
|
|
|
|
2016-09-15 04:51:21 +09:00
|
|
|
it('should not set invalid text alignment, maximum length and flags',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2016-09-13 21:57:11 +09:00
|
|
|
textWidgetDict.set('Q', 'center');
|
|
|
|
textWidgetDict.set('MaxLen', 'five');
|
2016-09-15 04:51:21 +09:00
|
|
|
textWidgetDict.set('Ff', 'readonly');
|
2016-09-13 21:57:11 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const textWidgetRef = new Ref(43, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-13 21:57:11 +09:00
|
|
|
{ ref: textWidgetRef, data: textWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, textWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-13 21:57:11 +09:00
|
|
|
});
|
|
|
|
|
2016-09-15 04:51:21 +09:00
|
|
|
it('should set valid text alignment, maximum length and flags',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2016-09-13 21:57:11 +09:00
|
|
|
textWidgetDict.set('Q', 1);
|
|
|
|
textWidgetDict.set('MaxLen', 20);
|
2016-09-22 04:06:44 +09:00
|
|
|
textWidgetDict.set('Ff', AnnotationFieldFlag.READONLY +
|
|
|
|
AnnotationFieldFlag.MULTILINE);
|
2016-09-13 21:57:11 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const textWidgetRef = new Ref(84, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-13 21:57:11 +09:00
|
|
|
{ ref: textWidgetRef, data: textWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, textWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-13 21:57:11 +09:00
|
|
|
});
|
2016-09-20 07:04:11 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should reject comb fields without a maximum length', function(done) {
|
2016-09-22 04:06:44 +09:00
|
|
|
textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);
|
2016-09-20 07:04:11 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const textWidgetRef = new Ref(46, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-20 07:04:11 +09:00
|
|
|
{ ref: textWidgetRef, data: textWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, textWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.comb).toEqual(false);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-20 07:04:11 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should accept comb fields with a maximum length', function(done) {
|
2016-09-20 07:04:11 +09:00
|
|
|
textWidgetDict.set('MaxLen', 20);
|
2016-09-22 04:06:44 +09:00
|
|
|
textWidgetDict.set('Ff', AnnotationFieldFlag.COMB);
|
2016-09-20 07:04:11 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const textWidgetRef = new Ref(46, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-20 07:04:11 +09:00
|
|
|
{ ref: textWidgetRef, data: textWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, textWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.comb).toEqual(true);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-20 07:04:11 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should only accept comb fields when the flags are valid',
|
|
|
|
function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
let invalidFieldFlags = [
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFieldFlag.MULTILINE,
|
|
|
|
AnnotationFieldFlag.PASSWORD,
|
|
|
|
AnnotationFieldFlag.FILESELECT,
|
2016-09-20 07:04:11 +09:00
|
|
|
];
|
|
|
|
|
2016-09-22 04:06:44 +09:00
|
|
|
// Start with all invalid flags set and remove them one by one.
|
|
|
|
// The field may only use combs when all invalid flags are unset.
|
2018-08-12 01:50:31 +09:00
|
|
|
let flags = AnnotationFieldFlag.COMB + AnnotationFieldFlag.MULTILINE +
|
2016-09-22 04:06:44 +09:00
|
|
|
AnnotationFieldFlag.PASSWORD + AnnotationFieldFlag.FILESELECT;
|
2016-09-20 07:04:11 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
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 = new Ref(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) {
|
2018-10-24 20:08:08 +09:00
|
|
|
flags -= invalidFieldFlags.pop();
|
2018-03-21 09:43:40 +09:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2016-09-20 07:04:11 +09:00
|
|
|
}
|
2018-03-21 09:43:40 +09:00
|
|
|
promise.then(done, done.fail);
|
2016-09-20 07:04:11 +09:00
|
|
|
});
|
2016-09-13 21:57:11 +09:00
|
|
|
});
|
|
|
|
|
2016-12-16 07:49:46 +09:00
|
|
|
describe('ButtonWidgetAnnotation', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
let buttonWidgetDict;
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
beforeEach(function(done) {
|
2016-12-16 07:49:46 +09:00
|
|
|
buttonWidgetDict = new Dict();
|
|
|
|
buttonWidgetDict.set('Type', Name.get('Annot'));
|
|
|
|
buttonWidgetDict.set('Subtype', Name.get('Widget'));
|
|
|
|
buttonWidgetDict.set('FT', Name.get('Btn'));
|
2016-11-04 21:01:42 +09:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
afterEach(function() {
|
2016-12-16 07:49:46 +09:00
|
|
|
buttonWidgetDict = null;
|
2016-11-04 21:01:42 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle checkboxes with export value', function(done) {
|
2018-07-07 09:51:10 +09:00
|
|
|
buttonWidgetDict.set('V', Name.get('1'));
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const appearanceStatesDict = new Dict();
|
|
|
|
const exportValueOptionsDict = new Dict();
|
2018-07-07 09:51:10 +09:00
|
|
|
|
|
|
|
exportValueOptionsDict.set('Off', 0);
|
|
|
|
exportValueOptionsDict.set('Checked', 1);
|
|
|
|
appearanceStatesDict.set('D', exportValueOptionsDict);
|
|
|
|
buttonWidgetDict.set('AP', appearanceStatesDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const buttonWidgetRef = new Ref(124, 0);
|
|
|
|
const xref = new XRefMock([
|
2018-07-07 09:51:10 +09:00
|
|
|
{ ref: buttonWidgetRef, data: buttonWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, buttonWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.checkBox).toEqual(true);
|
|
|
|
expect(data.fieldValue).toEqual('1');
|
|
|
|
expect(data.radioButton).toEqual(false);
|
|
|
|
expect(data.exportValue).toEqual('Checked');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2018-07-07 09:51:10 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle checkboxes without export value', function(done) {
|
2016-12-16 07:49:46 +09:00
|
|
|
buttonWidgetDict.set('V', Name.get('1'));
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const buttonWidgetRef = new Ref(124, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-12-16 07:49:46 +09:00
|
|
|
{ ref: buttonWidgetRef, data: buttonWidgetDict, }
|
|
|
|
]);
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, buttonWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.checkBox).toEqual(true);
|
|
|
|
expect(data.fieldValue).toEqual('1');
|
|
|
|
expect(data.radioButton).toEqual(false);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-11-04 21:01:42 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle radio buttons with a field value', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const parentDict = new Dict();
|
2016-12-16 07:49:46 +09:00
|
|
|
parentDict.set('V', Name.get('1'));
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const normalAppearanceStateDict = new Dict();
|
2016-12-16 07:49:46 +09:00
|
|
|
normalAppearanceStateDict.set('2', null);
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const appearanceStatesDict = new Dict();
|
2016-12-16 07:49:46 +09:00
|
|
|
appearanceStatesDict.set('N', normalAppearanceStateDict);
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2016-12-16 07:49:46 +09:00
|
|
|
buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);
|
|
|
|
buttonWidgetDict.set('Parent', parentDict);
|
|
|
|
buttonWidgetDict.set('AP', appearanceStatesDict);
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const buttonWidgetRef = new Ref(124, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-12-16 07:49:46 +09:00
|
|
|
{ ref: buttonWidgetRef, data: buttonWidgetDict, }
|
|
|
|
]);
|
2016-11-04 21:01:42 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, buttonWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-12-16 07:49:46 +09:00
|
|
|
});
|
2017-02-08 07:44:31 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle radio buttons without a field value', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const normalAppearanceStateDict = new Dict();
|
2017-02-08 07:44:31 +09:00
|
|
|
normalAppearanceStateDict.set('2', null);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const appearanceStatesDict = new Dict();
|
2017-02-08 07:44:31 +09:00
|
|
|
appearanceStatesDict.set('N', normalAppearanceStateDict);
|
|
|
|
|
|
|
|
buttonWidgetDict.set('Ff', AnnotationFieldFlag.RADIO);
|
|
|
|
buttonWidgetDict.set('AP', appearanceStatesDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const buttonWidgetRef = new Ref(124, 0);
|
|
|
|
const xref = new XRefMock([
|
2017-02-08 07:44:31 +09:00
|
|
|
{ ref: buttonWidgetRef, data: buttonWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, buttonWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ 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');
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2017-02-08 07:44:31 +09:00
|
|
|
});
|
2016-11-04 21:01:42 +09:00
|
|
|
});
|
|
|
|
|
2016-09-26 00:08:17 +09:00
|
|
|
describe('ChoiceWidgetAnnotation', function() {
|
2018-08-12 01:50:31 +09:00
|
|
|
let choiceWidgetDict;
|
2016-09-26 00:08:17 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
beforeEach(function(done) {
|
2016-09-26 00:08:17 +09:00
|
|
|
choiceWidgetDict = new Dict();
|
|
|
|
choiceWidgetDict.set('Type', Name.get('Annot'));
|
|
|
|
choiceWidgetDict.set('Subtype', Name.get('Widget'));
|
|
|
|
choiceWidgetDict.set('FT', Name.get('Ch'));
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
afterEach(function() {
|
2016-09-26 00:08:17 +09:00
|
|
|
choiceWidgetDict = null;
|
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle missing option arrays', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(122, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-26 00:08:17 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.options).toEqual([]);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle option arrays with array elements', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const optionBarRef = new Ref(20, 0);
|
|
|
|
const optionBarStr = 'Bar';
|
|
|
|
const optionOneRef = new Ref(10, 0);
|
|
|
|
const optionOneArr = ['bar_export', optionBarRef];
|
2016-12-17 21:34:18 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const options = [['foo_export', 'Foo'], optionOneRef];
|
|
|
|
const expected = [
|
2017-11-19 00:47:32 +09:00
|
|
|
{ exportValue: 'foo_export', displayValue: 'Foo', },
|
|
|
|
{ exportValue: 'bar_export', displayValue: 'Bar', },
|
2016-09-26 00:08:17 +09:00
|
|
|
];
|
|
|
|
|
|
|
|
choiceWidgetDict.set('Opt', options);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(123, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-12-17 21:34:18 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, },
|
|
|
|
{ ref: optionBarRef, data: optionBarStr, },
|
|
|
|
{ ref: optionOneRef, data: optionOneArr, },
|
2016-09-26 00:08:17 +09:00
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.options).toEqual(expected);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle option arrays with string elements', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const optionBarRef = new Ref(10, 0);
|
|
|
|
const optionBarStr = 'Bar';
|
2016-12-17 21:34:18 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const options = ['Foo', optionBarRef];
|
|
|
|
const expected = [
|
2017-11-19 00:47:32 +09:00
|
|
|
{ exportValue: 'Foo', displayValue: 'Foo', },
|
|
|
|
{ exportValue: 'Bar', displayValue: 'Bar', },
|
2016-09-26 00:08:17 +09:00
|
|
|
];
|
|
|
|
|
|
|
|
choiceWidgetDict.set('Opt', options);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(981, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-12-17 21:34:18 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, },
|
|
|
|
{ ref: optionBarRef, data: optionBarStr, }
|
2016-09-26 00:08:17 +09:00
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.options).toEqual(expected);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle inherited option arrays (issue 8094)', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const options = [
|
2017-02-26 07:34:26 +09:00
|
|
|
['Value1', 'Description1'],
|
|
|
|
['Value2', 'Description2'],
|
|
|
|
];
|
2018-08-12 01:50:31 +09:00
|
|
|
const expected = [
|
Fix inconsistent spacing and trailing commas in objects in `test/` files, so we can enable the `comma-dangle` and `object-curly-spacing` ESLint rules later on
http://eslint.org/docs/rules/comma-dangle
http://eslint.org/docs/rules/object-curly-spacing
Given that we currently have quite inconsistent object formatting, fixing this in *one* big patch probably wouldn't be feasible (since I cannot imagine anyone wanting to review that); hence I've opted to try and do this piecewise instead.
Please note: This patch was created automatically, using the ESLint `--fix` command line option. In a couple of places this caused lines to become too long, and I've fixed those manually; please refer to the interdiff below for the only hand-edits in this patch.
```diff
diff --git a/test/chromium/test-telemetry.js b/test/chromium/test-telemetry.js
index cc412a31..2e5bdfa1 100755
--- a/test/chromium/test-telemetry.js
+++ b/test/chromium/test-telemetry.js
@@ -324,7 +324,7 @@ var tests = [
var window = createExtensionGlobal();
telemetryScript.runInNewContext(window);
window.chrome.runtime.getManifest = function() {
- return { version: '1.0.1', };
+ return { version: '1.0.1', };
};
window.Date.test_now_value += 12 * 36E5;
telemetryScript.runInNewContext(window);
diff --git a/test/unit/api_spec.js b/test/unit/api_spec.js
index 1f00747a..f22988e7 100644
--- a/test/unit/api_spec.js
+++ b/test/unit/api_spec.js
@@ -503,8 +503,9 @@ describe('api', function() {
it('gets destinations, from /Dests dictionary', function(done) {
var promise = doc.getDestinations();
promise.then(function(data) {
- expect(data).toEqual({ chapter1: [{ gen: 0, num: 17, }, { name: 'XYZ', },
- 0, 841.89, null], });
+ expect(data).toEqual({
+ chapter1: [{ gen: 0, num: 17, }, { name: 'XYZ', }, 0, 841.89, null],
+ });
done();
}).catch(function (reason) {
done.fail(reason);
diff --git a/test/unit/function_spec.js b/test/unit/function_spec.js
index 66441212..62127eb9 100644
--- a/test/unit/function_spec.js
+++ b/test/unit/function_spec.js
@@ -492,9 +492,11 @@ describe('function', function() {
it('check compiled mul', function() {
check([0.25, 0.5, 'mul'], [], [0, 1], [{ input: [], output: [0.125], }]);
check([0, 'mul'], [0, 1], [0, 1], [{ input: [0.25], output: [0], }]);
- check([0.5, 'mul'], [0, 1], [0, 1], [{ input: [0.25], output: [0.125], }]);
+ check([0.5, 'mul'], [0, 1], [0, 1],
+ [{ input: [0.25], output: [0.125], }]);
check([1, 'mul'], [0, 1], [0, 1], [{ input: [0.25], output: [0.25], }]);
- check([0, 'exch', 'mul'], [0, 1], [0, 1], [{ input: [0.25], output: [0], }]);
+ check([0, 'exch', 'mul'], [0, 1], [0, 1],
+ [{ input: [0.25], output: [0], }]);
check([0.5, 'exch', 'mul'], [0, 1], [0, 1],
[{ input: [0.25], output: [0.125], }]);
check([1, 'exch', 'mul'], [0, 1], [0, 1],
```
2017-06-02 19:55:01 +09:00
|
|
|
{ exportValue: 'Value1', displayValue: 'Description1', },
|
|
|
|
{ exportValue: 'Value2', displayValue: 'Description2', },
|
2017-02-26 07:34:26 +09:00
|
|
|
];
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const parentDict = new Dict();
|
2017-02-26 07:34:26 +09:00
|
|
|
parentDict.set('Opt', options);
|
|
|
|
|
|
|
|
choiceWidgetDict.set('Parent', parentDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(123, 0);
|
|
|
|
const xref = new XRefMock([
|
2017-02-26 07:34:26 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, },
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.options).toEqual(expected);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2017-11-19 00:47:32 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should sanitize display values in option arrays (issue 8947)',
|
2018-03-21 09:43:40 +09:00
|
|
|
function(done) {
|
2017-11-19 00:47:32 +09:00
|
|
|
// The option value is a UTF-16BE string. The display value should be
|
|
|
|
// sanitized, but the export value should remain the same since that
|
|
|
|
// may be used as a unique identifier when exporting form values.
|
2018-08-12 01:50:31 +09:00
|
|
|
const options = ['\xFE\xFF\x00F\x00o\x00o'];
|
|
|
|
const expected = [
|
2017-11-19 00:47:32 +09:00
|
|
|
{ exportValue: '\xFE\xFF\x00F\x00o\x00o', displayValue: 'Foo', },
|
|
|
|
];
|
|
|
|
|
|
|
|
choiceWidgetDict.set('Opt', options);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(984, 0);
|
|
|
|
const xref = new XRefMock([
|
2017-11-19 00:47:32 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, },
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.options).toEqual(expected);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2017-02-26 07:34:26 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle array field values', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const fieldValue = ['Foo', 'Bar'];
|
2016-09-26 00:08:17 +09:00
|
|
|
|
|
|
|
choiceWidgetDict.set('V', fieldValue);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(968, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-26 00:08:17 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.fieldValue).toEqual(fieldValue);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle string field values', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const fieldValue = 'Foo';
|
2016-09-26 00:08:17 +09:00
|
|
|
|
|
|
|
choiceWidgetDict.set('V', fieldValue);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(978, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-26 00:08:17 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.fieldValue).toEqual([fieldValue]);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should handle unknown flags', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(166, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-26 00:08:17 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.readOnly).toEqual(false);
|
|
|
|
expect(data.combo).toEqual(false);
|
|
|
|
expect(data.multiSelect).toEqual(false);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should not set invalid flags', function(done) {
|
2016-09-26 00:08:17 +09:00
|
|
|
choiceWidgetDict.set('Ff', 'readonly');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(165, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-26 00:08:17 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.readOnly).toEqual(false);
|
|
|
|
expect(data.combo).toEqual(false);
|
|
|
|
expect(data.multiSelect).toEqual(false);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should set valid flags', function(done) {
|
2016-09-26 00:08:17 +09:00
|
|
|
choiceWidgetDict.set('Ff', AnnotationFieldFlag.READONLY +
|
|
|
|
AnnotationFieldFlag.COMBO +
|
|
|
|
AnnotationFieldFlag.MULTISELECT);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const choiceWidgetRef = new Ref(512, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-09-26 00:08:17 +09:00
|
|
|
{ ref: choiceWidgetRef, data: choiceWidgetDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, choiceWidgetRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.WIDGET);
|
|
|
|
expect(data.readOnly).toEqual(true);
|
|
|
|
expect(data.combo).toEqual(true);
|
|
|
|
expect(data.multiSelect).toEqual(true);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-09-26 00:08:17 +09:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2017-04-03 03:50:17 +09:00
|
|
|
describe('LineAnnotation', function() {
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should set the line coordinates', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const lineDict = new Dict();
|
2017-04-03 03:50:17 +09:00
|
|
|
lineDict.set('Type', Name.get('Annot'));
|
|
|
|
lineDict.set('Subtype', Name.get('Line'));
|
|
|
|
lineDict.set('L', [1, 2, 3, 4]);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const lineRef = new Ref(122, 0);
|
|
|
|
const xref = new XRefMock([
|
2017-04-03 03:50:17 +09:00
|
|
|
{ ref: lineRef, data: lineDict, }
|
|
|
|
]);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, lineRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.LINE);
|
|
|
|
expect(data.lineCoordinates).toEqual([1, 2, 3, 4]);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2017-04-03 03:50:17 +09:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-02-15 05:27:53 +09:00
|
|
|
describe('FileAttachmentAnnotation', function() {
|
2018-03-21 09:43:40 +09:00
|
|
|
it('should correctly parse a file attachment', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const fileStream = new StringStream(
|
2017-01-12 22:11:22 +09:00
|
|
|
'<<\n' +
|
|
|
|
'/Type /EmbeddedFile\n' +
|
|
|
|
'/Subtype /text#2Fplain\n' +
|
|
|
|
'>>\n' +
|
|
|
|
'stream\n' +
|
|
|
|
'Test attachment' +
|
|
|
|
'endstream\n'
|
|
|
|
);
|
2018-08-12 01:50:31 +09:00
|
|
|
const lexer = new Lexer(fileStream);
|
|
|
|
const parser = new Parser(lexer, /* allowStreams = */ true);
|
2016-02-15 05:27:53 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const fileStreamRef = new Ref(18, 0);
|
|
|
|
const fileStreamDict = parser.getObj();
|
2016-02-15 05:27:53 +09:00
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const embeddedFileDict = new Dict();
|
2017-01-12 22:11:22 +09:00
|
|
|
embeddedFileDict.set('F', fileStreamRef);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const fileSpecRef = new Ref(19, 0);
|
|
|
|
const fileSpecDict = new Dict();
|
2017-01-12 22:11:22 +09:00
|
|
|
fileSpecDict.set('Type', Name.get('Filespec'));
|
|
|
|
fileSpecDict.set('Desc', '');
|
|
|
|
fileSpecDict.set('EF', embeddedFileDict);
|
|
|
|
fileSpecDict.set('UF', 'Test.txt');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const fileAttachmentRef = new Ref(20, 0);
|
|
|
|
const fileAttachmentDict = new Dict();
|
2017-01-12 22:11:22 +09:00
|
|
|
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');
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const xref = new XRefMock([
|
2017-01-12 22:11:22 +09:00
|
|
|
{ ref: fileStreamRef, data: fileStreamDict, },
|
|
|
|
{ ref: fileSpecRef, data: fileSpecDict, },
|
|
|
|
{ ref: fileAttachmentRef, data: fileAttachmentDict, }
|
|
|
|
]);
|
|
|
|
embeddedFileDict.assignXref(xref);
|
|
|
|
fileSpecDict.assignXref(xref);
|
|
|
|
fileAttachmentDict.assignXref(xref);
|
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, fileAttachmentRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
expect(data.annotationType).toEqual(AnnotationType.FILEATTACHMENT);
|
|
|
|
expect(data.file.filename).toEqual('Test.txt');
|
|
|
|
expect(data.file.content).toEqual(stringToBytes('Test attachment'));
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-02-15 05:27:53 +09:00
|
|
|
});
|
|
|
|
});
|
2016-05-25 00:35:45 +09:00
|
|
|
|
|
|
|
describe('PopupAnnotation', function() {
|
|
|
|
it('should inherit the parent flags when the Popup is not viewable, ' +
|
2018-03-21 09:43:40 +09:00
|
|
|
'but the parent is (PR 7352)', function(done) {
|
2018-08-12 01:50:31 +09:00
|
|
|
const parentDict = new Dict();
|
2016-05-25 00:35:45 +09:00
|
|
|
parentDict.set('Type', Name.get('Annot'));
|
|
|
|
parentDict.set('Subtype', Name.get('Text'));
|
|
|
|
parentDict.set('F', 28); // viewable
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const popupDict = new Dict();
|
2016-05-25 00:35:45 +09:00
|
|
|
popupDict.set('Type', Name.get('Annot'));
|
|
|
|
popupDict.set('Subtype', Name.get('Popup'));
|
|
|
|
popupDict.set('F', 25); // not viewable
|
|
|
|
popupDict.set('Parent', parentDict);
|
|
|
|
|
2018-08-12 01:50:31 +09:00
|
|
|
const popupRef = new Ref(13, 0);
|
|
|
|
const xref = new XRefMock([
|
2016-08-26 23:01:25 +09:00
|
|
|
{ ref: popupRef, data: popupDict, }
|
|
|
|
]);
|
2016-05-25 00:35:45 +09:00
|
|
|
|
2018-03-21 09:43:40 +09:00
|
|
|
AnnotationFactory.create(xref, popupRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, viewable, }) => {
|
|
|
|
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);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
2016-05-25 00:35:45 +09:00
|
|
|
});
|
|
|
|
});
|
2018-09-30 23:29:16 +09:00
|
|
|
|
|
|
|
describe('InkAnnotation', function() {
|
|
|
|
it('should handle a single ink list', function(done) {
|
|
|
|
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 = new Ref(142, 0);
|
|
|
|
const xref = new XRefMock([
|
|
|
|
{ ref: inkRef, data: inkDict, }
|
|
|
|
]);
|
|
|
|
|
|
|
|
AnnotationFactory.create(xref, inkRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
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, },
|
|
|
|
]);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle multiple ink lists', function(done) {
|
|
|
|
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 = new Ref(143, 0);
|
|
|
|
const xref = new XRefMock([
|
|
|
|
{ ref: inkRef, data: inkDict, }
|
|
|
|
]);
|
|
|
|
|
|
|
|
AnnotationFactory.create(xref, inkRef, pdfManagerMock,
|
|
|
|
idFactoryMock).then(({ data, }) => {
|
|
|
|
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, }
|
|
|
|
]);
|
|
|
|
done();
|
|
|
|
}, done.fail);
|
|
|
|
});
|
|
|
|
});
|
2014-12-26 05:04:01 +09:00
|
|
|
});
|