941b65f683
Given that the API will now, after PR 12039, automatically pick the correct factories to use depending on the environment (browser vs. Node.js), we can utilize that in the unit-tests as well. This way we don't have to manually repeat the same initialization code in *multiple* unit-tests. *Note:* The *official* PDF.js API is defined in `src/pdf.js`, hence the new exports in `src/display/api.js` will not affect that. Also, updates the unit-test `FileReaderFactory` helpers similarily. *Drive-by change:* Fix the `CMapReaderFactory` usage in the annotation unit-tests, since the cache should only contain raw data and not a Promise. While this obviously works as-is, having unit-tests that "abuse" the intended data format can easily lead to unnecessary failures if changes are made to the relevant `src/core/` code.
226 lines
6.3 KiB
JavaScript
226 lines
6.3 KiB
JavaScript
/* Copyright 2017 Mozilla Foundation
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
import { DefaultCanvasFactory, getDocument } from "../../src/display/api.js";
|
|
import { buildGetDocumentParams } from "./test_utils.js";
|
|
|
|
function getTopLeftPixel(canvasContext) {
|
|
const imgData = canvasContext.getImageData(0, 0, 1, 1);
|
|
return {
|
|
r: imgData.data[0],
|
|
g: imgData.data[1],
|
|
b: imgData.data[2],
|
|
a: imgData.data[3],
|
|
};
|
|
}
|
|
|
|
describe("custom canvas rendering", function () {
|
|
const transparentGetDocumentParams = buildGetDocumentParams(
|
|
"transparent.pdf"
|
|
);
|
|
|
|
let CanvasFactory;
|
|
let loadingTask;
|
|
let page;
|
|
|
|
beforeAll(function (done) {
|
|
CanvasFactory = new DefaultCanvasFactory();
|
|
|
|
loadingTask = getDocument(transparentGetDocumentParams);
|
|
loadingTask.promise
|
|
.then(function (doc) {
|
|
return doc.getPage(1);
|
|
})
|
|
.then(function (data) {
|
|
page = data;
|
|
done();
|
|
})
|
|
.catch(done.fail);
|
|
});
|
|
|
|
afterAll(function (done) {
|
|
CanvasFactory = null;
|
|
page = null;
|
|
loadingTask.destroy().then(done);
|
|
});
|
|
|
|
it("renders to canvas with a default white background", function (done) {
|
|
const viewport = page.getViewport({ scale: 1 });
|
|
const canvasAndCtx = CanvasFactory.create(viewport.width, viewport.height);
|
|
|
|
const renderTask = page.render({
|
|
canvasContext: canvasAndCtx.context,
|
|
viewport,
|
|
});
|
|
renderTask.promise
|
|
.then(function () {
|
|
expect(getTopLeftPixel(canvasAndCtx.context)).toEqual({
|
|
r: 255,
|
|
g: 255,
|
|
b: 255,
|
|
a: 255,
|
|
});
|
|
CanvasFactory.destroy(canvasAndCtx);
|
|
done();
|
|
})
|
|
.catch(done.fail);
|
|
});
|
|
|
|
it("renders to canvas with a custom background", function (done) {
|
|
const viewport = page.getViewport({ scale: 1 });
|
|
const canvasAndCtx = CanvasFactory.create(viewport.width, viewport.height);
|
|
|
|
const renderTask = page.render({
|
|
canvasContext: canvasAndCtx.context,
|
|
viewport,
|
|
background: "rgba(255,0,0,1.0)",
|
|
});
|
|
renderTask.promise
|
|
.then(function () {
|
|
expect(getTopLeftPixel(canvasAndCtx.context)).toEqual({
|
|
r: 255,
|
|
g: 0,
|
|
b: 0,
|
|
a: 255,
|
|
});
|
|
CanvasFactory.destroy(canvasAndCtx);
|
|
done();
|
|
})
|
|
.catch(done.fail);
|
|
});
|
|
});
|
|
|
|
describe("custom ownerDocument", function () {
|
|
const FontFace = globalThis.FontFace;
|
|
|
|
const checkFont = font => /g_d\d+_f1/.test(font.family);
|
|
const checkFontFaceRule = rule =>
|
|
/^@font-face {font-family:"g_d\d+_f1";src:/.test(rule);
|
|
|
|
beforeEach(() => {
|
|
globalThis.FontFace = function MockFontFace(name) {
|
|
this.family = name;
|
|
};
|
|
});
|
|
|
|
afterEach(() => {
|
|
globalThis.FontFace = FontFace;
|
|
});
|
|
|
|
function getMocks() {
|
|
const elements = [];
|
|
const createElement = name => {
|
|
let element =
|
|
typeof document !== "undefined" && document.createElement(name);
|
|
if (name === "style") {
|
|
element = {
|
|
tagName: name,
|
|
sheet: {
|
|
cssRules: [],
|
|
insertRule(rule) {
|
|
this.cssRules.push(rule);
|
|
},
|
|
},
|
|
};
|
|
Object.assign(element, {
|
|
remove() {
|
|
this.remove.called = true;
|
|
},
|
|
});
|
|
}
|
|
elements.push(element);
|
|
return element;
|
|
};
|
|
const ownerDocument = {
|
|
fonts: new Set(),
|
|
createElement,
|
|
documentElement: {
|
|
getElementsByTagName: () => [{ appendChild: () => {} }],
|
|
},
|
|
};
|
|
const CanvasFactory = new DefaultCanvasFactory({ ownerDocument });
|
|
|
|
return {
|
|
elements,
|
|
ownerDocument,
|
|
CanvasFactory,
|
|
};
|
|
}
|
|
|
|
it("should use given document for loading fonts (with Font Loading API)", async function () {
|
|
const { ownerDocument, elements, CanvasFactory } = getMocks();
|
|
const getDocumentParams = buildGetDocumentParams(
|
|
"TrueType_without_cmap.pdf",
|
|
{
|
|
disableFontFace: false,
|
|
ownerDocument,
|
|
}
|
|
);
|
|
|
|
const loadingTask = getDocument(getDocumentParams);
|
|
const doc = await loadingTask.promise;
|
|
const page = await doc.getPage(1);
|
|
|
|
const viewport = page.getViewport({ scale: 1 });
|
|
const canvasAndCtx = CanvasFactory.create(viewport.width, viewport.height);
|
|
|
|
await page.render({
|
|
canvasContext: canvasAndCtx.context,
|
|
viewport,
|
|
}).promise;
|
|
|
|
const style = elements.find(element => element.tagName === "style");
|
|
expect(style).toBeFalsy();
|
|
expect(ownerDocument.fonts.size).toBeGreaterThanOrEqual(1);
|
|
expect(Array.from(ownerDocument.fonts).find(checkFont)).toBeTruthy();
|
|
await doc.destroy();
|
|
await loadingTask.destroy();
|
|
CanvasFactory.destroy(canvasAndCtx);
|
|
expect(ownerDocument.fonts.size).toBe(0);
|
|
});
|
|
|
|
it("should use given document for loading fonts (with CSS rules)", async function () {
|
|
const { ownerDocument, elements, CanvasFactory } = getMocks();
|
|
ownerDocument.fonts = null;
|
|
const getDocumentParams = buildGetDocumentParams(
|
|
"TrueType_without_cmap.pdf",
|
|
{
|
|
disableFontFace: false,
|
|
ownerDocument,
|
|
}
|
|
);
|
|
|
|
const loadingTask = getDocument(getDocumentParams);
|
|
const doc = await loadingTask.promise;
|
|
const page = await doc.getPage(1);
|
|
|
|
const viewport = page.getViewport({ scale: 1 });
|
|
const canvasAndCtx = CanvasFactory.create(viewport.width, viewport.height);
|
|
|
|
await page.render({
|
|
canvasContext: canvasAndCtx.context,
|
|
viewport,
|
|
}).promise;
|
|
|
|
const style = elements.find(element => element.tagName === "style");
|
|
expect(style.sheet.cssRules.length).toBeGreaterThanOrEqual(1);
|
|
expect(style.sheet.cssRules.find(checkFontFaceRule)).toBeTruthy();
|
|
await doc.destroy();
|
|
await loadingTask.destroy();
|
|
CanvasFactory.destroy(canvasAndCtx);
|
|
expect(style.remove.called).toBe(true);
|
|
});
|
|
});
|