// -*- Mode: js2; tab-width: 2; indent-tabs-mode: nil; js2-basic-offset: 2; js2-skip-preprocessor-directives: t; -*- /* * ***** 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 Mozilla Mobile Browser. * * The Initial Developer of the Original Code is * Mozilla Corporation. * Portions created by the Initial Developer are Copyright (C) 2008 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Brad Lassey * Mark Finkle * Aleks Totic * Johnathan Nightingale * Stuart Parmenter * Taras Glek * Roy Frostig * Ben Combee * * 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 ***** */ let Cc = Components.classes; let Ci = Components.interfaces; let Cu = Components.utils; const endl = '\n'; Cu.import("resource://gre/modules/SpatialNavigation.js"); Cu.import("resource://gre/modules/PluralForm.jsm"); function getBrowser() { return Browser.selectedBrowser; } const kDefaultBrowserWidth = 800; function debug() { let bv = Browser._browserView; let tc = bv._tileManager._tileCache; let scrollbox = document.getElementById("content-scrollbox") .boxObject.QueryInterface(Ci.nsIScrollBoxObject); let x = {}; let y = {}; let w = {}; let h = {}; scrollbox.getPosition(x, y); scrollbox.getScrolledSize(w, h); let container = document.getElementById("tile-container"); let [x, y] = [x.value, y.value]; let [w, h] = [w.value, h.value]; if (bv) { dump('----------------------DEBUG!-------------------------\n'); dump(bv._browserViewportState.toString() + endl); dump(endl); dump('location from Browser: ' + Browser.selectedBrowser.contentWindow.location + endl); dump('location from BV : ' + bv.getBrowser().contentWindow.location + endl); dump(endl + endl); let cr = bv._tileManager._criticalRect; dump('criticalRect from BV: ' + (cr ? cr.toString() : null) + endl); dump('visibleRect from BV : ' + bv.getVisibleRect().toString() + endl); dump('visibleRect from foo: ' + Browser.getVisibleRect().toString() + endl); dump('bv batchops depth: ' + bv._batchOps.length + endl); dump('renderpause depth: ' + bv._renderMode + endl); dump(endl); dump('window.innerWidth : ' + window.innerWidth + endl); dump('window.innerHeight: ' + window.innerHeight + endl); dump(endl); dump('container width,height from BV: ' + bv._container.style.width + ', ' + bv._container.style.height + endl); dump('container width,height via DOM: ' + container.style.width + ', ' + container.style.height + endl); dump(endl); dump('scrollbox position : ' + x + ', ' + y + endl); dump('scrollbox scrolledsize: ' + w + ', ' + h + endl); let sb = document.getElementById("content-scrollbox"); dump('container location: ' + Math.round(container.getBoundingClientRect().left) + " " + Math.round(container.getBoundingClientRect().top) + endl); dump(endl); let mouseModule = ih._modules[0]; dump('ih grabber : ' + ih._grabber + endl); dump('ih grabdepth: ' + ih._grabDepth + endl); dump('ih listening: ' + !ih._ignoreEvents + endl); dump('ih suppress : ' + ih._suppressNextClick + endl); dump('mouseModule : ' + mouseModule + endl); dump(endl); dump('tilecache capacity: ' + bv._tileManager._tileCache.getCapacity() + endl); dump('tilecache size : ' + bv._tileManager._tileCache.size + endl); dump('tilecache iBound : ' + bv._tileManager._tileCache.iBound + endl); dump('tilecache jBound : ' + bv._tileManager._tileCache.jBound + endl); dump('-----------------------------------------------------\n'); } } function debugTile(i, j) { let bv = Browser._browserView; let tc = bv._tileManager._tileCache; let t = tc.getTile(i, j); dump('------ DEBUGGING TILE (' + i + ',' + j + ') --------\n'); dump('in bounds: ' + tc.inBounds(i, j) + endl); dump('occupied : ' + tc.isOccupied(i, j) + endl); if (t) { dump('toString : ' + t.toString(true) + endl); dump('free : ' + t.free + endl); dump('dirtyRect: ' + t._dirtyTileCanvasRect + endl); let len = tc._tilePool.length; for (let k = 0; k < len; ++k) if (tc._tilePool[k] === t) dump('found in tilePool at index ' + k + endl); } dump('------------------------------------\n'); } function onDebugKeyPress(ev) { let bv = Browser._browserView; if (!ev.ctrlKey) return; // use capitals so we require SHIFT here too const a = 65; // debug all critical tiles const b = 66; // dump an ASCII graphic of the tile map const c = 67; // set tilecache capacity const d = 68; // debug dump const e = 69; const f = 70; // free memory by clearing a tab. const g = 71; const h = 72; const i = 73; // toggle info click mode const j = 74; const k = 75; const l = 76; // restart lazy crawl const m = 77; // fix mouseout const n = 78; const o = 79; const p = 80; // debug tiles in pool order const q = 81; const r = 82; // reset visible rect const s = 83; const t = 84; // debug given list of tiles separated by space const u = 85; const v = 86; const w = 87; const x = 88; const y = 89; const z = 90; // set zoom level to 1 if (window.tileMapMode) { function putChar(ev, col, row) { let tile = tc.getTile(col, row); switch (ev.charCode) { case h: // held tiles dump(tile ? (tile.free ? '*' : 'h') : ' '); break; case d: // dirty tiles dump(tile ? (tile.isDirty() ? 'd' : '*') : ' '); break; case o: // occupied tileholders dump(tc.isOccupied(col, row) ? 'o' : ' '); break; } } let tc = Browser._browserView._tileManager._tileCache; let col, row; dump(endl); dump(' '); for (col = 0; col < tc.iBound; ++col) dump(col % 10); dump(endl); for (row = 0; row < tc.jBound; ++row) { dump((row % 10) + ' '); for (col = 0; col < tc.iBound; ++col) { putChar(ev, col, row); } dump(endl); } dump(endl + endl); for (let ii = 0; ii < tc._tilePool.length; ++ii) { let tile = tc._tilePool[ii]; putChar(ev, tile.i, tile.j); } dump(endl + endl); window.tileMapMode = false; return; } switch (ev.charCode) { case f: var result = Browser.sacrificeTab(); if (result) dump("Freed a tab\n"); else dump("There are no tabs left to free\n"); break; case r: bv.onAfterVisibleMove(); //bv.setVisibleRect(Browser.getVisibleRect()); case d: debug(); break; case l: bv._tileManager.restartLazyCrawl(bv._tileManager._criticalRect); break; case c: let cap = parseInt(window.prompt('new capacity')); bv._tileManager._tileCache.setCapacity(cap); break; case b: window.tileMapMode = true; break; case t: let ijstrs = window.prompt('row,col plz').split(' '); for each (let ijstr in ijstrs) { let [i, j] = ijstr.split(',').map(function (x) { return parseInt(x); }); debugTile(i, j); } break; case a: let cr = bv._tileManager._criticalRect; dump('>>>>>> critical rect is ' + (cr ? cr.toString() : cr) + '\n'); if (cr) { let starti = cr.left >> kTileExponentWidth; let endi = cr.right >> kTileExponentWidth; let startj = cr.top >> kTileExponentHeight; let endj = cr.bottom >> kTileExponentHeight; for (var jj = startj; jj <= endj; ++jj) for (var ii = starti; ii <= endi; ++ii) debugTile(ii, jj); } break; case i: window.infoMode = !window.infoMode; break; case m: Util.dumpLn("renderMode:", bv._renderMode); Util.dumpLn("batchOps:",bv._batchOps.length); bv.resumeRendering(); break; case p: let tc = bv._tileManager._tileCache; dump('************* TILE POOL ****************\n'); for (let ii = 0, len = tc._tilePool.length; ii < len; ++ii) { if (window.infoMode) debugTile(tc._tilePool[ii].i, tc._tilePool[ii].j); else dump(tc._tilePool[ii].i + ',' + tc._tilePool[ii].j + '\n'); } dump('****************************************\n'); break; case z: bv.setZoomLevel(1.0); break; default: break; } } window.infoMode = false; window.tileMapMode = false; var ih = null; var Browser = { _tabs : [], _selectedTab : null, windowUtils: window.QueryInterface(Ci.nsIInterfaceRequestor) .getInterface(Ci.nsIDOMWindowUtils), contentScrollbox: null, contentScrollboxScroller: null, controlsScrollbox: null, controlsScrollboxScroller: null, pageScrollbox: null, pageScrollboxScroller: null, styles: {}, startup: function() { var self = this; let container = document.getElementById("tile-container"); let bv = this._browserView = new BrowserView(container, Browser.getVisibleRect); /* handles dispatching clicks on tiles into clicks in content or zooms */ container.customClicker = new ContentCustomClicker(bv); /* scrolling box that contains tiles */ let contentScrollbox = this.contentScrollbox = document.getElementById("content-scrollbox"); this.contentScrollboxScroller = contentScrollbox.boxObject.QueryInterface(Ci.nsIScrollBoxObject); contentScrollbox.customDragger = new Browser.MainDragger(bv); /* horizontally scrolling box that holds the sidebars as well as the contentScrollbox */ let controlsScrollbox = this.controlsScrollbox = document.getElementById("controls-scrollbox"); this.controlsScrollboxScroller = controlsScrollbox.boxObject.QueryInterface(Ci.nsIScrollBoxObject); controlsScrollbox.customDragger = { isDraggable: function isDraggable(target, content) { return false; }, dragStart: function dragStart(cx, cy, target, scroller) {}, dragStop: function dragStop(dx, dy, scroller) { return false; }, dragMove: function dragMove(dx, dy, scroller) { return false; } }; /* vertically scrolling box that contains the url bar, notifications, and content */ let pageScrollbox = this.pageScrollbox = document.getElementById("page-scrollbox"); this.pageScrollboxScroller = pageScrollbox.boxObject.QueryInterface(Ci.nsIScrollBoxObject); pageScrollbox.customDragger = controlsScrollbox.customDragger; // during startup a lot of viewportHandler calls happen due to content and window resizes bv.beginBatchOperation(); let stylesheet = document.styleSheets[0]; for each (let style in ["window-width", "window-height", "toolbar-height", "browser", "browser-handheld", "browser-viewport"]) { let index = stylesheet.insertRule("." + style + " {}", stylesheet.cssRules.length); this.styles[style] = stylesheet.cssRules[index].style; } function resizeHandler(e) { if (e.target != window) return; // XXX is this code right here actually needed? let w = window.innerWidth; let h = window.innerHeight; let maximize = (document.documentElement.getAttribute("sizemode") == "maximized"); if (maximize && w > screen.width) return; bv.beginBatchOperation(); let toolbarHeight = Math.round(document.getElementById("toolbar-main").getBoundingClientRect().height); let scaledDefaultH = (kDefaultBrowserWidth * (h / w)); let scaledScreenH = (window.screen.width * (h / w)); Browser.styles["window-width"].width = w + "px"; Browser.styles["window-height"].height = h + "px"; Browser.styles["toolbar-height"].height = toolbarHeight + "px"; Browser.styles["browser"].width = kDefaultBrowserWidth + "px"; Browser.styles["browser"].height = scaledDefaultH + "px"; Browser.styles["browser-handheld"].width = window.screen.width + "px"; Browser.styles["browser-handheld"].height = scaledScreenH + "px"; // Tell the UI to resize the browser controls before calling updateSize BrowserUI.sizeControls(w, h); bv.updateDefaultZoom(); if (bv.isDefaultZoom()) // XXX this should really only happen on browser startup, not every resize Browser.hideSidebars(); bv.onAfterVisibleMove(); bv.commitBatchOperation(); } window.addEventListener("resize", resizeHandler, false); function fullscreenHandler() { if (!window.fullScreen) document.getElementById("toolbar-main").setAttribute("fullscreen", "true"); else document.getElementById("toolbar-main").removeAttribute("fullscreen"); } window.addEventListener("fullscreen", fullscreenHandler, false); function notificationHandler() { // Let the view know that the layout might have changed Browser.forceChromeReflow(); bv.onAfterVisibleMove(); } let notifications = document.getElementById("notifications"); notifications.addEventListener("AlertActive", notificationHandler, false); notifications.addEventListener("AlertClose", notificationHandler, false); // initialize input handling ih = new InputHandler(container); BrowserUI.init(); window.controllers.appendController(this); window.controllers.appendController(BrowserUI); var styleSheets = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService); // Should we hide the cursors var hideCursor = gPrefService.getBoolPref("browser.ui.cursor") == false; if (hideCursor) { window.QueryInterface(Ci.nsIDOMChromeWindow).setCursor("none"); var styleURI = gIOService.newURI("chrome://browser/content/cursor.css", null, null); styleSheets.loadAndRegisterSheet(styleURI, styleSheets.AGENT_SHEET); } // load styles for scrollbars var styleURI = gIOService.newURI("chrome://browser/content/content.css", null, null); styleSheets.loadAndRegisterSheet(styleURI, styleSheets.AGENT_SHEET); var os = Cc["@mozilla.org/observer-service;1"].getService(Ci.nsIObserverService); os.addObserver(gXPInstallObserver, "xpinstall-install-blocked", false); os.addObserver(gSessionHistoryObserver, "browser:purge-session-history", false); #ifdef WINCE os.addObserver(SoftKeyboardObserver, "softkb-change", false); #endif // clear out tabs the user hasn't touched lately on memory crunch os.addObserver(MemoryObserver, "memory-pressure", false); // search engine changes os.addObserver(BrowserSearch, "browser-search-engine-modified", false); window.QueryInterface(Ci.nsIDOMChromeWindow).browserDOMWindow = new nsBrowserAccess(); let browsers = document.getElementById("browsers"); browsers.addEventListener("command", this._handleContentCommand, true); browsers.addEventListener("MozApplicationManifest", OfflineApps, false); browsers.addEventListener("DOMUpdatePageReport", gPopupBlockerObserver.onUpdatePageReport, false); // Initialize Spatial Navigation function panCallback(aElement) { if (!aElement) return; // XXX We need to add this back //canvasBrowser.ensureElementIsVisible(aElement); } // Init it with the "browsers" element, which will receive keypress events // for all of our s SpatialNavigation.init(browsers, panCallback); // Login Manager Cc["@mozilla.org/login-manager;1"].getService(Ci.nsILoginManager); // Make sure we're online before attempting to load Util.forceOnline(); // Command line arguments/initial homepage let whereURI = "about:blank"; switch (Util.needHomepageOverride()) { case "new profile": whereURI = "about:firstrun"; break; case "new version": case "none": whereURI = "about:blank"; break; } // If this is an intial window launch (was a nsICommandLine passed via window params) // we execute some logic to load the initial launch page if (window.arguments && window.arguments[0] && window.arguments[0] instanceof Ci.nsICommandLine) { try { var cmdLine = window.arguments[0]; // Check for and use a single commandline parameter if (cmdLine.length == 1) { // Assume the first arg is a URI if it is not a flag var uri = cmdLine.getArgument(0); if (uri != "" && uri[0] != '-') { whereURI = cmdLine.resolveURI(uri); if (whereURI) whereURI = whereURI.spec; } } // Check for the "url" flag var uriFlag = cmdLine.handleFlagWithParam("url", false); if (uriFlag) { whereURI = cmdLine.resolveURI(uriFlag); if (whereURI) whereURI = whereURI.spec; } } catch (e) {} } this.addTab(whereURI, true); if (whereURI == "about:blank") BrowserUI.showAutoComplete(); // JavaScript Error Console if (gPrefService.getBoolPref("browser.console.showInPanel")){ let tool_console = document.getElementById("tool-console"); tool_console.hidden = false; } bv.commitBatchOperation(); // If some add-ons were disabled during during an application update, alert user if (gPrefService.prefHasUserValue("extensions.disabledAddons")) { let addons = gPrefService.getCharPref("extensions.disabledAddons").split(","); if (addons.length > 0) { let disabledStrings = Elements.browserBundle.getString("alertAddonsDisabled"); let label = PluralForm.get(addons.length, disabledStrings).replace("#1", addons.length); let alerts = Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService); alerts.showAlertNotification(URI_GENERIC_ICON_XPINSTALL, strings.getString("alertAddons"), label, false, "", null); } gPrefService.clearUserPref("extensions.disabledAddons"); } // some day (maybe fennec 1.0), we can remove both of these // preference reseting checks. they are here because we wanted to // disable plugins and flash explictly, then we fixed things and // needed to re-enable support. as time goes on, fewer people // will have ever had these temporary.* preference set. // Re-enable plugins if we had previously disabled them. We should get rid of // this code eventually... if (gPrefService.prefHasUserValue("temporary.disablePlugins")) { gPrefService.clearUserPref("temporary.disablePlugins"); this.setPluginState(true); } // Re-enable flash if (gPrefService.prefHasUserValue("temporary.disabledFlash")) { this.setPluginState(true, /flash/i); gPrefService.clearUserPref("temporary.disabledFlash"); } // Force commonly used border-images into the image cache ImagePreloader.cache(); this._pluginObserver = new PluginObserver(bv); new PreferenceToggle("plugins.enabled", this._pluginObserver); // broadcast a UIReady message so add-ons know we are finished with startup let event = document.createEvent("Events"); event.initEvent("UIReady", true, false); window.dispatchEvent(event); }, shutdown: function() { this._browserView.uninit(); BrowserUI.uninit(); this._pluginObserver.stop(); var os = Cc["@mozilla.org/observer-service;1"].getService(Ci.nsIObserverService); os.removeObserver(gXPInstallObserver, "xpinstall-install-blocked"); os.removeObserver(gSessionHistoryObserver, "browser:purge-session-history"); os.removeObserver(MemoryObserver, "memory-pressure"); #ifdef WINCE os.removeObserver(SoftKeyboardObserver, "softkb-change"); #endif os.removeObserver(BrowserSearch, "browser-search-engine-modified"); window.controllers.removeController(this); window.controllers.removeController(BrowserUI); }, setPluginState: function(enabled, nameMatch) { // XXX clear this out so that we always disable flash on startup, even // after the user has disabled/re-enabled plugins try { gPrefService.clearUserPref("temporary.disabledFlash"); } catch (ex) {} var phs = Cc["@mozilla.org/plugin/host;1"].getService(Ci.nsIPluginHost); var plugins = phs.getPluginTags({ }); for (var i = 0; i < plugins.length; ++i) { if (nameMatch && !nameMatch.test(plugins[i].name)) continue; plugins[i].disabled = !enabled; } }, get browsers() { return this._tabs.map(function(tab) { return tab.browser; }); }, scrollContentToTop: function scrollContentToTop() { this.contentScrollboxScroller.scrollTo(0, 0); this.pageScrollboxScroller.scrollTo(0, 0); this._browserView.onAfterVisibleMove(); }, /** Let current browser's scrollbox know about where content has been panned. */ scrollBrowserToContent: function scrollBrowserToContent() { let browser = this.selectedBrowser; if (browser) { let scroll = Browser.getScrollboxPosition(Browser.contentScrollboxScroller); let windowUtils = BrowserView.Util.getBrowserDOMWindowUtils(browser); browser.contentWindow.scrollTo(scroll.x, scroll.y); } }, /** Update viewport to location of browser's scrollbars. */ scrollContentToBrowser: function scrollContentToBrowser() { let bv = this._browserView; let pos = BrowserView.Util.getContentScrollOffset(this.selectedBrowser); pos.map(bv.browserToViewport); if (pos.y != 0) Browser.hideTitlebar(); else Browser.pageScrollboxScroller.scrollTo(0, 0); Browser.contentScrollboxScroller.scrollTo(pos.x, pos.y); bv.onAfterVisibleMove(); }, hideSidebars: function scrollSidebarsOffscreen() { let container = this.contentScrollbox; let rect = container.getBoundingClientRect(); this.controlsScrollboxScroller.scrollBy(Math.round(rect.left), 0); this._browserView.onAfterVisibleMove(); }, hideTitlebar: function hideTitlebar() { let container = this.contentScrollbox; let rect = container.getBoundingClientRect(); this.pageScrollboxScroller.scrollBy(0, Math.round(rect.top)); this.tryUnfloatToolbar(); this._browserView.onAfterVisibleMove(); }, /** * Return the currently active object */ get selectedBrowser() { return this._selectedTab.browser; }, getTabForDocument: function(aDocument) { let tabs = this._tabs; for (let i = 0; i < tabs.length; i++) { if (tabs[i].browser.contentDocument == aDocument) return tabs[i]; } return null; }, getTabAtIndex: function(index) { if (index > this._tabs.length || index < 0) return null; return this._tabs[index]; }, getTabFromChrome: function(chromeTab) { for (var t = 0; t < this._tabs.length; t++) { if (this._tabs[t].chromeTab == chromeTab) return this._tabs[t]; } return null; }, addTab: function(uri, bringFront) { let newTab = new Tab(); this._tabs.push(newTab); if (bringFront) this.selectedTab = newTab; newTab.load(uri); let event = document.createEvent("Events"); event.initEvent("TabOpen", true, false); newTab.chromeTab.dispatchEvent(event); return newTab; }, closeTab: function(tab) { if (tab instanceof XULElement) tab = this.getTabFromChrome(tab); if (!tab) return; let tabIndex = this._tabs.indexOf(tab); let nextTab = this._selectedTab; if (this._selectedTab == tab) { nextTab = this.getTabAtIndex(tabIndex + 1) || this.getTabAtIndex(tabIndex - 1); if (!nextTab) return; } let event = document.createEvent("Events"); event.initEvent("TabClose", true, false); tab.chromeTab.dispatchEvent(event); this.selectedTab = nextTab; tab.destroy(); this._tabs.splice(tabIndex, 1); }, get selectedTab() { return this._selectedTab; }, set selectedTab(tab) { let bv = this._browserView; if (tab instanceof XULElement) tab = this.getTabFromChrome(tab); if (!tab || this._selectedTab == tab) return; if (this._selectedTab) { this._selectedTab.scrollOffset = this.getScrollboxPosition(this.contentScrollboxScroller); } let isFirstTab = this._selectedTab == null; let lastTab = this._selectedTab; this._selectedTab = tab; tab.ensureBrowserExists(); bv.beginBatchOperation(); bv.setBrowser(tab.browser, tab.browserViewportState); bv.forceContainerResize(); document.getElementById("tabs").selectedTab = tab.chromeTab; if (!isFirstTab) { // Update all of our UI to reflect the new tab's location BrowserUI.updateURI(); getIdentityHandler().checkIdentity(); let event = document.createEvent("Events"); event.initEvent("TabSelect", true, false); event.lastTab = lastTab; tab.chromeTab.dispatchEvent(event); } tab.lastSelected = Date.now(); if (tab.scrollOffset) { // XXX incorrect behavior if page was scrolled by tab in the background. let { x: scrollX, y: scrollY } = tab.scrollOffset; Browser.contentScrollboxScroller.scrollTo(scrollX, scrollY); } bv.setAggressive(!tab._loading); bv.commitBatchOperation(); }, supportsCommand: function(cmd) { var isSupported = false; switch (cmd) { case "cmd_fullscreen": isSupported = true; break; default: isSupported = false; break; } return isSupported; }, isCommandEnabled: function(cmd) { return true; }, doCommand: function(cmd) { switch (cmd) { case "cmd_fullscreen": window.fullScreen = !window.fullScreen; break; } }, getNotificationBox: function getNotificationBox() { return document.getElementById("notifications"); }, removeTransientNotificationsForTab: function removeTransientNotificationsForTab(aTab) { let notificationBox = this.getNotificationBox(); let notifications = notificationBox.allNotifications; for (let n = notifications.length - 1; n >= 0; n--) { let notification = notifications[n]; if (notification._chromeTab != aTab.chromeTab) continue; if (notification.persistence) notification.persistence--; else if (Date.now() > notification.timeout) notificationBox.removeNotification(notification); } }, /** Returns true iff a tab's browser has been destroyed to free up memory. */ sacrificeTab: function sacrificeTab() { let tabToClear = this._tabs.reduce(function(prevTab, currentTab) { if (currentTab == Browser.selectedTab || !currentTab.browser) { return prevTab; } else { return (prevTab && prevTab.lastSelected <= currentTab.lastSelected) ? prevTab : currentTab; } }, null); if (tabToClear) { tabToClear.saveState(); tabToClear._destroyBrowser(); return true; } else { return false; } }, /** * Handle command event bubbling up from content. This allows us to do chrome- * privileged things based on buttons in, e.g., unprivileged error pages. * Obviously, care should be taken not to trust events that web pages could have * synthesized. */ _handleContentCommand: function _handleContentCommand(aEvent) { // Don't trust synthetic events if (!aEvent.isTrusted) return; var ot = aEvent.originalTarget; var errorDoc = ot.ownerDocument; // If the event came from an ssl error page, it is probably either the "Add // Exception…" or "Get me out of here!" button if (/^about:certerror\?e=nssBadCert/.test(errorDoc.documentURI)) { if (ot == errorDoc.getElementById("temporaryExceptionButton") || ot == errorDoc.getElementById("permanentExceptionButton")) { try { // add a new SSL exception for this URL let uri = gIOService.newURI(errorDoc.location.href, null, null); let sslExceptions = new SSLExceptions(); if (ot == errorDoc.getElementById("permanentExceptionButton")) { sslExceptions.addPermanentException(uri); } else { sslExceptions.addTemporaryException(uri); } } catch (e) { dump("EXCEPTION handle content command: " + e + "\n" ); } // automatically reload after the exception was added errorDoc.location.reload(); } else if (ot == errorDoc.getElementById('getMeOutOfHereButton')) { // Get the start page from the *default* pref branch, not the user's var defaultPrefs = Cc["@mozilla.org/preferences-service;1"] .getService(Ci.nsIPrefService).getDefaultBranch(null); var url = "about:blank"; try { url = defaultPrefs.getCharPref("browser.startup.homepage"); // If url is a pipe-delimited set of pages, just take the first one. if (url.indexOf("|") != -1) url = url.split("|")[0]; } catch (e) { /* Fall back on about blank */ } Browser.selectedBrowser.loadURI(url, null, null, false); } } else if (/^about:neterror\?e=netOffline/.test(errorDoc.documentURI)) { if (ot == errorDoc.getElementById("errorTryAgain")) { // Make sure we're online before attempting to load Util.forceOnline(); } } }, /** * Compute the sidebar percentage visibility. * * @param [optional] dx * @param [optional] dy an offset distance at which to perform the visibility * computation * @return [leftVisibility, rightVisiblity, leftTotalWidth, rightTotalWidth] */ computeSidebarVisibility: function computeSidebarVisibility(dx, dy) { function visibility(bar, visrect) { let w = bar.width; bar.restrictTo(visrect); return bar.width / w; } if (!dx) dx = 0; if (!dy) dy = 0; let leftbarCBR = document.getElementById('tabs-container').getBoundingClientRect(); let ritebarCBR = document.getElementById('browser-controls').getBoundingClientRect(); let leftbar = new Rect(Math.round(leftbarCBR.left) - dx, 0, Math.round(leftbarCBR.width), 1); let ritebar = new Rect(Math.round(ritebarCBR.left) - dx, 0, Math.round(ritebarCBR.width), 1); let leftw = leftbar.width; let ritew = ritebar.width; let visrect = new Rect(0, 0, window.innerWidth, 1); let leftvis = visibility(leftbar, visrect); let ritevis = visibility(ritebar, visrect); return [leftvis, ritevis, leftw, ritew]; }, /** * Compute the horizontal distance needed to scroll in order to snap the * sidebars into place. * * Visibility is computed by creating dummy rectangles for the sidebar and the * visible rect. Sidebar rectangles come from getBoundingClientRect(), so * they are in absolute client coordinates (and since we're in a scrollbox, * this means they are positioned relative to the window, which is anchored at * (0, 0) regardless of the scrollbox's scroll position. The rectangles are * made to have a top of 0 and a height of 1, since these do not affect how we * compute visibility (we care only about width), and using rectangles allows * us to use restrictTo(), which comes in handy. * * @return scrollBy dx needed to make snap happen */ snapSidebars: function snapSidebars() { let [leftvis, ritevis, leftw, ritew] = Browser.computeSidebarVisibility(); let snappedX = 0; if (leftvis != 0 && leftvis != 1) { if (leftvis >= 0.6666) { snappedX = -((1 - leftvis) * leftw); } else { snappedX = leftvis * leftw; } } else if (ritevis != 0 && ritevis != 1) { if (ritevis >= 0.6666) { snappedX = (1 - ritevis) * ritew; } else { snappedX = -ritevis * ritew; } } return Math.round(snappedX); }, tryFloatToolbar: function tryFloatToolbar(dx, dy) { if (this.floatedWhileDragging) return; let [leftvis, ritevis, leftw, ritew] = Browser.computeSidebarVisibility(dx, dy); if (leftvis > 0 || ritevis > 0) { BrowserUI.lockToolbar(); this.floatedWhileDragging = true; } }, tryUnfloatToolbar: function tryUnfloatToolbar(dx, dy) { if (!this.floatedWhileDragging) return true; let [leftvis, ritevis, leftw, ritew] = Browser.computeSidebarVisibility(dx, dy); if (leftvis == 0 && ritevis == 0) { BrowserUI.unlockToolbar(); this.floatedWhileDragging = false; return true; } return false; }, zoom: function zoom(aDirection) { let bv = this._browserView; let zoomLevel = bv.getZoomLevel() + (aDirection > 0 ? -0.1 : 0.1); let center = this.getVisibleRect().center().map(bv.viewportToBrowser); this.setVisibleRect(this._getZoomRectForPoint(center.x, center.y, zoomLevel)); }, /** * Find the needed zoom level for zooming on an element */ _getZoomLevelForElement: function _getZoomLevelForElement(element) { const margin = 15; let bv = this._browserView; let elRect = bv.browserToViewportRect(Browser.getBoundingContentRect(element)); let vis = bv.getVisibleRect(); return BrowserView.Util.clampZoomLevel(bv.getZoomLevel() * vis.width / (elRect.width + margin * 2)); }, /** Find an appropriate zoom rect for an element, if it exists. */ _getZoomRectForElement: function _getZoomRectForElement(element, elementY) { let bv = this._browserView; let oldZoomLevel = bv.getZoomLevel(); let zoomLevel = this._getZoomLevelForElement(element); let zoomRatio = oldZoomLevel / zoomLevel; // Don't zoom in a marginal amount, but be more lenient for the first zoom. // > 2/3 means operation increases the zoom level by less than 1.5 // > 9/10 means operation increases the zoom level by less than 1.1 let zoomTolerance = (bv.isDefaultZoom()) ? .9 : .6666; if (zoomRatio >= zoomTolerance) { return null; } else { let elRect = this.getBoundingContentRect(element); return this._getZoomRectForPoint(elRect.center().x, elementY, zoomLevel); } }, /** * Find a good zoom rectangle for point specified in browser coordinates. * @return Point in viewport coordinates */ _getZoomRectForPoint: function _getZoomRectForPoint(x, y, zoomLevel) { let bv = this._browserView; let vis = bv.getVisibleRect(); x = bv.browserToViewport(x); y = bv.browserToViewport(y); zoomLevel = Math.min(kBrowserViewZoomLevelMax, zoomLevel); let zoomRatio = zoomLevel / bv.getZoomLevel(); let newVisW = vis.width / zoomRatio, newVisH = vis.height / zoomRatio; let result = new Rect(x - newVisW / 2, y - newVisH / 2, newVisW, newVisH); // Make sure rectangle doesn't poke out of viewport return result.translateInside(bv._browserViewportState.viewportRect); }, setVisibleRect: function setVisibleRect(rect) { let bv = this._browserView; let vis = bv.getVisibleRect(); let zoomRatio = vis.width / rect.width; let zoomLevel = bv.getZoomLevel() * zoomRatio; let scrollX = rect.left * zoomRatio; let scrollY = rect.top * zoomRatio; // The order of operations below is important for artifacting and for performance. Important // side effects of functions are noted below. // Hardware scrolling happens immediately when scrollTo is called. Hide to prevent artifacts. bv.beginOffscreenOperation(); // We must scroll to the correct area before TileManager is informed of the change // so that only one render is done. Ensures setZoomLevel puts it off. bv.beginBatchOperation(); // Critical rect changes when controls are hidden. Must hide before tilemanager viewport. this.hideSidebars(); this.hideTitlebar(); bv.setZoomLevel(zoomLevel); // Ensure container is big enough for scroll values. bv.forceContainerResize(); this.forceChromeReflow(); this.contentScrollboxScroller.scrollTo(scrollX, scrollY); bv.onAfterVisibleMove(); // Inform tile manager, which happens to render new tiles too. Must call in case a batch // operation was in progress before zoom. bv.forceViewportChange(); bv.commitBatchOperation(); bv.commitOffscreenOperation(); }, zoomToPoint: function zoomToPoint(cX, cY) { let [elementX, elementY] = this.transformClientToBrowser(cX, cY); let zoomRect = null; let element = this.elementFromPoint(elementX, elementY); let bv = this._browserView; if (element) zoomRect = this._getZoomRectForElement(element, elementY); if (!zoomRect && bv.isDefaultZoom()) zoomRect = this._getZoomRectForPoint(elementX, elementY, bv.getZoomLevel() * 2); if (zoomRect) { this.setVisibleRect(zoomRect); return true; } else { return false; } }, zoomFromPoint: function zoomFromPoint(cX, cY) { let bv = this._browserView; let zoomLevel = bv.getZoomForPage(); if (!bv.isDefaultZoom()) { let [elementX, elementY] = this.transformClientToBrowser(cX, cY); let zoomRect = this._getZoomRectForPoint(elementX, elementY, zoomLevel); this.setVisibleRect(zoomRect); } }, getBoundingContentRect: function getBoundingContentRect(contentElem) { let document = contentElem.ownerDocument; while(document.defaultView.frameElement) document = document.defaultView.frameElement.ownerDocument; let tab = Browser.getTabForDocument(document); if (!tab || !tab.browser) return null; let browser = tab.browser; let offset = BrowserView.Util.getContentScrollOffset(browser); let r = contentElem.getBoundingClientRect(); // step out of iframes and frames, offsetting scroll values for (let frame = contentElem.ownerDocument.defaultView; frame != browser.contentWindow; frame = frame.parent) { // adjust client coordinates' origin to be top left of iframe viewport let rect = frame.frameElement.getBoundingClientRect(); offset.add(rect.left, rect.top); } return new Rect(r.left + offset.x, r.top + offset.y, r.width, r.height); }, /** * Transform x and y from client coordinates to BrowserView coordinates. */ clientToBrowserView: function clientToBrowserView(x, y) { let container = document.getElementById("tile-container"); let containerBCR = container.getBoundingClientRect(); let x0 = Math.round(containerBCR.left); let y0; if (arguments.length > 1) y0 = Math.round(containerBCR.top); return (arguments.length > 1) ? [x - x0, y - y0] : (x - x0); }, browserViewToClient: function browserViewToClient(x, y) { let container = document.getElementById("tile-container"); let containerBCR = container.getBoundingClientRect(); let x0 = Math.round(-containerBCR.left); let y0; if (arguments.length > 1) y0 = Math.round(-containerBCR.top); return (arguments.length > 1) ? [x - x0, y - y0] : (x - x0); }, browserViewToClientRect: function browserViewToClientRect(rect) { let container = document.getElementById("tile-container"); let containerBCR = container.getBoundingClientRect(); return rect.clone().translate(Math.round(containerBCR.left), Math.round(containerBCR.top)); }, /** * turn client coordinates into page-relative ones (adjusted for * zoom and page position) */ transformClientToBrowser: function transformClientToBrowser(cX, cY) { return this.clientToBrowserView(cX, cY).map(this._browserView.viewportToBrowser); }, /** * @param x,y Browser coordinates * @return Element at position, null if no active browser or no element found */ elementFromPoint: function elementFromPoint(x, y) { //Util.dumpLn("*** elementFromPoint: page ", x, ",", y); let browser = this._browserView.getBrowser(); if (!browser) return null; // browser's elementFromPoint expect browser-relative client coordinates. // subtract browser's scroll values to adjust let cwu = BrowserView.Util.getBrowserDOMWindowUtils(browser); let scrollX = {}, scrollY = {}; cwu.getScrollXY(false, scrollX, scrollY); x = x - scrollX.value; y = y - scrollY.value; let elem = cwu.elementFromPoint(x, y, true, /* ignore root scroll frame*/ false); /* don't flush layout */ // step through layers of IFRAMEs and FRAMES to find innermost element while (elem && (elem instanceof HTMLIFrameElement || elem instanceof HTMLFrameElement)) { // adjust client coordinates' origin to be top left of iframe viewport let rect = elem.getBoundingClientRect(); x = x - rect.left; y = y - rect.top; elem = elem.contentDocument.elementFromPoint(x, y); } return elem; }, /** * Return the visible rect in coordinates with origin at the (left, top) of * the tile container, i.e. BrowserView coordinates. */ getVisibleRect: function getVisibleRect() { let stack = document.getElementById("tile-stack"); let container = document.getElementById("tile-container"); let containerBCR = container.getBoundingClientRect(); let x = Math.round(-containerBCR.left); let y = Math.round(-containerBCR.top); let w = window.innerWidth; let h = stack.getBoundingClientRect().height; return new Rect(x, y, w, h); }, /** * Convenience function for getting the scrollbox position off of a * scrollBoxObject interface. Returns the actual values instead of the * wrapping objects. * * @param scroller a scrollBoxObject on which to call scroller.getPosition() */ getScrollboxPosition: function getScrollboxPosition(scroller) { let x = {}; let y = {}; scroller.getPosition(x, y); return new Point(x.value, y.value); }, forceChromeReflow: function forceChromeReflow() { let dummy = getComputedStyle(document.documentElement, "").width; } }; Browser.MainDragger = function MainDragger(browserView) { this.bv = browserView; this.draggedFrame = null; this.contentScrollbox = null; }; Browser.MainDragger.prototype = { isDraggable: function isDraggable(target, scroller) { return true; }, dragStart: function dragStart(clientX, clientY, target, scroller) { let [x, y] = Browser.transformClientToBrowser(clientX, clientY); let element = Browser.elementFromPoint(x, y); this.draggedFrame = null; this.contentScrollbox = null; // Check if we are in a scrollable HTML element let htmlElement = element; if (htmlElement && htmlElement instanceof HTMLElement) { let win = htmlElement.ownerDocument.defaultView; for (; htmlElement; htmlElement = htmlElement.parentNode) { try { let cs = win.getComputedStyle(htmlElement, null); let overflow = cs.getPropertyValue("overflow"); let overflowX = cs.getPropertyValue("overflow-x"); let overflowY = cs.getPropertyValue("overflow-y"); let cbr = htmlElement.getBoundingClientRect(); let oScroll = (overflow == "scroll") || (overflowX == "scroll") || (overflowY == "scroll"); let oAuto = (overflow == "auto") || (overflowX == "auto") || (overflowY == "auto"); if (oScroll || (oAuto && (cbr.height < target.scrollHeight || cbr.width < target.scrollWidth))) { this.contentScrollbox = this._createDivScrollBox(htmlElement); return; } } catch(e) {} } } // Check if we are in XUL land let xulElement = element; if (xulElement && xulElement instanceof XULElement) { for (; xulElement; xulElement = xulElement.parentNode) { if (xulElement.localName == "treechildren") { this.contentScrollbox = this._createTreeScrollBox(xulElement.parentNode); return; } let wrapper = xulElement.wrappedJSObject; let scrollable = false; try { scrollable = (wrapper.scrollBoxObject != null) || (wrapper.boxObject.QueryInterface(Ci.nsIScrollBoxObject)); } catch(e) {} if (scrollable) { this.contentScrollbox = wrapper.scrollBoxObject || wrapper.boxObject.QueryInterface(Ci.nsIScrollBoxObject); return; } } } if (element) this.draggedFrame = element.ownerDocument.defaultView; this.bv.pauseRendering(); }, dragStop: function dragStop(dx, dy, scroller) { this.draggedFrame = null; this.dragMove(Browser.snapSidebars(), 0, scroller); Browser.tryUnfloatToolbar(); this.bv.resumeRendering(); }, dragMove: function dragMove(dx, dy, scroller) { let elem = this.draggedFrame; let doffset = new Point(dx, dy); let render = false; this.bv.onBeforeVisibleMove(dx, dy); // First calculate any panning to take sidebars out of view let panOffset = this._panControlsAwayOffset(doffset); // do HTML overflow or XUL panning if (this.contentScrollbox && !doffset.isZero()) this._panScrollbox(this.contentScrollbox, doffset); // Do all iframe panning if (elem) { while (elem.frameElement && !doffset.isZero()) { this._panFrame(elem, doffset); elem = elem.frameElement; render = true; } } // Do content panning this._panScroller(Browser.contentScrollboxScroller, doffset); // Any leftover panning in doffset would bring controls into view. Add to sidebar // away panning for the total scroll offset. doffset.add(panOffset); Browser.tryFloatToolbar(doffset.x, 0); this._panScroller(Browser.controlsScrollboxScroller, doffset); this._panScroller(Browser.pageScrollboxScroller, doffset); this.bv.onAfterVisibleMove(); if (render) this.bv.renderNow(); return !doffset.equals(dx, dy); }, /** * builds a minimal implementation of scrollBoxObject for div */ _createDivScrollBox: function(div) { let sbo = { getScrolledSize: function(width, height) { width.value = div.scrollWidth; height.value = div.scrollHeight; }, getPosition: function(x, y) { x.value = div.scrollLeft; y.value = div.scrollTop; }, scrollBy: function(dx, dy) { div.scrollTop += dy; div.scrollLeft += dx; } } return sbo; }, /** * builds a minimal implementation of scrollBoxObject for trees */ _createTreeScrollBox: function(tree) { let treeBox = tree.boxObject.QueryInterface(Ci.nsITreeBoxObject); let sbo = { pageLength: treeBox.getPageLength(), rowHeight: treeBox.rowHeight, rowWidth: treeBox.rowWidth, rowCount: treeBox.view.rowCount, targetY: treeBox.getFirstVisibleRow() * treeBox.rowHeight, getScrolledSize: function(width, height) { width.value = this.rowWidth; height.value = this.rowHeight * this.rowCount; }, getPosition: function(x, y) { x.value = treeBox.horizontalPosition; y.value = this.targetY; }, scrollBy: function(dx, dy) { this.targetY += dy; if (this.targetY < 0) this.targetY = 0; let targetRow = Math.floor(this.targetY / this.rowHeight); if ((targetRow + this.pageLength) > this.rowCount) { targetRow = this.rowCount - this.pageLength; this.targetY = targetRow * this.rowHeight; } treeBox.scrollToRow(targetRow); treeBox.scrollToHorizontalPosition(treeBox.horizontalPosition + dx); } } return sbo; }, /** * pans a scrollbox, updating doffset */ _panScrollbox: function(sbo, doffset) { let origX = {}, origY = {}, newX = {}, newY = {}; sbo.getPosition(origX, origY); sbo.scrollBy(doffset.x, doffset.y); sbo.getPosition(newX, newY); doffset.subtract(newX.value - origX.value, newY.value - origY.value); }, /** Return offset that pans controls away from screen. Updates doffset with leftovers. */ _panControlsAwayOffset: function(doffset) { let x = 0, y = 0, rect; rect = Rect.fromRect(Browser.pageScrollbox.getBoundingClientRect()).map(Math.round); if (doffset.x < 0 && rect.right < window.innerWidth) x = Math.max(doffset.x, rect.right - window.innerWidth); if (doffset.x > 0 && rect.left > 0) x = Math.min(doffset.x, rect.left); let height = document.getElementById("tile-stack").getBoundingClientRect().height; rect = Rect.fromRect(Browser.contentScrollbox.getBoundingClientRect()).map(Math.round); if (doffset.y < 0 && rect.bottom < height) y = Math.max(doffset.y, rect.bottom - height); if (doffset.y > 0 && rect.top > 0) y = Math.min(doffset.y, rect.top); doffset.subtract(x, y); return new Point(x, y); }, /** Pan scroller by the given amount. Updates doffset with leftovers. */ _panScroller: function _panScroller(scroller, doffset) { let { x: x0, y: y0 } = Browser.getScrollboxPosition(scroller); scroller.scrollBy(doffset.x, doffset.y); let { x: x1, y: y1 } = Browser.getScrollboxPosition(scroller); doffset.subtract(x1 - x0, y1 - y0); }, /** Pan frame by the given amount. Updates doffset with leftovers. */ _panFrame: function _panFrame(frame, doffset) { let origX = {}, origY = {}, newX = {}, newY = {}; let windowUtils = frame.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindowUtils); windowUtils.getScrollXY(false, origX, origY); frame.scrollBy(doffset.x, doffset.y); windowUtils.getScrollXY(false, newX, newY); doffset.subtract(newX.value - origX.value, newY.value - origY.value); } }; function nsBrowserAccess() { } nsBrowserAccess.prototype = { QueryInterface: function(aIID) { if (aIID.equals(Ci.nsIBrowserDOMWindow) || aIID.equals(Ci.nsISupports)) return this; throw Components.results.NS_NOINTERFACE; }, openURI: function(aURI, aOpener, aWhere, aContext) { var isExternal = (aContext == Ci.nsIBrowserDOMWindow.OPEN_EXTERNAL); if (isExternal && aURI && aURI.schemeIs("chrome")) { //dump("use -chrome command-line option to load external chrome urls\n"); return null; } var loadflags = isExternal ? Ci.nsIWebNavigation.LOAD_FLAGS_FROM_EXTERNAL : Ci.nsIWebNavigation.LOAD_FLAGS_NONE; var location; if (aWhere == Ci.nsIBrowserDOMWindow.OPEN_DEFAULTWINDOW) { switch (aContext) { case Ci.nsIBrowserDOMWindow.OPEN_EXTERNAL : aWhere = gPrefService.getIntPref("browser.link.open_external"); break; default : // OPEN_NEW or an illegal value aWhere = gPrefService.getIntPref("browser.link.open_newwindow"); } } var newWindow; if (aWhere == Ci.nsIBrowserDOMWindow.OPEN_NEWWINDOW) { var url = aURI ? aURI.spec : "about:blank"; newWindow = openDialog("chrome://browser/content/browser.xul", "_blank", "all,dialog=no", url, null, null, null); } else { if (aWhere == Ci.nsIBrowserDOMWindow.OPEN_NEWTAB) newWindow = Browser.addTab("about:blank", true).browser.contentWindow; else newWindow = aOpener ? aOpener.top : browser.contentWindow; } try { var referrer; if (aURI) { if (aOpener) { location = aOpener.location; referrer = gIOService.newURI(location, null, null); } newWindow.QueryInterface(Ci.nsIInterfaceRequestor) .getInterface(Ci.nsIWebNavigation) .loadURI(aURI.spec, loadflags, referrer, null, null); } newWindow.focus(); } catch(e) { } return newWindow; }, isTabContentWindow: function(aWindow) { return Browser.browsers.some(function (browser) browser.contentWindow == aWindow); } }; const BrowserSearch = { engines: null, _allEngines: [], observe: function (aSubject, aTopic, aData) { if (aTopic != "browser-search-engine-modified") return; switch (aData) { case "engine-added": case "engine-removed": // force a rebuild of the prefs list, if needed // XXX this is inefficient, shouldn't have to rebuild the entire list if (ExtensionsView._list) ExtensionsView.getAddonsFromLocal(); // fall through case "engine-changed": // XXX we should probably also update the ExtensionsView list here once // that's efficient, since the icon can change (happen during an async // installs from the web) // blow away our cache this._engines = null; break; case "engine-current": // Not relevant break; } }, get _currentEngines() { let doc = getBrowser().contentDocument; return this._allEngines.filter(function(element) element.doc === doc, this); }, get searchService() { delete this.searchService; return this.searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); }, get engines() { if (this._engines) return this._engines; return this._engines = this.searchService.getVisibleEngines({ }); }, addPageSearchEngine: function (aEngine, aDocument) { // Clean the engine referenced for document that didn't exist anymore let browsers = Browser.browsers; this._allEngines = this._allEngines.filter(function(element) { return browsers.some(function (browser) browser.contentDocument == element.doc); }, this); // Prevent duplicate if (!this._allEngines.some(function (e) { return (e.engine.title == aEngine.title) && (e.doc == aDocument); })) this._allEngines.push( {engine:aEngine, doc:aDocument}); }, updatePageSearchEngines: function() { // Check to see whether we've already added an engine with this title in // the search list let newEngines = this._currentEngines.filter(function(element) { return !this.engines.some(function (e) e.name == element.engine.title); }, this); let container = document.getElementById('search-container'); let buttons = container.getElementsByAttribute("class", "search-engine-button button-dark"); for (let i=0; i