Source code
Revision control
Copy as Markdown
Other Tools
/* Any copyright is dedicated to the Public Domain.
// eslint-disable-next-line no-unused-vars
let ok;
let is;
// eslint-disable-next-line no-unused-vars
let isnot;
let ContentTaskUtils;
/** @type {{ document: Document, window: Window }} */
let content;
/**
* Inject the global variables from the test scope into the ES module scope.
*/
export function setup(config) {
// When a function is provided to `ContentTask.spawn`, that function is provided the
// Assert library through variable capture. In this case, this code is an ESM module,
// and does not have access to that scope. To work around this issue, pass any any
// relevant variables to that can be bound to the module scope.
//
const { Assert } = config;
ok = Assert.ok.bind(Assert);
is = Assert.equal.bind(Assert);
isnot = Assert.notEqual.bind(Assert);
ContentTaskUtils = config.ContentTaskUtils;
content = config.content;
}
export function getSelectors() {
return {
getH1() {
return content.document.querySelector("h1");
},
getH1Title() {
return content.document.querySelector("h1")?.getAttribute("title");
},
getH1AriaLabel() {
return content.document.querySelector("h1")?.getAttribute("aria-label");
},
getPdfSpan() {
return waitForCondition(
() =>
!!content.document.querySelector(
`.page[data-page-number='1'] .textLayer .endOfContent`
),
"The text layer must be displayed"
).then(() =>
content.document.querySelector(
".page[data-page-number='1'] .textLayer span"
)
);
},
getHeader() {
return content.document.querySelector("header");
},
getFinalParagraph() {
return content.document.querySelector("p:last-of-type");
},
getFinalParagraphTitle() {
return content.document
.querySelector("p:last-of-type")
?.getAttribute("title");
},
getFinalParagraphAriaLabel() {
return content.document
.querySelector("p:last-of-type")
?.getAttribute("aria-label");
},
getFrenchSection() {
return content.document.getElementById("french-section");
},
getEnglishSection() {
return content.document.getElementById("english-section");
},
getSpanishSection() {
return content.document.getElementById("spanish-section");
},
getFrenchSentence() {
return content.document.getElementById("french-sentence");
},
getEnglishSentence() {
return content.document.getElementById("english-sentence");
},
getSpanishSentence() {
return content.document.getElementById("spanish-sentence");
},
getEnglishHyperlink() {
return content.document.getElementById("english-hyperlink");
},
getFrenchHyperlink() {
return content.document.getElementById("french-hyperlink");
},
getSpanishHyperlink() {
return content.document.getElementById("spanish-hyperlink");
},
getURLHyperlink() {
return content.document.getElementById("url-hyperlink");
},
getTextCleaningWhitespace() {
return content.document.getElementById("clean-whitespace");
},
getTextCleaningSoftHyphens() {
return content.document.getElementById("clean-soft-hyphens");
},
};
}
/**
* Provide longer defaults for the waitForCondition.
*
* @param {Function} callback
* @param {string} message
*/
export function waitForCondition(callback, message) {
const interval = 100;
// Use 4 times the defaults to guard against intermittents. Many of the tests rely on
// communication between the parent and child process, which is inherently async.
const maxTries = 50 * 4;
return ContentTaskUtils.waitForCondition(
callback,
message,
interval,
maxTries
);
}
/**
* Asserts that a page was translated with a specific result.
*
* @param {string} message The assertion message.
* @param {Function} getNodeOrText A function to get the node or plain text.
* @param {string | Array<string>} oneOrMoreTranslations The translated message.
*/
export async function assertTranslationResult(
message,
getNodeOrText,
oneOrMoreTranslations
) {
const getText = () => {
const nodeOrText = getNodeOrText();
if (typeof nodeOrText === "string") {
return nodeOrText;
}
return nodeOrText?.innerText;
};
let translation;
try {
if (typeof oneOrMoreTranslations === "string") {
await waitForCondition(
() => oneOrMoreTranslations === getText(),
`Waiting for: "${oneOrMoreTranslations}"`
);
translation = oneOrMoreTranslations;
} else {
await waitForCondition(
() =>
oneOrMoreTranslations.find(translation => translation === getText()),
`Waiting for: "${oneOrMoreTranslations}"`
);
translation = oneOrMoreTranslations.find(
translation => translation === getText()
);
}
} catch (error) {
// The result wasn't found, but the assertion below will report the error.
console.error(error);
}
is(translation, getText(), message);
}
/**
* Simulates right-clicking an element with the mouse.
*
* @param {element} element - The element to right-click.
*/
export function rightClickContentElement(element) {
return new Promise(resolve => {
element.addEventListener(
"contextmenu",
function () {
resolve();
},
{ once: true }
);
const EventUtils = ContentTaskUtils.getEventUtils(content);
EventUtils.sendMouseEvent({ type: "contextmenu" }, element, content.window);
});
}
/**
* Selects all the content within a specified element.
*
* @param {Element} element - The element containing the content to be selected.
* @returns {string} - The text content of the selection.
*/
export function selectContentElement(element) {
content.focus();
content.getSelection().selectAllChildren(element);
return element.textContent;
}