summaryrefslogtreecommitdiffstats
path: root/browser/components/preferences/tests/browser_subdialogs.js
diff options
context:
space:
mode:
Diffstat (limited to 'browser/components/preferences/tests/browser_subdialogs.js')
-rw-r--r--browser/components/preferences/tests/browser_subdialogs.js639
1 files changed, 639 insertions, 0 deletions
diff --git a/browser/components/preferences/tests/browser_subdialogs.js b/browser/components/preferences/tests/browser_subdialogs.js
new file mode 100644
index 0000000000..9342575886
--- /dev/null
+++ b/browser/components/preferences/tests/browser_subdialogs.js
@@ -0,0 +1,639 @@
+/* Any copyright is dedicated to the Public Domain.
+ * http://creativecommons.org/publicdomain/zero/1.0/ */
+
+"use strict";
+
+/**
+ * Tests for the sub-dialog infrastructure, not for actual sub-dialog functionality.
+ */
+
+const gDialogURL = getRootDirectory(gTestPath) + "subdialog.xhtml";
+const gDialogURL2 = getRootDirectory(gTestPath) + "subdialog2.xhtml";
+
+function open_subdialog_and_test_generic_start_state(
+ browser,
+ domcontentloadedFn,
+ url = gDialogURL
+) {
+ let domcontentloadedFnStr = domcontentloadedFn
+ ? "(" + domcontentloadedFn.toString() + ")()"
+ : "";
+ return SpecialPowers.spawn(
+ browser,
+ [{ url, domcontentloadedFnStr }],
+ async function (args) {
+ let rv = { acceptCount: 0 };
+ let win = content.window;
+ content.gSubDialog.open(args.url, undefined, rv);
+ let subdialog = content.gSubDialog._topDialog;
+
+ info("waiting for subdialog DOMFrameContentLoaded");
+ let dialogOpenPromise;
+ await new Promise(resolve => {
+ win.addEventListener(
+ "DOMFrameContentLoaded",
+ function frameContentLoaded(ev) {
+ // We can get events for loads in other frames, so we have to filter
+ // those out.
+ if (ev.target != subdialog._frame) {
+ return;
+ }
+ win.removeEventListener(
+ "DOMFrameContentLoaded",
+ frameContentLoaded
+ );
+ dialogOpenPromise = ContentTaskUtils.waitForEvent(
+ subdialog._overlay,
+ "dialogopen"
+ );
+ resolve();
+ },
+ { capture: true }
+ );
+ });
+ let result;
+ if (args.domcontentloadedFnStr) {
+ // eslint-disable-next-line no-eval
+ result = eval(args.domcontentloadedFnStr);
+ }
+
+ info("waiting for subdialog load");
+ await dialogOpenPromise;
+ info("subdialog window is loaded");
+
+ let expectedStyleSheetURLs = subdialog._injectedStyleSheets.slice(0);
+ for (let styleSheet of subdialog._frame.contentDocument.styleSheets) {
+ let index = expectedStyleSheetURLs.indexOf(styleSheet.href);
+ if (index >= 0) {
+ expectedStyleSheetURLs.splice(index, 1);
+ }
+ }
+
+ Assert.ok(
+ !!subdialog._frame.contentWindow,
+ "The dialog should be non-null"
+ );
+ Assert.notEqual(
+ subdialog._frame.contentWindow.location.toString(),
+ "about:blank",
+ "Subdialog URL should not be about:blank"
+ );
+ Assert.equal(
+ win.getComputedStyle(subdialog._overlay).visibility,
+ "visible",
+ "Overlay should be visible"
+ );
+ Assert.equal(
+ expectedStyleSheetURLs.length,
+ 0,
+ "No stylesheets that were expected are missing"
+ );
+ return result;
+ }
+ );
+}
+
+async function close_subdialog_and_test_generic_end_state(
+ browser,
+ closingFn,
+ closingButton,
+ acceptCount,
+ options
+) {
+ let getDialogsCount = () => {
+ return SpecialPowers.spawn(
+ browser,
+ [],
+ () => content.window.gSubDialog._dialogs.length
+ );
+ };
+ let getStackChildrenCount = () => {
+ return SpecialPowers.spawn(
+ browser,
+ [],
+ () => content.window.gSubDialog._dialogStack.children.length
+ );
+ };
+ let dialogclosingPromise = SpecialPowers.spawn(
+ browser,
+ [{ closingButton, acceptCount }],
+ async function (expectations) {
+ let win = content.window;
+ let subdialog = win.gSubDialog._topDialog;
+ let frame = subdialog._frame;
+
+ let frameWinUnload = ContentTaskUtils.waitForEvent(
+ frame.contentWindow,
+ "unload",
+ true
+ );
+
+ let actualAcceptCount;
+ info("waiting for dialogclosing");
+ info("URI " + frame.currentURI?.spec);
+ let closingEvent = await ContentTaskUtils.waitForEvent(
+ frame.contentWindow,
+ "dialogclosing",
+ true,
+ () => {
+ actualAcceptCount = frame.contentWindow?.arguments[0].acceptCount;
+ return true;
+ }
+ );
+
+ info("Waiting for subdialog unload");
+ await frameWinUnload;
+
+ let contentClosingButton = closingEvent.detail.button;
+
+ Assert.notEqual(
+ win.getComputedStyle(subdialog._overlay).visibility,
+ "visible",
+ "overlay is not visible"
+ );
+ Assert.equal(
+ frame.getAttribute("style"),
+ "",
+ "inline styles should be cleared"
+ );
+ Assert.equal(
+ contentClosingButton,
+ expectations.closingButton,
+ "closing event should indicate button was '" +
+ expectations.closingButton +
+ "'"
+ );
+ Assert.equal(
+ actualAcceptCount,
+ expectations.acceptCount,
+ "should be 1 if accepted, 0 if canceled, undefined if closed w/out button"
+ );
+ }
+ );
+ let initialDialogsCount = await getDialogsCount();
+ let initialStackChildrenCount = await getStackChildrenCount();
+ if (options && options.runClosingFnOutsideOfContentTask) {
+ await closingFn();
+ } else {
+ SpecialPowers.spawn(browser, [], closingFn);
+ }
+
+ await dialogclosingPromise;
+ let endDialogsCount = await getDialogsCount();
+ let endStackChildrenCount = await getStackChildrenCount();
+ Assert.equal(
+ initialDialogsCount - 1,
+ endDialogsCount,
+ "dialog count should decrease by 1"
+ );
+ Assert.equal(
+ initialStackChildrenCount - 1,
+ endStackChildrenCount,
+ "stack children count should decrease by 1"
+ );
+}
+
+let tab;
+
+add_task(async function test_initialize() {
+ tab = await BrowserTestUtils.openNewForegroundTab(
+ gBrowser,
+ "about:preferences"
+ );
+});
+
+add_task(
+ async function check_titlebar_focus_returnval_titlechanges_accepting() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+
+ let domtitlechangedPromise = BrowserTestUtils.waitForEvent(
+ tab.linkedBrowser,
+ "DOMTitleChanged"
+ );
+ await SpecialPowers.spawn(tab.linkedBrowser, [], async function () {
+ let dialog = content.window.gSubDialog._topDialog;
+ let dialogWin = dialog._frame.contentWindow;
+ let dialogTitleElement = dialog._titleElement;
+ Assert.equal(
+ dialogTitleElement.textContent,
+ "Sample sub-dialog",
+ "Title should be correct initially"
+ );
+ Assert.equal(
+ dialogWin.document.activeElement.value,
+ "Default text",
+ "Textbox with correct text is focused"
+ );
+ dialogWin.document.title = "Updated title";
+ });
+
+ info("waiting for DOMTitleChanged event");
+ await domtitlechangedPromise;
+
+ SpecialPowers.spawn(tab.linkedBrowser, [], async function () {
+ let dialogTitleElement =
+ content.window.gSubDialog._topDialog._titleElement;
+ Assert.equal(
+ dialogTitleElement.textContent,
+ "Updated title",
+ "subdialog should have updated title"
+ );
+ });
+
+ // Accept the dialog
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentDocument
+ .getElementById("subDialog")
+ .acceptDialog();
+ },
+ "accept",
+ 1
+ );
+ }
+);
+
+add_task(async function check_canceling_dialog() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+
+ info("canceling the dialog");
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentDocument
+ .getElementById("subDialog")
+ .cancelDialog();
+ },
+ "cancel",
+ 0
+ );
+});
+
+add_task(async function check_reopening_dialog() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+ info("opening another dialog which will close the first");
+ await open_subdialog_and_test_generic_start_state(
+ tab.linkedBrowser,
+ "",
+ gDialogURL2
+ );
+
+ SpecialPowers.spawn(tab.linkedBrowser, [], async function () {
+ let win = content.window;
+ let dialogs = win.gSubDialog._dialogs;
+ let lowerDialog = dialogs[0];
+ let topDialog = dialogs[1];
+ Assert.equal(dialogs.length, 2, "There should be two visible dialogs");
+ Assert.equal(
+ win.getComputedStyle(topDialog._overlay).visibility,
+ "visible",
+ "The top dialog should be visible"
+ );
+ Assert.equal(
+ win.getComputedStyle(lowerDialog._overlay).visibility,
+ "visible",
+ "The lower dialog should be visible"
+ );
+ Assert.equal(
+ win.getComputedStyle(topDialog._overlay).backgroundColor,
+ "rgba(0, 0, 0, 0.5)",
+ "The top dialog should have a semi-transparent overlay"
+ );
+ Assert.equal(
+ win.getComputedStyle(lowerDialog._overlay).backgroundColor,
+ "rgba(0, 0, 0, 0)",
+ "The lower dialog should not have an overlay"
+ );
+ });
+
+ info("closing two dialogs");
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentDocument
+ .getElementById("subDialog")
+ .acceptDialog();
+ },
+ "accept",
+ 1
+ );
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentDocument
+ .getElementById("subDialog")
+ .acceptDialog();
+ },
+ "accept",
+ 1
+ );
+});
+
+add_task(async function check_opening_while_closing() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+ info("closing");
+ content.window.gSubDialog._topDialog.close();
+ info("reopening immediately after calling .close()");
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentDocument
+ .getElementById("subDialog")
+ .acceptDialog();
+ },
+ "accept",
+ 1
+ );
+});
+
+add_task(async function window_close_on_dialog() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+
+ info("canceling the dialog");
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentWindow.close();
+ },
+ null,
+ 0
+ );
+});
+
+add_task(async function click_close_button_on_dialog() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+
+ info("canceling the dialog");
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ return BrowserTestUtils.synthesizeMouseAtCenter(
+ ".dialogClose",
+ {},
+ tab.linkedBrowser
+ );
+ },
+ null,
+ 0,
+ { runClosingFnOutsideOfContentTask: true }
+ );
+});
+
+add_task(async function background_click_should_close_dialog() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+
+ // Clicking on an inactive part of dialog itself should not close the dialog.
+ // Click the dialog title bar here to make sure nothing happens.
+ info("clicking the dialog title bar");
+ BrowserTestUtils.synthesizeMouseAtCenter(
+ ".dialogTitle",
+ {},
+ tab.linkedBrowser
+ );
+
+ // Close the dialog by clicking on the overlay background. Simulate a click
+ // at point (2,2) instead of (0,0) so we are sure we're clicking on the
+ // overlay background instead of some boundary condition that a real user
+ // would never click.
+ info("clicking the overlay background");
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ return BrowserTestUtils.synthesizeMouseAtPoint(
+ 2,
+ 2,
+ {},
+ tab.linkedBrowser
+ );
+ },
+ null,
+ 0,
+ { runClosingFnOutsideOfContentTask: true }
+ );
+});
+
+add_task(async function escape_should_close_dialog() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+
+ info("canceling the dialog");
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ return BrowserTestUtils.synthesizeKey("VK_ESCAPE", {}, tab.linkedBrowser);
+ },
+ "cancel",
+ 0,
+ { runClosingFnOutsideOfContentTask: true }
+ );
+});
+
+add_task(async function correct_width_and_height_should_be_used_for_dialog() {
+ await open_subdialog_and_test_generic_start_state(tab.linkedBrowser);
+
+ await SpecialPowers.spawn(tab.linkedBrowser, [], async function () {
+ function fuzzyEqual(value, expectedValue, fuzz, msg) {
+ Assert.greaterOrEqual(expectedValue + fuzz, value, msg);
+ Assert.lessOrEqual(expectedValue - fuzz, value, msg);
+ }
+ let topDialog = content.gSubDialog._topDialog;
+ let frameStyle = content.getComputedStyle(topDialog._frame);
+ let dialogStyle = topDialog.frameContentWindow.getComputedStyle(
+ topDialog.frameContentWindow.document.documentElement
+ );
+ let fontSize = parseFloat(dialogStyle.fontSize);
+ let height = parseFloat(frameStyle.height);
+ let width = parseFloat(frameStyle.width);
+
+ fuzzyEqual(
+ width,
+ fontSize * 32,
+ 2,
+ "Width should be set on the frame from the dialog"
+ );
+ fuzzyEqual(
+ height,
+ fontSize * 5,
+ 2,
+ "Height should be set on the frame from the dialog"
+ );
+ });
+
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentWindow.close();
+ },
+ null,
+ 0
+ );
+});
+
+add_task(
+ async function wrapped_text_in_dialog_should_have_expected_scrollHeight() {
+ let oldHeight = await open_subdialog_and_test_generic_start_state(
+ tab.linkedBrowser,
+ function domcontentloadedFn() {
+ let frame = content.window.gSubDialog._topDialog._frame;
+ let doc = frame.contentDocument;
+ let scrollHeight = doc.documentElement.scrollHeight;
+ doc.documentElement.style.removeProperty("height");
+ doc.getElementById("desc").textContent = `
+ Sed ut perspiciatis unde omnis iste natus error sit voluptatem accusantium doloremque
+ laudantium, totam rem aperiam, eaque ipsa quae ab illo inventore veritatis et quasi
+ architecto beatae vitae dicta sunt explicabo. Nemo enim ipsam voluptatem quia voluptas
+ sit aspernatur aut odit aut fugit, sed quia consequuntur magni dolores eos qui ratione
+ laudantium, totam rem aperiam, eaque ipsa quae ab illo inventore veritatis et quasi
+ architecto beatae vitae dicta sunt explicabo. Nemo enim ipsam voluptatem quia voluptas
+ sit aspernatur aut odit aut fugit, sed quia consequuntur magni dolores eos qui ratione
+ laudantium, totam rem aperiam, eaque ipsa quae ab illo inventore veritatis et quasi
+ architecto beatae vitae dicta sunt explicabo. Nemo enim ipsam voluptatem quia voluptas
+ sit aspernatur aut odit aut fugit, sed quia consequuntur magni dolores eos qui ratione
+ voluptatem sequi nesciunt.`;
+ return scrollHeight;
+ }
+ );
+
+ await SpecialPowers.spawn(
+ tab.linkedBrowser,
+ [oldHeight],
+ async function (contentOldHeight) {
+ function fuzzyEqual(value, expectedValue, fuzz, msg) {
+ Assert.greaterOrEqual(expectedValue + fuzz, value, msg);
+ Assert.lessOrEqual(expectedValue - fuzz, value, msg);
+ }
+ let topDialog = content.gSubDialog._topDialog;
+ let frame = topDialog._frame;
+ let frameStyle = content.getComputedStyle(frame);
+ let docEl = frame.contentDocument.documentElement;
+ let dialogStyle = topDialog.frameContentWindow.getComputedStyle(docEl);
+ let fontSize = parseFloat(dialogStyle.fontSize);
+ let height = parseFloat(frameStyle.height);
+ let width = parseFloat(frameStyle.width);
+
+ fuzzyEqual(
+ width,
+ 32 * fontSize,
+ 2,
+ "Width should be set on the frame from the dialog"
+ );
+ Assert.ok(
+ docEl.scrollHeight > contentOldHeight,
+ "Content height increased (from " +
+ contentOldHeight +
+ " to " +
+ docEl.scrollHeight +
+ ")."
+ );
+ fuzzyEqual(
+ height,
+ docEl.scrollHeight,
+ 2,
+ "Height on the frame should be higher now. " +
+ "This test may fail on certain screen resoluition. " +
+ "See bug 1420576 and bug 1205717."
+ );
+ }
+ );
+
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentWindow.window.close();
+ },
+ null,
+ 0
+ );
+ }
+);
+
+add_task(async function dialog_too_tall_should_get_reduced_in_height() {
+ await open_subdialog_and_test_generic_start_state(
+ tab.linkedBrowser,
+ function domcontentloadedFn() {
+ let frame = content.window.gSubDialog._topDialog._frame;
+ frame.contentDocument.documentElement.style.height = "100000px";
+ }
+ );
+
+ await SpecialPowers.spawn(tab.linkedBrowser, [], async function () {
+ function fuzzyEqual(value, expectedValue, fuzz, msg) {
+ Assert.greaterOrEqual(expectedValue + fuzz, value, msg);
+ Assert.lessOrEqual(expectedValue - fuzz, value, msg);
+ }
+ let topDialog = content.gSubDialog._topDialog;
+ let frame = topDialog._frame;
+ let frameStyle = content.getComputedStyle(frame);
+ let dialogStyle = topDialog.frameContentWindow.getComputedStyle(
+ frame.contentDocument.documentElement
+ );
+ let fontSize = parseFloat(dialogStyle.fontSize);
+ let height = parseFloat(frameStyle.height);
+ let width = parseFloat(frameStyle.width);
+ fuzzyEqual(
+ width,
+ 32 * fontSize,
+ 2,
+ "Width should be set on the frame from the dialog"
+ );
+ Assert.less(
+ height,
+ content.window.innerHeight,
+ "Height on the frame should be smaller than window's innerHeight"
+ );
+ });
+
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentWindow.window.close();
+ },
+ null,
+ 0
+ );
+});
+
+add_task(
+ async function scrollWidth_and_scrollHeight_from_subdialog_should_size_the_browser() {
+ await open_subdialog_and_test_generic_start_state(
+ tab.linkedBrowser,
+ function domcontentloadedFn() {
+ let frame = content.window.gSubDialog._topDialog._frame;
+ frame.contentDocument.documentElement.style.removeProperty("height");
+ frame.contentDocument.documentElement.style.removeProperty("width");
+ }
+ );
+
+ await SpecialPowers.spawn(tab.linkedBrowser, [], async function () {
+ let frame = content.window.gSubDialog._topDialog._frame;
+ Assert.ok(
+ frame.style.width.endsWith("px"),
+ "Width (" +
+ frame.style.width +
+ ") should be set to a px value of the scrollWidth from the dialog"
+ );
+ let cs = content.getComputedStyle(frame);
+ Assert.stringMatches(
+ cs.getPropertyValue("--subdialog-inner-height"),
+ /px$/,
+ "Height (" +
+ cs.getPropertyValue("--subdialog-inner-height") +
+ ") should be set to a px value of the scrollHeight from the dialog"
+ );
+ });
+
+ await close_subdialog_and_test_generic_end_state(
+ tab.linkedBrowser,
+ function () {
+ content.window.gSubDialog._topDialog._frame.contentWindow.window.close();
+ },
+ null,
+ 0
+ );
+ }
+);
+
+add_task(async function test_shutdown() {
+ gBrowser.removeTab(tab);
+});