Merge pull request #10605 from timvandermeij/display-utils
Convert `let` to `const` if possible in, and improve unit test coverage for, `src/display/display_utils.js`
This commit is contained in:
commit
e1b01a601c
@ -12,6 +12,7 @@
|
|||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
/* eslint no-var: error */
|
||||||
|
|
||||||
import {
|
import {
|
||||||
assert, CMapCompressionType, removeNullCharacters, stringToBytes,
|
assert, CMapCompressionType, removeNullCharacters, stringToBytes,
|
||||||
@ -24,10 +25,10 @@ const SVG_NS = 'http://www.w3.org/2000/svg';
|
|||||||
class DOMCanvasFactory {
|
class DOMCanvasFactory {
|
||||||
create(width, height) {
|
create(width, height) {
|
||||||
if (width <= 0 || height <= 0) {
|
if (width <= 0 || height <= 0) {
|
||||||
throw new Error('invalid canvas size');
|
throw new Error('Invalid canvas size');
|
||||||
}
|
}
|
||||||
let canvas = document.createElement('canvas');
|
const canvas = document.createElement('canvas');
|
||||||
let context = canvas.getContext('2d');
|
const context = canvas.getContext('2d');
|
||||||
canvas.width = width;
|
canvas.width = width;
|
||||||
canvas.height = height;
|
canvas.height = height;
|
||||||
return {
|
return {
|
||||||
@ -38,10 +39,10 @@ class DOMCanvasFactory {
|
|||||||
|
|
||||||
reset(canvasAndContext, width, height) {
|
reset(canvasAndContext, width, height) {
|
||||||
if (!canvasAndContext.canvas) {
|
if (!canvasAndContext.canvas) {
|
||||||
throw new Error('canvas is not specified');
|
throw new Error('Canvas is not specified');
|
||||||
}
|
}
|
||||||
if (width <= 0 || height <= 0) {
|
if (width <= 0 || height <= 0) {
|
||||||
throw new Error('invalid canvas size');
|
throw new Error('Invalid canvas size');
|
||||||
}
|
}
|
||||||
canvasAndContext.canvas.width = width;
|
canvasAndContext.canvas.width = width;
|
||||||
canvasAndContext.canvas.height = height;
|
canvasAndContext.canvas.height = height;
|
||||||
@ -49,7 +50,7 @@ class DOMCanvasFactory {
|
|||||||
|
|
||||||
destroy(canvasAndContext) {
|
destroy(canvasAndContext) {
|
||||||
if (!canvasAndContext.canvas) {
|
if (!canvasAndContext.canvas) {
|
||||||
throw new Error('canvas is not specified');
|
throw new Error('Canvas is not specified');
|
||||||
}
|
}
|
||||||
// Zeroing the width and height cause Firefox to release graphics
|
// Zeroing the width and height cause Firefox to release graphics
|
||||||
// resources immediately, which can greatly reduce memory consumption.
|
// resources immediately, which can greatly reduce memory consumption.
|
||||||
@ -137,7 +138,7 @@ class DOMSVGFactory {
|
|||||||
create(width, height) {
|
create(width, height) {
|
||||||
assert(width > 0 && height > 0, 'Invalid SVG dimensions');
|
assert(width > 0 && height > 0, 'Invalid SVG dimensions');
|
||||||
|
|
||||||
let svg = document.createElementNS(SVG_NS, 'svg:svg');
|
const svg = document.createElementNS(SVG_NS, 'svg:svg');
|
||||||
svg.setAttribute('version', '1.1');
|
svg.setAttribute('version', '1.1');
|
||||||
svg.setAttribute('width', width + 'px');
|
svg.setAttribute('width', width + 'px');
|
||||||
svg.setAttribute('height', height + 'px');
|
svg.setAttribute('height', height + 'px');
|
||||||
@ -194,8 +195,8 @@ class PageViewport {
|
|||||||
|
|
||||||
// creating transform to convert pdf coordinate system to the normal
|
// creating transform to convert pdf coordinate system to the normal
|
||||||
// canvas like coordinates taking in account scale and rotation
|
// canvas like coordinates taking in account scale and rotation
|
||||||
let centerX = (viewBox[2] + viewBox[0]) / 2;
|
const centerX = (viewBox[2] + viewBox[0]) / 2;
|
||||||
let centerY = (viewBox[3] + viewBox[1]) / 2;
|
const centerY = (viewBox[3] + viewBox[1]) / 2;
|
||||||
let rotateA, rotateB, rotateC, rotateD;
|
let rotateA, rotateB, rotateC, rotateD;
|
||||||
rotation = rotation % 360;
|
rotation = rotation % 360;
|
||||||
rotation = rotation < 0 ? rotation + 360 : rotation;
|
rotation = rotation < 0 ? rotation + 360 : rotation;
|
||||||
@ -287,9 +288,9 @@ class PageViewport {
|
|||||||
* @see {@link convertToViewportPoint}
|
* @see {@link convertToViewportPoint}
|
||||||
*/
|
*/
|
||||||
convertToViewportRectangle(rect) {
|
convertToViewportRectangle(rect) {
|
||||||
let tl = Util.applyTransform([rect[0], rect[1]], this.transform);
|
const topLeft = Util.applyTransform([rect[0], rect[1]], this.transform);
|
||||||
let br = Util.applyTransform([rect[2], rect[3]], this.transform);
|
const bottomRight = Util.applyTransform([rect[2], rect[3]], this.transform);
|
||||||
return [tl[0], tl[1], br[0], br[1]];
|
return [topLeft[0], topLeft[1], bottomRight[0], bottomRight[1]];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -306,7 +307,7 @@ class PageViewport {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var RenderingCancelledException = (function RenderingCancelledException() {
|
const RenderingCancelledException = (function RenderingCancelledException() {
|
||||||
function RenderingCancelledException(msg, type) {
|
function RenderingCancelledException(msg, type) {
|
||||||
this.message = msg;
|
this.message = msg;
|
||||||
this.type = type;
|
this.type = type;
|
||||||
@ -332,7 +333,7 @@ const LinkTargetStringMap = [
|
|||||||
'_self',
|
'_self',
|
||||||
'_blank',
|
'_blank',
|
||||||
'_parent',
|
'_parent',
|
||||||
'_top'
|
'_top',
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -355,7 +356,7 @@ function addLinkAttributes(link, { url, target, rel, } = {}) {
|
|||||||
|
|
||||||
if (url) {
|
if (url) {
|
||||||
const LinkTargetValues = Object.values(LinkTarget);
|
const LinkTargetValues = Object.values(LinkTarget);
|
||||||
let targetIndex =
|
const targetIndex =
|
||||||
LinkTargetValues.includes(target) ? target : LinkTarget.NONE;
|
LinkTargetValues.includes(target) ? target : LinkTarget.NONE;
|
||||||
link.target = LinkTargetStringMap[targetIndex];
|
link.target = LinkTargetStringMap[targetIndex];
|
||||||
|
|
||||||
@ -365,11 +366,10 @@ function addLinkAttributes(link, { url, target, rel, } = {}) {
|
|||||||
|
|
||||||
// Gets the file name from a given URL.
|
// Gets the file name from a given URL.
|
||||||
function getFilenameFromUrl(url) {
|
function getFilenameFromUrl(url) {
|
||||||
var anchor = url.indexOf('#');
|
const anchor = url.indexOf('#');
|
||||||
var query = url.indexOf('?');
|
const query = url.indexOf('?');
|
||||||
var end = Math.min(
|
const end = Math.min(anchor > 0 ? anchor : url.length,
|
||||||
anchor > 0 ? anchor : url.length,
|
query > 0 ? query : url.length);
|
||||||
query > 0 ? query : url.length);
|
|
||||||
return url.substring(url.lastIndexOf('/', end) + 1, end);
|
return url.substring(url.lastIndexOf('/', end) + 1, end);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -407,19 +407,17 @@ class StatTimer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
toString() {
|
toString() {
|
||||||
let times = this.times;
|
|
||||||
// Find the longest name for padding purposes.
|
// Find the longest name for padding purposes.
|
||||||
let out = '', longest = 0;
|
let out = '', longest = 0;
|
||||||
for (let i = 0, ii = times.length; i < ii; ++i) {
|
for (const time of this.times) {
|
||||||
let name = times[i]['name'];
|
const name = time.name;
|
||||||
if (name.length > longest) {
|
if (name.length > longest) {
|
||||||
longest = name.length;
|
longest = name.length;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (let i = 0, ii = times.length; i < ii; ++i) {
|
for (const time of this.times) {
|
||||||
let span = times[i];
|
const duration = time.end - time.start;
|
||||||
let duration = span.end - span.start;
|
out += `${time.name.padEnd(longest)} ${duration}ms\n`;
|
||||||
out += `${span['name'].padEnd(longest)} ${duration}ms\n`;
|
|
||||||
}
|
}
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
@ -466,7 +464,7 @@ function isValidFetchUrl(url, baseUrl) {
|
|||||||
|
|
||||||
function loadScript(src) {
|
function loadScript(src) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
let script = document.createElement('script');
|
const script = document.createElement('script');
|
||||||
script.src = src;
|
script.src = src;
|
||||||
|
|
||||||
script.onload = resolve;
|
script.onload = resolve;
|
||||||
|
@ -12,22 +12,120 @@
|
|||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
/* eslint no-var: error */
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DOMSVGFactory, getFilenameFromUrl, isValidFetchUrl
|
DOMCanvasFactory, DOMSVGFactory, getFilenameFromUrl, isValidFetchUrl
|
||||||
} from '../../src/display/display_utils';
|
} from '../../src/display/display_utils';
|
||||||
import isNodeJS from '../../src/shared/is_node';
|
import isNodeJS from '../../src/shared/is_node';
|
||||||
|
|
||||||
describe('display_utils', function() {
|
describe('display_utils', function() {
|
||||||
|
describe('DOMCanvasFactory', function() {
|
||||||
|
let canvasFactory;
|
||||||
|
|
||||||
|
beforeAll(function(done) {
|
||||||
|
canvasFactory = new DOMCanvasFactory();
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(function() {
|
||||||
|
canvasFactory = null;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('`create` should throw an error if the dimensions are invalid',
|
||||||
|
function() {
|
||||||
|
// Invalid width.
|
||||||
|
expect(function() {
|
||||||
|
return canvasFactory.create(-1, 1);
|
||||||
|
}).toThrow(new Error('Invalid canvas size'));
|
||||||
|
|
||||||
|
// Invalid height.
|
||||||
|
expect(function() {
|
||||||
|
return canvasFactory.create(1, -1);
|
||||||
|
}).toThrow(new Error('Invalid canvas size'));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('`create` should return a canvas if the dimensions are valid',
|
||||||
|
function() {
|
||||||
|
if (isNodeJS()) {
|
||||||
|
pending('Document is not supported in Node.js.');
|
||||||
|
}
|
||||||
|
|
||||||
|
const { canvas, context, } = canvasFactory.create(20, 40);
|
||||||
|
expect(canvas instanceof HTMLCanvasElement).toBe(true);
|
||||||
|
expect(context instanceof CanvasRenderingContext2D).toBe(true);
|
||||||
|
expect(canvas.width).toBe(20);
|
||||||
|
expect(canvas.height).toBe(40);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('`reset` should throw an error if no canvas is provided', function() {
|
||||||
|
const canvasAndContext = { canvas: null, context: null, };
|
||||||
|
|
||||||
|
expect(function() {
|
||||||
|
return canvasFactory.reset(canvasAndContext, 20, 40);
|
||||||
|
}).toThrow(new Error('Canvas is not specified'));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('`reset` should throw an error if the dimensions are invalid',
|
||||||
|
function() {
|
||||||
|
const canvasAndContext = { canvas: 'foo', context: 'bar', };
|
||||||
|
|
||||||
|
// Invalid width.
|
||||||
|
expect(function() {
|
||||||
|
return canvasFactory.reset(canvasAndContext, -1, 1);
|
||||||
|
}).toThrow(new Error('Invalid canvas size'));
|
||||||
|
|
||||||
|
// Invalid height.
|
||||||
|
expect(function() {
|
||||||
|
return canvasFactory.reset(canvasAndContext, 1, -1);
|
||||||
|
}).toThrow(new Error('Invalid canvas size'));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('`reset` should alter the canvas/context if the dimensions are valid',
|
||||||
|
function() {
|
||||||
|
if (isNodeJS()) {
|
||||||
|
pending('Document is not supported in Node.js.');
|
||||||
|
}
|
||||||
|
|
||||||
|
const canvasAndContext = canvasFactory.create(20, 40);
|
||||||
|
canvasFactory.reset(canvasAndContext, 60, 80);
|
||||||
|
|
||||||
|
const { canvas, context, } = canvasAndContext;
|
||||||
|
expect(canvas instanceof HTMLCanvasElement).toBe(true);
|
||||||
|
expect(context instanceof CanvasRenderingContext2D).toBe(true);
|
||||||
|
expect(canvas.width).toBe(60);
|
||||||
|
expect(canvas.height).toBe(80);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('`destroy` should throw an error if no canvas is provided', function() {
|
||||||
|
expect(function() {
|
||||||
|
return canvasFactory.destroy({});
|
||||||
|
}).toThrow(new Error('Canvas is not specified'));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('`destroy` should clear the canvas/context', function() {
|
||||||
|
if (isNodeJS()) {
|
||||||
|
pending('Document is not supported in Node.js.');
|
||||||
|
}
|
||||||
|
|
||||||
|
const canvasAndContext = canvasFactory.create(20, 40);
|
||||||
|
canvasFactory.destroy(canvasAndContext);
|
||||||
|
|
||||||
|
const { canvas, context, } = canvasAndContext;
|
||||||
|
expect(canvas).toBe(null);
|
||||||
|
expect(context).toBe(null);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('DOMSVGFactory', function() {
|
describe('DOMSVGFactory', function() {
|
||||||
let svgFactory;
|
let svgFactory;
|
||||||
|
|
||||||
beforeAll(function (done) {
|
beforeAll(function(done) {
|
||||||
svgFactory = new DOMSVGFactory();
|
svgFactory = new DOMSVGFactory();
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
afterAll(function () {
|
afterAll(function() {
|
||||||
svgFactory = null;
|
svgFactory = null;
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -50,8 +148,7 @@ describe('display_utils', function() {
|
|||||||
pending('Document is not supported in Node.js.');
|
pending('Document is not supported in Node.js.');
|
||||||
}
|
}
|
||||||
|
|
||||||
let svg = svgFactory.create(20, 40);
|
const svg = svgFactory.create(20, 40);
|
||||||
|
|
||||||
expect(svg instanceof SVGSVGElement).toBe(true);
|
expect(svg instanceof SVGSVGElement).toBe(true);
|
||||||
expect(svg.getAttribute('version')).toBe('1.1');
|
expect(svg.getAttribute('version')).toBe('1.1');
|
||||||
expect(svg.getAttribute('width')).toBe('20px');
|
expect(svg.getAttribute('width')).toBe('20px');
|
||||||
@ -73,25 +170,30 @@ describe('display_utils', function() {
|
|||||||
pending('Document is not supported in Node.js.');
|
pending('Document is not supported in Node.js.');
|
||||||
}
|
}
|
||||||
|
|
||||||
let svg = svgFactory.createElement('svg:rect');
|
const svg = svgFactory.createElement('svg:rect');
|
||||||
|
|
||||||
expect(svg instanceof SVGRectElement).toBe(true);
|
expect(svg instanceof SVGRectElement).toBe(true);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getFilenameFromUrl', function() {
|
describe('getFilenameFromUrl', function() {
|
||||||
it('should get the filename from an absolute URL', function() {
|
it('should get the filename from an absolute URL', function() {
|
||||||
var url = 'http://server.org/filename.pdf';
|
const url = 'https://server.org/filename.pdf';
|
||||||
var result = getFilenameFromUrl(url);
|
expect(getFilenameFromUrl(url)).toEqual('filename.pdf');
|
||||||
var expected = 'filename.pdf';
|
|
||||||
expect(result).toEqual(expected);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should get the filename from a relative URL', function() {
|
it('should get the filename from a relative URL', function() {
|
||||||
var url = '../../filename.pdf';
|
const url = '../../filename.pdf';
|
||||||
var result = getFilenameFromUrl(url);
|
expect(getFilenameFromUrl(url)).toEqual('filename.pdf');
|
||||||
var expected = 'filename.pdf';
|
});
|
||||||
expect(result).toEqual(expected);
|
|
||||||
|
it('should get the filename from a URL with an anchor', function() {
|
||||||
|
const url = 'https://server.org/filename.pdf#foo';
|
||||||
|
expect(getFilenameFromUrl(url)).toEqual('filename.pdf');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should get the filename from a URL with query parameters', function() {
|
||||||
|
const url = 'https://server.org/filename.pdf?foo=bar';
|
||||||
|
expect(getFilenameFromUrl(url)).toEqual('filename.pdf');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user