/* Any copyright is dedicated to the Public Domain. * http://creativecommons.org/publicdomain/zero/1.0/ */ ChromeUtils.import("resource://testing-common/CustomizableUITestUtils.jsm", this); let gCUITestUtils = new CustomizableUITestUtils(window); /** * Recursively compare two objects and check that every property of expectedObj has the same value * on actualObj. */ function isSubObjectOf(expectedObj, actualObj, name) { for (let prop in expectedObj) { if (typeof expectedObj[prop] == "function") continue; if (expectedObj[prop] instanceof Object) { is(actualObj[prop].length, expectedObj[prop].length, name + "[" + prop + "]"); isSubObjectOf(expectedObj[prop], actualObj[prop], name + "[" + prop + "]"); } else { is(actualObj[prop], expectedObj[prop], name + "[" + prop + "]"); } } } function getLocale() { return Services.locale.getRequestedLocale() || undefined; } function promiseEvent(aTarget, aEventName, aPreventDefault) { function cancelEvent(event) { if (aPreventDefault) { event.preventDefault(); } return true; } return BrowserTestUtils.waitForEvent(aTarget, aEventName, false, cancelEvent); } /** * Adds a new search engine to the search service and confirms it completes. * * @param {String} basename The file to load that contains the search engine * details. * @param {Object} [options] Options for the test: * - {String} [iconURL] The icon to use for the search engine. * - {Boolean} [setAsCurrent] Whether to set the new engine to be the * current engine or not. * - {String} [testPath] Used to override the current test path if this * file is used from a different directory. * @returns {Promise} The promise is resolved once the engine is added, or * rejected if the addition failed. */ function promiseNewEngine(basename, options = {}) { return new Promise((resolve, reject) => { // Default the setAsCurrent option to true. let setAsCurrent = options.setAsCurrent == undefined ? true : options.setAsCurrent; info("Waiting for engine to be added: " + basename); Services.search.init({ onInitComplete() { let url = getRootDirectory(options.testPath || gTestPath) + basename; let current = Services.search.currentEngine; Services.search.addEngine(url, null, options.iconURL || "", false, { onSuccess(engine) { info("Search engine added: " + basename); if (setAsCurrent) { Services.search.currentEngine = engine; } registerCleanupFunction(() => { if (setAsCurrent) { Services.search.currentEngine = current; } Services.search.removeEngine(engine); info("Search engine removed: " + basename); }); resolve(engine); }, onError(errCode) { ok(false, "addEngine failed with error code " + errCode); reject(); } }); } }); }); } let promiseStateChangeFrameScript = "data:," + encodeURIComponent(`(${ () => { ChromeUtils.import("resource://gre/modules/XPCOMUtils.jsm"); /* globals docShell, sendAsyncMessage */ const global = this; const LISTENER = Symbol("listener"); let listener = { QueryInterface: ChromeUtils.generateQI(["nsISupportsWeakReference", "nsIWebProgressListener"]), onStateChange: function onStateChange(webProgress, req, flags, status) { // Only care about top-level document starts if (!webProgress.isTopLevel || !(flags & Ci.nsIWebProgressListener.STATE_START)) { return; } req.QueryInterface(Ci.nsIChannel); let spec = req.originalURI.spec; if (spec == "about:blank") return; delete global[LISTENER]; docShell.removeProgressListener(listener); req.cancel(Cr.NS_ERROR_FAILURE); sendAsyncMessage("PromiseStateChange::StateChanged", spec); }, }; // Make sure the weak reference stays alive. global[LISTENER] = listener; docShell.QueryInterface(Ci.nsIWebProgress); docShell.addProgressListener(listener, Ci.nsIWebProgress.NOTIFY_STATE_DOCUMENT); } })()`); function promiseStateChangeURI() { const MSG = "PromiseStateChange::StateChanged"; return new Promise(resolve => { let mm = window.getGroupMessageManager("browsers"); mm.loadFrameScript(promiseStateChangeFrameScript, true); let listener = msg => { mm.removeMessageListener(MSG, listener); mm.removeDelayedFrameScript(promiseStateChangeFrameScript); resolve(msg.data); }; mm.addMessageListener(MSG, listener); }); } /** * Waits for a load (or custom) event to finish in a given tab. If provided * load an uri into the tab. * * @param tab * The tab to load into. * @param [optional] url * The url to load, or the current url. * @return {Promise} resolved when the event is handled. * @resolves to the received event * @rejects if a valid load event is not received within a meaningful interval */ function promiseTabLoadEvent(tab, url) { info("Wait tab event: load"); function handle(loadedUrl) { if (loadedUrl === "about:blank" || (url && loadedUrl !== url)) { info(`Skipping spurious load event for ${loadedUrl}`); return false; } info("Tab event received: load"); return true; } let loaded = BrowserTestUtils.browserLoaded(tab.linkedBrowser, false, handle); if (url) BrowserTestUtils.loadURI(tab.linkedBrowser, url); return loaded; } // Get an array of the one-off buttons. function getOneOffs() { let oneOffs = []; let searchPopup = document.getElementById("PopupSearchAutoComplete"); let oneOffsContainer = document.getAnonymousElementByAttribute(searchPopup, "anonid", "search-one-off-buttons"); let oneOff = document.getAnonymousElementByAttribute(oneOffsContainer, "anonid", "search-panel-one-offs"); for (oneOff = oneOff.firstChild; oneOff; oneOff = oneOff.nextSibling) { if (oneOff.nodeType == Node.ELEMENT_NODE) { if (oneOff.classList.contains("dummy") || oneOff.classList.contains("search-setting-button-compact")) break; oneOffs.push(oneOff); } } return oneOffs; }