123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427 |
- /* This Source Code Form is subject to the terms of the Mozilla Public
- * License, v. 2.0. If a copy of the MPL was not distributed with this
- * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
- 'use strict';
- module.metadata = {
- 'stability': 'unstable'
- };
- const { Cc, Ci } = require('chrome');
- const array = require('../util/array');
- const { defer } = require('sdk/core/promise');
- const windowWatcher = Cc['@mozilla.org/embedcomp/window-watcher;1'].
- getService(Ci.nsIWindowWatcher);
- const appShellService = Cc['@mozilla.org/appshell/appShellService;1'].
- getService(Ci.nsIAppShellService);
- const WM = Cc['@mozilla.org/appshell/window-mediator;1'].
- getService(Ci.nsIWindowMediator);
- const io = Cc['@mozilla.org/network/io-service;1'].
- getService(Ci.nsIIOService);
- const XUL_NS = 'http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul';
- const BROWSER = 'navigator:browser',
- URI_BROWSER = 'chrome://browser/content/browser.xul',
- NAME = '_blank',
- FEATURES = 'chrome,all,dialog=no,non-private';
- function isWindowPrivate(win) {
- if (!win)
- return false;
- // if the pbService is undefined, the PrivateBrowsingUtils.jsm is available,
- // and the app is Firefox, then assume per-window private browsing is
- // enabled.
- try {
- return win.QueryInterface(Ci.nsIInterfaceRequestor)
- .getInterface(Ci.nsIWebNavigation)
- .QueryInterface(Ci.nsILoadContext)
- .usePrivateBrowsing;
- }
- catch(e) {}
- // Sometimes the input is not a nsIDOMWindow.. but it is still a winodw.
- try {
- return !!win.docShell.QueryInterface(Ci.nsILoadContext).usePrivateBrowsing;
- }
- catch (e) {}
- return false;
- }
- exports.isWindowPrivate = isWindowPrivate;
- function getMostRecentBrowserWindow() {
- return getMostRecentWindow(BROWSER);
- }
- exports.getMostRecentBrowserWindow = getMostRecentBrowserWindow;
- function getHiddenWindow() {
- return appShellService.hiddenDOMWindow;
- }
- exports.getHiddenWindow = getHiddenWindow;
- function getMostRecentWindow(type) {
- return WM.getMostRecentWindow(type);
- }
- exports.getMostRecentWindow = getMostRecentWindow;
- /**
- * Returns the ID of the window's current inner window.
- */
- function getInnerId(window) {
- return window.QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIDOMWindowUtils).currentInnerWindowID;
- };
- exports.getInnerId = getInnerId;
- /**
- * Returns the ID of the window's outer window.
- */
- function getOuterId(window) {
- return window.QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIDOMWindowUtils).outerWindowID;
- };
- exports.getOuterId = getOuterId;
- /**
- * Returns window by the outer window id.
- */
- const getByOuterId = WM.getOuterWindowWithId;
- exports.getByOuterId = getByOuterId;
- const getByInnerId = WM.getCurrentInnerWindowWithId;
- exports.getByInnerId = getByInnerId;
- /**
- * Returns `nsIXULWindow` for the given `nsIDOMWindow`.
- */
- function getXULWindow(window) {
- return window.QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIWebNavigation).
- QueryInterface(Ci.nsIDocShellTreeItem).
- treeOwner.QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIXULWindow);
- };
- exports.getXULWindow = getXULWindow;
- function getDOMWindow(xulWindow) {
- return xulWindow.QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIDOMWindow);
- }
- exports.getDOMWindow = getDOMWindow;
- /**
- * Returns `nsIBaseWindow` for the given `nsIDOMWindow`.
- */
- function getBaseWindow(window) {
- return window.QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIWebNavigation).
- QueryInterface(Ci.nsIDocShell).
- QueryInterface(Ci.nsIDocShellTreeItem).
- treeOwner.
- QueryInterface(Ci.nsIBaseWindow);
- }
- exports.getBaseWindow = getBaseWindow;
- /**
- * Returns the `nsIDOMWindow` toplevel window for any child/inner window
- */
- function getToplevelWindow(window) {
- return window.QueryInterface(Ci.nsIInterfaceRequestor)
- .getInterface(Ci.nsIWebNavigation)
- .QueryInterface(Ci.nsIDocShellTreeItem)
- .rootTreeItem
- .QueryInterface(Ci.nsIInterfaceRequestor)
- .getInterface(Ci.nsIDOMWindow);
- }
- exports.getToplevelWindow = getToplevelWindow;
- function getWindowDocShell(window) window.gBrowser.docShell;
- exports.getWindowDocShell = getWindowDocShell;
- function getWindowLoadingContext(window) {
- return getWindowDocShell(window).
- QueryInterface(Ci.nsILoadContext);
- }
- exports.getWindowLoadingContext = getWindowLoadingContext;
- const isTopLevel = window => window && getToplevelWindow(window) === window;
- exports.isTopLevel = isTopLevel;
- /**
- * Takes hash of options and serializes it to a features string that
- * can be used passed to `window.open`. For more details on features string see:
- * https://developer.mozilla.org/en/DOM/window.open#Position_and_size_features
- */
- function serializeFeatures(options) {
- return Object.keys(options).reduce(function(result, name) {
- let value = options[name];
- // the chrome and private features are special
- if ((name == 'private' || name == 'chrome'))
- return result + ((value === true) ? ',' + name : '');
- return result + ',' + name + '=' +
- (value === true ? 'yes' : value === false ? 'no' : value);
- }, '').substr(1);
- }
- /**
- * Opens a top level window and returns it's `nsIDOMWindow` representation.
- * @params {String} uri
- * URI of the document to be loaded into window.
- * @params {nsIDOMWindow} options.parent
- * Used as parent for the created window.
- * @params {String} options.name
- * Optional name that is assigned to the window.
- * @params {Object} options.features
- * Map of key, values like: `{ width: 10, height: 15, chrome: true, private: true }`.
- */
- function open(uri, options) {
- uri = uri || URI_BROWSER;
- options = options || {};
- if (['chrome', 'resource', 'data'].indexOf(io.newURI(uri, null, null).scheme) < 0)
- throw new Error('only chrome, resource and data uris are allowed');
- let newWindow = windowWatcher.
- openWindow(options.parent || null,
- uri,
- options.name || null,
- options.features ? serializeFeatures(options.features) : null,
- options.args || null);
- return newWindow;
- }
- exports.open = open;
- function onFocus(window) {
- let { resolve, promise } = defer();
- if (isFocused(window)) {
- resolve(window);
- }
- else {
- window.addEventListener("focus", function focusListener() {
- window.removeEventListener("focus", focusListener, true);
- resolve(window);
- }, true);
- }
- return promise;
- }
- exports.onFocus = onFocus;
- function isFocused(window) {
- const FM = Cc["@mozilla.org/focus-manager;1"].
- getService(Ci.nsIFocusManager);
- let childTargetWindow = {};
- FM.getFocusedElementForWindow(window, true, childTargetWindow);
- childTargetWindow = childTargetWindow.value;
- let focusedChildWindow = {};
- if (FM.activeWindow) {
- FM.getFocusedElementForWindow(FM.activeWindow, true, focusedChildWindow);
- focusedChildWindow = focusedChildWindow.value;
- }
- return (focusedChildWindow === childTargetWindow);
- }
- exports.isFocused = isFocused;
- /**
- * Opens a top level window and returns it's `nsIDOMWindow` representation.
- * Same as `open` but with more features
- * @param {Object} options
- *
- */
- function openDialog(options) {
- options = options || {};
- let features = options.features || FEATURES;
- let featureAry = features.toLowerCase().split(',');
- if (!!options.private) {
- // add private flag if private window is desired
- if (!array.has(featureAry, 'private')) {
- featureAry.push('private');
- }
- // remove the non-private flag ig a private window is desired
- let nonPrivateIndex = featureAry.indexOf('non-private');
- if (nonPrivateIndex >= 0) {
- featureAry.splice(nonPrivateIndex, 1);
- }
- features = featureAry.join(',');
- }
- let browser = getMostRecentBrowserWindow();
- // if there is no browser then do nothing
- if (!browser)
- return undefined;
- let newWindow = browser.openDialog.apply(
- browser,
- array.flatten([
- options.url || URI_BROWSER,
- options.name || NAME,
- features,
- options.args || null
- ])
- );
- return newWindow;
- }
- exports.openDialog = openDialog;
- /**
- * Returns an array of all currently opened windows.
- * Note that these windows may still be loading.
- */
- function windows(type, options) {
- options = options || {};
- let list = [];
- let winEnum = WM.getEnumerator(type);
- while (winEnum.hasMoreElements()) {
- let window = winEnum.getNext().QueryInterface(Ci.nsIDOMWindow);
- // Only add non-private windows when pb permission isn't set,
- // unless an option forces the addition of them.
- if (!window.closed && (options.includePrivate || !isWindowPrivate(window))) {
- list.push(window);
- }
- }
- return list;
- }
- exports.windows = windows;
- /**
- * Check if the given window is interactive.
- * i.e. if its "DOMContentLoaded" event has already been fired.
- * @params {nsIDOMWindow} window
- */
- const isInteractive = window =>
- window.document.readyState === "interactive" ||
- isDocumentLoaded(window) ||
- // XUL documents stays '"uninitialized"' until it's `readyState` becomes
- // `"complete"`.
- isXULDocumentWindow(window) && window.document.readyState === "interactive";
- exports.isInteractive = isInteractive;
- const isXULDocumentWindow = ({document}) =>
- document.documentElement &&
- document.documentElement.namespaceURI === XUL_NS;
- /**
- * Check if the given window is completely loaded.
- * i.e. if its "load" event has already been fired and all possible DOM content
- * is done loading (the whole DOM document, images content, ...)
- * @params {nsIDOMWindow} window
- */
- function isDocumentLoaded(window) {
- return window.document.readyState == "complete";
- }
- exports.isDocumentLoaded = isDocumentLoaded;
- function isBrowser(window) {
- try {
- return window.document.documentElement.getAttribute("windowtype") === BROWSER;
- }
- catch (e) {}
- return false;
- };
- exports.isBrowser = isBrowser;
- function getWindowTitle(window) {
- return window && window.document ? window.document.title : null;
- }
- exports.getWindowTitle = getWindowTitle;
- function isXULBrowser(window) {
- return !!(isBrowser(window) && window.XULBrowserWindow);
- }
- exports.isXULBrowser = isXULBrowser;
- /**
- * Returns the most recent focused window
- */
- function getFocusedWindow() {
- let window = WM.getMostRecentWindow(BROWSER);
- return window ? window.document.commandDispatcher.focusedWindow : null;
- }
- exports.getFocusedWindow = getFocusedWindow;
- /**
- * Returns the focused element in the most recent focused window
- */
- function getFocusedElement() {
- let window = WM.getMostRecentWindow(BROWSER);
- return window ? window.document.commandDispatcher.focusedElement : null;
- }
- exports.getFocusedElement = getFocusedElement;
- function getFrames(window) {
- return Array.slice(window.frames).reduce(function(frames, frame) {
- return frames.concat(frame, getFrames(frame));
- }, []);
- }
- exports.getFrames = getFrames;
- function getScreenPixelsPerCSSPixel(window) {
- return window.QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIDOMWindowUtils).screenPixelsPerCSSPixel;
- }
- exports.getScreenPixelsPerCSSPixel = getScreenPixelsPerCSSPixel;
- function getOwnerBrowserWindow(node) {
- /**
- Takes DOM node and returns browser window that contains it.
- **/
- let window = getToplevelWindow(node.ownerDocument.defaultView);
- // If anchored window is browser then it's target browser window.
- return isBrowser(window) ? window : null;
- }
- exports.getOwnerBrowserWindow = getOwnerBrowserWindow;
- function getParentWindow(window) {
- try {
- return window.QueryInterface(Ci.nsIInterfaceRequestor)
- .getInterface(Ci.nsIWebNavigation)
- .QueryInterface(Ci.nsIDocShellTreeItem).parent
- .QueryInterface(Ci.nsIInterfaceRequestor)
- .getInterface(Ci.nsIDOMWindow);
- }
- catch (e) {}
- return null;
- }
- exports.getParentWindow = getParentWindow;
- function getParentFrame(window) {
- try {
- return window.QueryInterface(Ci.nsIInterfaceRequestor)
- .getInterface(Ci.nsIWebNavigation)
- .QueryInterface(Ci.nsIDocShellTreeItem).parent
- .QueryInterface(Ci.nsIInterfaceRequestor)
- .getInterface(Ci.nsIDOMWindow);
- }
- catch (e) {}
- return null;
- }
- exports.getParentWindow = getParentWindow;
- // The element in which the window is embedded, or `null`
- // if the window is top-level. Similar to `window.frameElement`
- // but can cross chrome-content boundries.
- const getFrameElement = target =>
- (target instanceof Ci.nsIDOMDocument ? target.defaultView : target).
- QueryInterface(Ci.nsIInterfaceRequestor).
- getInterface(Ci.nsIDOMWindowUtils).
- containerElement;
- exports.getFrameElement = getFrameElement;
|