/* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (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.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is MozMill Test code. * * The Initial Developer of the Original Code is the Mozilla Foundation. * Portions created by the Initial Developer are Copyright (C) 2009 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Henrik Skupin * Anthony Hughes * M.-A. Darche * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ /** * @fileoverview * The UtilsAPI offers various helper functions for any other API which is * not already covered by another shared module. * * @version 1.0.3 */ var MODULE_NAME = 'UtilsAPI'; const gTimeout = 5000; /** * Get application specific informations * @see http://mxr.mozilla.org/mozilla-central/source/xpcom/system/nsIXULAppInfo.idl */ var appInfo = { _service: null, /** * Get the application info service * @returns XUL runtime object * @type nsiXULRuntime */ get appInfo() { if (!this._appInfo) { this._service = Cc["@mozilla.org/xre/app-info;1"] .getService(Ci.nsIXULAppInfo) .QueryInterface(Ci.nsIXULRuntime); } return this._service; }, /** * Get the build id * @returns Build id * @type string */ get buildID() this.appInfo.appBuildID, /** * Get the application id * @returns Application id * @type string */ get ID() this.appInfo.ID, /** * Get the application name * @returns Application name * @type string */ get name() this.appInfo.name, /** * Get the operation system * @returns Operation system name * @type string */ get os() this.appInfo.OS, /** * Get the product vendor * @returns Vendor name * @type string */ get vendor() this.appInfo.vendor, /** * Get the application version * @returns Application version * @type string */ get version() this.appInfo.version, /** * Get the build id of the Gecko platform * @returns Platform build id * @type string */ get platformBuildID() this.appInfo.platformBuildID, /** * Get the version of the Gecko platform * @returns Platform version * @type string */ get platformVersion() this.appInfo.platformVersion, /** * Get the currently used locale * @returns Current locale * @type string */ get locale() { var registry = Cc["@mozilla.org/chrome/chrome-registry;1"] .getService(Ci.nsIXULChromeRegistry); return registry.getSelectedLocale("global"); }, /** * Get the user agent string * @returns User agent * @type string */ get userAgent() { var window = mozmill.wm.getMostRecentWindow("navigator:browser"); if (window) return window.navigator.userAgent; return ""; } }; /** * Checks the visibility of an element. * XXX: Mozmill doesn't check if an element is visible and also operates on * elements which are invisible. (Bug 490548) * * @param {MozmillController} controller * MozMillController of the window to operate on * @param {ElemBase} elem * Element to check its visibility * @param {boolean} expectedVisibility * Expected visibility state of the element */ function assertElementVisible(controller, elem, expectedVisibility) { var element = elem.getNode(); var visible; switch (element.nodeName) { case 'panel': visible = (element.state == 'open'); break; default: var style = controller.window.getComputedStyle(element, ''); var state = style.getPropertyValue('visibility'); visible = (state == 'visible'); } controller.assertJS('subject.visible == subject.expectedVisibility', { visible: visible, expectedVisibility: expectedVisibility }); } /** * Assert if the current URL is identical to the target URL. * With this function also redirects can be tested. * * @param {MozmillController} controller * MozMillController of the window to operate on * @param {string} targetURL * URL to check */ function assertLoadedUrlEqual(controller, targetUrl) { var locationBar = new elementslib.ID(controller.window.document, "urlbar"); var currentURL = locationBar.getNode().value; // Load the target URL controller.open(targetUrl); controller.waitForPageLoad(); // Check the same web page has been opened controller.waitForEval("subject.targetURL.value == subject.currentURL", gTimeout, 100, {targetURL: locationBar.getNode(), currentURL: currentURL}); } /** * Close the context menu inside the content area of the currently open tab * * @param {MozmillController} controller * MozMillController of the window to operate on */ function closeContentAreaContextMenu(controller) { var contextMenu = new elementslib.ID(controller.window.document, "contentAreaContextMenu"); controller.keypress(contextMenu, "VK_ESCAPE", {}); } /** * Run tests against a given search form * * @param {MozMillController} controller * MozMillController of the window to operate on * @param {ElemBase} searchField * The HTML input form element to test * @param {string} searchTerm * The search term for the test * @param {ElemBase} submitButton * (Optional) The forms submit button * @param {number} timeout * The timeout value for the single tests */ function checkSearchField(controller, searchField, searchTerm, submitButton, timeout) { controller.waitThenClick(searchField, timeout); controller.type(searchField, searchTerm); if (submitButton != undefined) { controller.waitThenClick(submitButton, timeout); } } /** * Create a new URI * * @param {string} spec * The URI string in UTF-8 encoding. * @param {string} originCharset * The charset of the document from which this URI string originated. * @param {string} baseURI * If null, spec must specify an absolute URI. Otherwise, spec may be * resolved relative to baseURI, depending on the protocol. * @return A URI object * @type nsIURI */ function createURI(spec, originCharset, baseURI) { let iosvc = Cc["@mozilla.org/network/io-service;1"]. getService(Ci.nsIIOService); return iosvc.newURI(spec, originCharset, baseURI); } /** * Empty the clipboard by assigning an empty string */ function emptyClipboard() { var clipboard = Cc["@mozilla.org/widget/clipboardhelper;1"]. getService(Ci.nsIClipboardHelper); clipboard.copyString(""); } /** * Format a URL by replacing all placeholders * * @param {string} prefName * The preference name which contains the URL * @return The formatted URL * @type string */ function formatUrlPref(prefName) { var formatter = Cc["@mozilla.org/toolkit/URLFormatterService;1"] .getService(Ci.nsIURLFormatter); return formatter.formatURLPref(prefName); } /** * Returns the default home page * * @return The URL of the default homepage * @type string */ function getDefaultHomepage() { var prefs = collector.getModule('PrefsAPI').preferences; var prefValue = prefs.getPref("browser.startup.homepage", "", true, Ci.nsIPrefLocalizedString); return prefValue.data; } /** * Returns the value of an individual entity in a DTD file. * * @param [string] urls * Array of DTD urls. * @param {string} entityId * The ID of the entity to get the value of. * * @return The value of the requested entity * @type string */ function getEntity(urls, entityId) { // Add htmlmathml-f.ent to prevent missing entity errors with XHTML files urls.push("resource:///res/dtd/htmlmathml-f.ent"); // Build a string of external entities var extEntities = ""; for (i = 0; i < urls.length; i++) { extEntities += '%dtd' + i + ';'; } var parser = Cc["@mozilla.org/xmlextras/domparser;1"] .createInstance(Ci.nsIDOMParser); var header = ''; var elem = '&' + entityId + ';'; var doc = parser.parseFromString(header + elem, 'text/xml'); var elemNode = doc.querySelector('elem[id="elementID"]'); if (elemNode == null) throw new Error(arguments.callee.name + ": Unknown entity - " + entityId); return elemNode.textContent; } /** * Returns the value of an individual property. * * @param {string} url * URL of the string bundle. * @param {string} prefName * The property to get the value of. * * @return The value of the requested property * @type string */ function getProperty(url, prefName) { var sbs = Cc["@mozilla.org/intl/stringbundle;1"] .getService(Ci.nsIStringBundleService); var bundle = sbs.createBundle(url); try { return bundle.GetStringFromName(prefName); } catch (ex) { throw new Error(arguments.callee.name + ": Unknown property - " + prefName); } } /** * Function to handle non-modal windows * * @param {string} type * Specifies how to check for the new window (possible values: type or title) * @param {string} text * The window type of title string to search for * @param {function} callback (optional) * Callback function to call for window specific tests * @param {boolean} dontClose (optional) * Doens't close the window after the return from the callback handler * @returns The MozMillController of the window (if the window hasn't been closed) */ function handleWindow(type, text, callback, dontClose) { var func_ptr = null; var window = null; if (dontClose === undefined) dontClose = false; // Set the window opener function to use depending on the type switch (type) { case "type": func_ptr = mozmill.utils.getWindowByType; break; case "title": func_ptr = mozmill.utils.getWindowByTitle; break; default: throw new Error(arguments.callee.name + ": Unknown opener type - " + type); } try { // Wait until the window has been opened mozmill.controller.waitForEval("subject.getWindow(subject.text) != null", gTimeout, 100, {getWindow: func_ptr, text: text}); window = func_ptr(text); // XXX: We still have to find a reliable way to wait until the new window // content has been finished loading. Let's wait for now. var ctrl = new mozmill.controller.MozMillController(window); ctrl.sleep(200); if (callback) callback(ctrl); } finally { // If a failure happened make sure we close the window if wanted if (dontClose != true & window != null) { window.close(); mozmill.controller.waitForEval("subject.getWindow(subject.text) != subject.window", gTimeout, 100, {getWindow: func_ptr, text: text, window: window}); return null; } return ctrl; } }