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:
Tim van der Meij 2019-03-06 23:46:53 +01:00 committed by GitHub
commit e1b01a601c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 143 additions and 43 deletions

View File

@ -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;

View File

@ -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');
}); });
}); });