gecko/mobile/chrome/content/browser.js

2517 lines
82 KiB
JavaScript
Raw Normal View History

// -*- 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 <blassey@mozilla.com>
* Mark Finkle <mfinkle@mozilla.com>
* Aleks Totic <a@totic.org>
* Johnathan Nightingale <johnath@mozilla.com>
* Stuart Parmenter <stuart@mozilla.com>
* Taras Glek <tglek@mozilla.com>
* Roy Frostig <rfrostig@mozilla.com>
* Ben Combee <bcombee@mozilla.com>
*
* 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 FINDSTATE_FIND = 0;
const FINDSTATE_FIND_AGAIN = 1;
const FINDSTATE_FIND_PREVIOUS = 2;
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;
2009-07-17 16:17:57 -07:00
function debug() {
let bv = Browser._browserView;
let tc = bv._tileManager._tileCache;
let scrollbox = document.getElementById("tile-container-container")
.boxObject.QueryInterface(Ci.nsIScrollBoxObject);
2009-07-17 16:17:57 -07:00
let x = {};
let y = {};
2009-07-17 16:17:57 -07:00
let w = {};
let h = {};
scrollbox.getPosition(x, y);
2009-07-17 16:17:57 -07:00
scrollbox.getScrolledSize(w, h);
let container = document.getElementById("tile-container");
let [x, y] = [x.value, y.value];
2009-07-17 16:17:57 -07:00
let [w, h] = [w.value, h.value];
if (bv) {
dump('----------------------DEBUG!-------------------------\n');
dump(bv._browserViewportState.toString() + endl);
dump(endl);
2009-08-03 20:32:17 -07:00
dump('location from Browser: ' + Browser.selectedBrowser.contentWindow.location + endl);
dump('location from BV : ' + bv.getBrowser().contentWindow.location + endl);
dump(endl + endl);
2009-07-17 16:17:57 -07:00
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);
2009-07-17 16:17:57 -07:00
dump('bv batchops depth: ' + bv._batchOps.length + endl);
dump('renderpause depth: ' + bv._renderMode + endl);
2009-07-17 16:17:57 -07:00
dump(endl);
dump('window.innerWidth : ' + window.innerWidth + endl);
dump('window.innerHeight: ' + window.innerHeight + endl);
dump(endl);
2009-07-17 16:17:57 -07:00
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("tile-container-container");
dump('container location: ' + Math.round(container.getBoundingClientRect().left) + " " +
Math.round(container.getBoundingClientRect().top) + endl);
2009-08-03 09:44:17 -07:00
dump(endl);
2009-08-03 09:44:17 -07:00
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);
2009-07-17 16:17:57 -07:00
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);
2009-07-17 16:17:57 -07:00
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) {
2009-07-17 16:17:57 -07:00
let bv = Browser._browserView;
if (!ev.ctrlKey)
2009-07-17 16:17:57 -07:00
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
2009-07-17 16:17:57 -07:00
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());
2009-07-17 16:17:57 -07:00
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;
2009-07-17 16:17:57 -07:00
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); });
2009-07-17 16:17:57 -07:00
debugTile(i, j);
}
break;
case a:
let cr = bv._tileManager._criticalRect;
dump('>>>>>> critical rect is ' + (cr ? cr.toString() : cr) + '\n');
2009-07-17 16:17:57 -07:00
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:
bv.resumeRendering();
2009-07-17 16:17:57 -07:00
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;
2009-07-17 16:17:57 -07:00
default:
break;
}
}
window.infoMode = false;
window.tileMapMode = false;
2009-07-17 16:17:57 -07:00
var ih = null;
var Browser = {
_tabs : [],
_selectedTab : null,
windowUtils: window.QueryInterface(Ci.nsIInterfaceRequestor)
.getInterface(Ci.nsIDOMWindowUtils),
2009-08-05 14:06:38 -07:00
contentScrollbox: null,
contentScrollboxScroller: null,
controlsScrollbox: null,
controlsScrollboxScroller: null,
startup: function() {
var self = this;
//dump("begin startup\n");
2009-07-22 06:12:02 -07:00
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);
/* vertically scrolling box that contains tiles and the urlbar */
2009-08-05 14:06:38 -07:00
let contentScrollbox = this.contentScrollbox = document.getElementById("tile-container-container");
this.contentScrollboxScroller = contentScrollbox.boxObject.QueryInterface(Ci.nsIScrollBoxObject);
contentScrollbox.customDragger = new Browser.MainDragger(bv);
2009-08-05 14:06:38 -07:00
/* horizontally scrolling box that holds the sidebars as well as the contentScrollbox */
2009-08-05 14:06:38 -07:00
let controlsScrollbox = this.controlsScrollbox = document.getElementById("scrollbox");
this.controlsScrollboxScroller = controlsScrollbox.boxObject.QueryInterface(Ci.nsIScrollBoxObject);
controlsScrollbox.customDragger = {
dragStart: function dragStart(cx, cy, target, scroller) {},
dragStop: function dragStop(dx, dy, scroller) { return false; },
dragMove: function dragMove(dx, dy, scroller) { return false; }
};
// during startup a lot of viewportHandler calls happen due to content and window resizes
2009-07-17 16:17:57 -07:00
bv.beginBatchOperation();
function resizeHandler(e) {
2009-07-17 16:17:57 -07:00
if (e.target != window)
return;
//dump(window.innerWidth + "," + window.innerHeight + "\n");
2009-07-22 06:12:02 -07:00
// 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;
2009-07-17 16:17:57 -07:00
bv.beginBatchOperation();
2009-08-05 13:28:43 -07:00
contentScrollbox.style.width = w + 'px';
contentScrollbox.style.height = h + 'px';
controlsScrollbox.style.width = w + 'px';
controlsScrollbox.style.height = h + 'px';
let toolbarHeight = Math.round(document.getElementById("toolbar-main").getBoundingClientRect().height);
let spacers = document.getElementsByClassName("sidebar-spacer");
2009-08-07 14:09:32 -07:00
for (let i = 0, len = spacers.length; i < len; i++) spacers[i].style.height = toolbarHeight + 'px';
// toolbar UI
document.getElementById("toolbar-main").width = w;
// Tell the UI to resize the browser controls before calling updateSize
BrowserUI.sizeControls(w, h);
// Resize the browsers...
let browsers = Browser.browsers;
if (browsers) {
let scaledDefaultH = (kDefaultBrowserWidth * (h / w));
let scaledScreenH = (window.screen.width * (h / w));
for (let i=0; i<browsers.length; i++) {
let browserStyle = browsers[i].style;
browserStyle.height = ((browsers[i].hasOwnProperty("handheld") &&
browsers[i].handheld) ?
scaledScreenH : scaledDefaultH) + "px";
}
}
2009-07-17 16:17:57 -07:00
bv.zoomToPage();
Browser.hideSidebars();
// hidesidebars calls bv.onAfterVisibleMove();
2009-07-17 16:17:57 -07:00
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 ios = Cc["@mozilla.org/network/io-service;1"].getService(Ci.nsIIOService);
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 = ios.newURI("chrome://browser/content/cursor.css", null, null);
styleSheets.loadAndRegisterSheet(styleURI, styleSheets.AGENT_SHEET);
}
// load styles for scrollbars
var styleURI = ios.newURI("chrome://browser/content/content.css", null, null);
styleSheets.loadAndRegisterSheet(styleURI, styleSheets.AGENT_SHEET);
2009-08-07 14:09:32 -07:00
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);
2008-09-24 09:14:06 -07:00
window.QueryInterface(Ci.nsIDOMChromeWindow).browserDOMWindow = new nsBrowserAccess();
let browsers = document.getElementById("browsers");
browsers.addEventListener("command", this._handleContentCommand, false);
browsers.addEventListener("DOMUpdatePageReport", gPopupBlockerObserver.onUpdatePageReport, false);
/* Initialize Spatial Navigation */
function panCallback(aElement) {
if (!aElement)
return;
2009-08-07 14:09:32 -07:00
// 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 <browser>s
SpatialNavigation.init(browsers, panCallback);
/* Login Manager */
Cc["@mozilla.org/login-manager;1"].getService(Ci.nsILoginManager);
/* Command line arguments/initial homepage */
var whereURI = "about:blank";
try {
// Check for and use a default homepage
whereURI = gPrefService.getCharPref("browser.startup.homepage");
} catch (e) {}
// 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] &&
2009-09-21 03:04:50 -07:00
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);
// JavaScript Error Console
if (gPrefService.getBoolPref("browser.console.showInPanel")){
let tool_console = document.getElementById("tool-console");
tool_console.hidden = false;
}
// 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);
}
2009-07-17 16:17:57 -07:00
bv.commitBatchOperation();
2009-07-22 06:12:02 -07:00
// 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 = document.getElementById("bundle_browser").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");
}
//dump("end startup\n");
},
shutdown: function() {
this._browserView.setBrowser(null, null, false);
BrowserUI.uninit();
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);
},
2008-10-06 12:30:55 -07:00
setPluginState: function(enabled)
{
var phs = Cc["@mozilla.org/plugin/host;1"].getService(Ci.nsIPluginHost);
var plugins = phs.getPluginTags({ });
for (var i = 0; i < plugins.length; ++i)
plugins[i].disabled = !enabled;
},
get browsers() {
return this._tabs.map(function(tab) { return tab.browser; });
},
scrollContentToTop: function scrollContentToTop() {
this.contentScrollboxScroller.scrollTo(0, 0);
this._browserView.onAfterVisibleMove();
},
hideSidebars: function scrollSidebarsOffscreen() {
let container = document.getElementById("tile-container");
let containerBCR = container.getBoundingClientRect();
// round here because when going to full screen, this value tends
// to be very small but negative
let dx = Math.round(containerBCR.left);
if (dx < 0)
dx = Math.min(Math.round(containerBCR.right - window.innerWidth), 0);
this.controlsScrollboxScroller.scrollBy(dx, 0);
Browser.contentScrollbox.customDragger.scrollingOuterX = false; // XXX ugh.
this._browserView.onAfterVisibleMove();
},
2009-07-17 16:17:57 -07:00
/**
* Return the currently active <browser> object
*/
get selectedBrowser() {
return this._selectedTab.browser;
},
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);
2009-07-17 16:17:57 -07:00
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) {
2009-07-17 16:17:57 -07:00
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 firstTab = this._selectedTab == null;
this._selectedTab = tab;
tab.ensureBrowserExists();
2009-07-17 16:17:57 -07:00
bv.beginBatchOperation();
bv.setBrowser(tab.browser, tab.browserViewportState, false);
bv.forceContainerResize();
document.getElementById("tabs").selectedTab = tab.chromeTab;
if (!firstTab) {
// 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);
tab.chromeTab.dispatchEvent(event);
}
tab.lastSelected = Date.now();
if (tab.scrollOffset) {
// XXX incorrect behavior if page was scrolled by tab in the background.
let [scrollX, scrollY] = tab.scrollOffset;
Browser.contentScrollboxScroller.scrollTo(scrollX, scrollY);
}
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() {
return document.getElementById("notifications");
},
findState: FINDSTATE_FIND,
openFind: function(aState) {
this.findState = aState;
var findbar = document.getElementById("findbar");
if (!findbar.browser)
findbar.browser = this.selectedBrowser;
var panel = document.getElementById("findbar-container");
if (panel.hidden) {
panel.hidden = false;
}
this.doFind();
},
doFind: function() {
var findbar = document.getElementById("findbar");
if (Browser.findState == FINDSTATE_FIND)
findbar.onFindCommand();
else
findbar.onFindAgainCommand(Browser.findState == FINDSTATE_FIND_PREVIOUS);
var panel = document.getElementById("findbar-container");
panel.top = window.innerHeight - Math.floor(findbar.getBoundingClientRect().height);
2008-08-05 13:19:22 -07:00
},
2008-08-05 13:19:22 -07:00
translatePhoneNumbers: function() {
let doc = getBrowser().contentDocument;
// jonas black magic (only match text nodes that contain a sequence of 4 numbers)
let textnodes = doc.evaluate('//text()[contains(translate(., "0123456789", "^^^^^^^^^^"), "^^^^")]',
doc,
null,
XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE,
null);
let s, node, lastLastIndex;
let re = /(\+?1? ?-?\(?\d{3}\)?[ +-\.]\d{3}[ +-\.]\d{4})/;
2008-08-05 13:19:22 -07:00
for (var i = 0; i < textnodes.snapshotLength; i++) {
node = textnodes.snapshotItem(i);
s = node.data;
if (s.match(re)) {
s = s.replace(re, "<a href='tel:$1'> $1 </a>");
try {
let replacement = doc.createElement("span");
replacement.innerHTML = s;
node.parentNode.insertBefore(replacement, node);
node.parentNode.removeChild(node);
} catch(e) {
//do nothing, but continue
}
}
}
},
/** 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.
*/
2009-08-07 14:59:06 -07:00
_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")) {
var params = { exceptionAdded : false };
try {
switch (gPrefService.getIntPref("browser.ssl_override_behavior")) {
case 2 : // Pre-fetch & pre-populate
params.prefetchCert = true;
case 1 : // Pre-populate
params.location = errorDoc.location.href;
}
} catch (e) {
Components.utils.reportError("Couldn't get ssl_override pref: " + e);
}
window.openDialog('chrome://pippki/content/exceptionDialog.xul',
'','chrome,centerscreen,modal', params);
// If the user added the exception cert, attempt to reload the page
if (params.exceptionAdded)
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);
}
}
},
/**
* 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) {
// XXX these should return 0 if the sidebars aren't visible
function visibility(bar, visrect) {
try {
let w = bar.width;
let h = bar.height;
bar.restrictTo(visrect); // throws exception if intersection of rects is empty
return bar.width / w;
} catch (e) {
return 0;
}
}
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 wsRect(Math.round(leftbarCBR.left) - dx, 0, Math.round(leftbarCBR.width), 1);
let ritebar = new wsRect(Math.round(ritebarCBR.left) - dx, 0, Math.round(ritebarCBR.width), 1);
let leftw = leftbar.width;
let ritew = ritebar.width;
let visrect = new wsRect(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;
2009-08-05 13:28:43 -07:00
let [leftvis, ritevis, leftw, ritew] = Browser.computeSidebarVisibility(dx, dy);
// XXX computeSideBarVisibility will normally return 0.0015... for ritevis
if (leftvis > 0.002 || ritevis > 0.002) {
BrowserUI.lockToolbar();
this.floatedWhileDragging = true;
}
},
2009-08-05 13:28:43 -07:00
tryUnfloatToolbar: function tryUnfloatToolbar(dx, dy) {
if (!this.floatedWhileDragging)
return true;
2009-08-05 13:28:43 -07:00
let [leftvis, ritevis, leftw, ritew] = Browser.computeSidebarVisibility(dx, dy);
if (leftvis <= 0.002 && ritevis <= 0.002) {
BrowserUI.unlockToolbar();
this.floatedWhileDragging = false;
return true;
2009-08-05 13:28:43 -07:00
}
return false;
2009-08-05 13:28:43 -07:00
},
zoom: function zoom(aDirection) {
Browser._browserView.zoom(aDirection);
//Browser.forceChromeReflow(); // Zoom causes a width/height change to the
// BrowserView's containing element, but
// sometimes doesn't cause the reflow that
// resizes the parent right away, so we
// can preempt it here. Not needed anywhere
// currently but if this becomes API or is
// needed then uncomment this line.
},
zoomToPoint: function zoomToPoint(cX, cY) {
const margin = 15;
let [elementX, elementY] = Browser.transformClientToBrowser(cX, cY);
let aElement = Browser.elementFromPoint(elementX, elementY);
let bv = Browser._browserView;
2009-08-05 14:06:38 -07:00
let scroller = Browser.contentScrollboxScroller;
let elRect = Browser.getBoundingContentRect(aElement);
let elWidth = elRect.width;
let vis = bv.getVisibleRect();
let vrWidth = vis.width;
/* Try to set zoom-level such that once zoomed element is as wide
* as the visible rect */
let zoomLevel = BrowserView.Util.clampZoomLevel((vrWidth - (2 * margin)) / elWidth);
let oldZoomLevel = bv.getZoomLevel();
//dump("element width: " + elWidth + "\n");
//dump("old zoom level: " + oldZoomLevel + "\n");
//dump("new zoom level: " + zoomLevel + "\n");
/* If the new zoom level we've calculated is less than or equal to
* the current zoom level, return early and don't progress further.
* If the new zoom level is higher, continue to zoom inward.
*/
if (oldZoomLevel >= zoomLevel)
return false;
bv.beginBatchOperation();
/* XXX
* This isn't ideal.
*/
this.hideSidebars();
bv.setZoomLevel(zoomLevel);
bv.forceContainerResize();
//Browser.forceChromeReflow();
let dx = Math.round(bv.browserToViewport(elRect.left) - margin - vis.left);
let dy = Math.round(bv.browserToViewport(elementY) - (window.innerHeight / 2) - vis.top);
Browser.contentScrollbox.customDragger.dragMove(dx, dy, scroller);
bv.commitBatchOperation();
return true;
},
zoomFromPoint: function zoomFromPoint(cX, cY) {
let [elementX, elementY] = this.transformClientToBrowser(cX, cY);
let bv = Browser._browserView;
bv.beginBatchOperation();
bv.zoomToPage();
bv.forceContainerResize();
let dy = Math.round(bv.browserToViewport(elementY) - (window.innerHeight / 2) - bv.getVisibleRect().top);
this.contentScrollbox.customDragger.dragMove(0, dy, this.contentScrollboxScroller);
Browser.forceChromeReflow();
this.hideSidebars();
bv.commitBatchOperation();
},
getBoundingContentRect: function getBoundingContentRect(contentElem) {
let browser = Browser._browserView.getBrowser();
if (!browser)
return null;
2009-08-03 20:32:17 -07:00
let scrollX = { value: 0 };
let scrollY = { value: 0 };
let cwu = BrowserView.Util.getBrowserDOMWindowUtils(browser);
cwu.getScrollXY(false, scrollX, scrollY);
let r = contentElem.getBoundingClientRect();
//dump('getBoundingContentRect: clientRect is at ' + r.left + ', ' + r.top + '; scrolls are ' + scrollX.value + ', ' + scrollY.value + '\n');
return new wsRect(r.left + scrollX.value,
r.top + scrollY.value,
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);
},
/**
* 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 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 = window.innerHeight;
return new wsRect(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 [x.value, y.value];
},
forceChromeReflow: function forceChromeReflow() {
let dummy = getComputedStyle(document.documentElement, "").width;
}
};
Browser.MainDragger = function MainDragger(browserView) {
this.bv = browserView;
this.draggedFrame = null;
};
Browser.MainDragger.prototype = {
dragStart: function dragStart(clientX, clientY, target, scroller) {
let [x, y] = Browser.transformClientToBrowser(clientX, clientY);
let element = Browser.elementFromPoint(x, y);
this.draggedFrame = null;
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();
2009-08-05 13:28:43 -07:00
this.bv.resumeRendering();
},
dragMove: function dragMove(dx, dy, scroller) {
let elem = this.draggedFrame;
let doffset = [dx, dy];
let render = false;
this.bv.onBeforeVisibleMove(dx, dy);
// First calculate any panning to take sidebars out of view
let offsetX = this._panSidebarsAwayOffset(doffset);
// Do all iframe panning
if (elem) {
while (elem.frameElement && (doffset[0] != 0 || doffset[1] != 0)) {
this._panFrame(elem, doffset);
elem = elem.frameElement;
render = true;
}
}
// Do content panning
this._panScroller(Browser.contentScrollboxScroller, doffset);
// Any leftover panning in doffset would bring sidebars into view. Add to sidebar
// away panning for the total scroll offset.
this._panScroller(Browser.controlsScrollboxScroller, [doffset[0] + offsetX, 0]);
this.bv.onAfterVisibleMove();
Browser.tryFloatToolbar();
if (render)
this.bv.renderNow();
return doffset[0] + offsetX != dx || doffset[1] != dy;
},
/** Return X offset needed to pan sidebars away if possible. Updates doffset with leftovers. */
_panSidebarsAwayOffset: function(doffset) {
let [scrollLeft] = Browser.getScrollboxPosition(Browser.controlsScrollboxScroller);
let scrollRight = scrollLeft + window.innerWidth;
let leftLock = document.getElementById("tabs-container").getBoundingClientRect().right + scrollLeft;
let rightLock = document.getElementById("browser-controls").getBoundingClientRect().left + scrollLeft;
let amount = 0;
if (doffset[0] > 0 && scrollLeft < leftLock) {
amount = Math.min(doffset[0], leftLock - scrollLeft);
} else if (doffset[0] < 0 && scrollRight > rightLock) {
amount = Math.max(doffset[0], rightLock - scrollRight);
}
amount = Math.round(amount);
doffset[0] -= amount;
return amount;
},
/** Pan scroller by the given amount. Updates doffset with leftovers. */
_panScroller: function _panScroller(scroller, doffset) {
let [x0, y0] = Browser.getScrollboxPosition(scroller);
scroller.scrollBy(doffset[0], doffset[1]);
let [x1, y1] = Browser.getScrollboxPosition(scroller);
doffset[0] -= x1 - x0;
doffset[1] -= 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[0], doffset[1]);
windowUtils.getScrollXY(false, newX, newY);
doffset[0] -= newX.value - origX.value;
doffset[1] -= newY.value - origY.value;
}
};
2008-09-24 09:14:06 -07:00
function nsBrowserAccess()
{
}
nsBrowserAccess.prototype = {
QueryInterface: function(aIID) {
if (aIID.equals(Ci.nsIBrowserDOMWindow) || aIID.equals(Ci.nsISupports))
2008-09-24 09:14:06 -07:00
return this;
throw Components.results.NS_NOINTERFACE;
},
openURI: function(aURI, aOpener, aWhere, aContext) {
2008-09-24 09:14:06 -07:00
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");
2008-09-24 09:14:06 -07:00
return null;
}
2008-09-24 09:14:06 -07:00
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
2008-09-24 09:14:06 -07:00
newWindow = aOpener ? aOpener.top : browser.contentWindow;
}
2008-09-24 09:14:06 -07:00
try {
var referrer;
if (aURI) {
if (aOpener) {
location = aOpener.location;
referrer = Cc["@mozilla.org/network/io-service;1"].getService(Ci.nsIIOService)
.newURI(location, null, null);
2008-09-24 09:14:06 -07:00
}
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);
2008-09-24 09:14:06 -07:00
}
};
2008-09-24 09:14:06 -07:00
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":
2009-09-24 12:15:16 -07:00
// 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();
2009-09-24 12:15:16 -07:00
// fall through
case "engine-changed":
2009-09-24 12:15:16 -07:00
// 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)
2009-09-24 12:15:16 -07:00
// blow away our cache
this._engines = null;
break;
case "engine-current":
2009-09-24 12:15:16 -07:00
// 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<buttons.length; i++)
container.removeChild(buttons[i]);
if (newEngines.length == 0) {
container.hidden = true;
return;
}
// XXX limit to the first search engine for now
for (let i = 0; i<1; i++) {
let button = document.createElement("button");
button.className = "search-engine-button button-dark";
button.setAttribute("oncommand", "BrowserSearch.addPermanentSearchEngine(this.engine);this.parentNode.hidden=true;");
let engine = newEngines[i];
button.engine = engine.engine;
button.setAttribute("label", engine.engine.title);
button.setAttribute("image", BrowserUI._favicon.src);
container.appendChild(button);
}
container.hidden = false;
},
addPermanentSearchEngine: function (aEngine) {
let iconURL = BrowserUI._favicon.src;
this.searchService.addEngine(aEngine.href, Ci.nsISearchEngine.DATA_XML, iconURL, false);
this._engines = null;
},
updateSearchButtons: function() {
if (this._engines)
return;
// Clean the previous search engines button
var container = document.getElementById("search-buttons");
while (container.hasChildNodes())
container.removeChild(container.lastChild);
let engines = this.engines;
for (var e = 0; e < engines.length; e++) {
var button = document.createElement("radio");
var engine = engines[e];
button.id = engine.name;
button.setAttribute("label", engine.name);
button.className = "searchengine";
if (engine.iconURI)
button.setAttribute("src", engine.iconURI.spec);
container.appendChild(button);
button.engine = engine;
}
}
}
/** Watches for mouse events in chrome and sends them to content. */
function ContentCustomClicker(browserView) {
this._browserView = browserView;
this._fm = Cc["@mozilla.org/focus-manager;1"].getService(Ci.nsIFocusManager);
}
ContentCustomClicker.prototype = {
/** Dispatch a mouse event with chrome client coordinates. */
_dispatchMouseEvent: function _dispatchMouseEvent(name, cX, cY) {
let browser = this._browserView.getBrowser();
let [x, y] = Browser.transformClientToBrowser(cX, cY);
let cwu = BrowserView.Util.getBrowserDOMWindowUtils(browser);
let scrollX = {}, scrollY = {};
if (browser) {
cwu.getScrollXY(false, scrollX, scrollY);
cwu.sendMouseEvent(name, x - scrollX.value, y - scrollY.value, 0, 1, 0, true);
}
},
mouseDown: function mouseDown(cX, cY) {
// if something is targeted by the mousedown, focus it and re-render
// the canvas now
let [x, y] = Browser.transformClientToBrowser(cX, cY);
let element = Browser.elementFromPoint(x, y);
if (!element)
return;
this._fm.setFocus(element, Ci.nsIFocusManager.FLAG_NOSCROLL);
Util.executeSoon(this._browserView.renderNow);
},
mouseUp: function mouseUp(cX, cY) {
},
singleClick: function singleClick(cX, cY) {
this._dispatchMouseEvent("mousedown", cX, cY);
this._dispatchMouseEvent("mouseup", cX, cY);
},
doubleClick: function doubleClick(cX1, cY1, cX2, cY2) {
if (!Browser.zoomToPoint(cX2, cY2))
Browser.zoomFromPoint(cX2, cY2);
},
toString: function toString() {
return "[ContentCustomClicker] { }";
}
};
/**
* Utility class to handle manipulations of the identity indicators in the UI
*/
function IdentityHandler() {
this._stringBundle = document.getElementById("bundle_browser");
this._staticStrings = {};
this._staticStrings[this.IDENTITY_MODE_DOMAIN_VERIFIED] = {
encryption_label: this._stringBundle.getString("identity.encrypted2")
};
this._staticStrings[this.IDENTITY_MODE_IDENTIFIED] = {
encryption_label: this._stringBundle.getString("identity.encrypted2")
};
this._staticStrings[this.IDENTITY_MODE_UNKNOWN] = {
encryption_label: this._stringBundle.getString("identity.unencrypted2")
};
this._cacheElements();
}
IdentityHandler.prototype = {
// Mode strings used to control CSS display
IDENTITY_MODE_IDENTIFIED : "verifiedIdentity", // High-quality identity information
IDENTITY_MODE_DOMAIN_VERIFIED : "verifiedDomain", // Minimal SSL CA-signed domain verification
IDENTITY_MODE_UNKNOWN : "unknownIdentity", // No trusted identity information
// Cache the most recent SSLStatus and Location seen in checkIdentity
_lastStatus : null,
_lastLocation : null,
/**
* Build out a cache of the elements that we need frequently.
*/
_cacheElements: function() {
this._identityBox = document.getElementById("identity-box");
this._identityPopup = document.getElementById("identity-container");
this._identityPopupContentBox = document.getElementById("identity-popup-content-box");
this._identityPopupContentHost = document.getElementById("identity-popup-content-host");
this._identityPopupContentOwner = document.getElementById("identity-popup-content-owner");
this._identityPopupContentSupp = document.getElementById("identity-popup-content-supplemental");
this._identityPopupContentVerif = document.getElementById("identity-popup-content-verifier");
this._identityPopupEncLabel = document.getElementById("identity-popup-encryption-label");
},
/**
* Helper to parse out the important parts of _lastStatus (of the SSL cert in
* particular) for use in constructing identity UI strings
*/
getIdentityData: function() {
var result = {};
var status = this._lastStatus.QueryInterface(Ci.nsISSLStatus);
var cert = status.serverCert;
// Human readable name of Subject
result.subjectOrg = cert.organization;
// SubjectName fields, broken up for individual access
if (cert.subjectName) {
result.subjectNameFields = {};
cert.subjectName.split(",").forEach(function(v) {
var field = v.split("=");
this[field[0]] = field[1];
}, result.subjectNameFields);
// Call out city, state, and country specifically
result.city = result.subjectNameFields.L;
result.state = result.subjectNameFields.ST;
result.country = result.subjectNameFields.C;
}
// Human readable name of Certificate Authority
result.caOrg = cert.issuerOrganization || cert.issuerCommonName;
result.cert = cert;
return result;
},
/**
* Determine the identity of the page being displayed by examining its SSL cert
* (if available) and, if necessary, update the UI to reflect this.
*/
checkIdentity: function() {
let state = Browser.selectedTab.getIdentityState();
let location = getBrowser().contentWindow.location;
let currentStatus = getBrowser().securityUI.QueryInterface(Ci.nsISSLStatusProvider).SSLStatus;
this._lastStatus = currentStatus;
this._lastLocation = {};
try {
// make a copy of the passed in location to avoid cycles
this._lastLocation = { host: location.host, hostname: location.hostname, port: location.port };
} catch (ex) { }
if (state & Ci.nsIWebProgressListener.STATE_IDENTITY_EV_TOPLEVEL)
this.setMode(this.IDENTITY_MODE_IDENTIFIED);
else if (state & Ci.nsIWebProgressListener.STATE_SECURE_HIGH)
this.setMode(this.IDENTITY_MODE_DOMAIN_VERIFIED);
else
this.setMode(this.IDENTITY_MODE_UNKNOWN);
},
/**
* Return the eTLD+1 version of the current hostname
*/
getEffectiveHost: function() {
// Cache the eTLDService if this is our first time through
if (!this._eTLDService)
this._eTLDService = Cc["@mozilla.org/network/effective-tld-service;1"]
.getService(Ci.nsIEffectiveTLDService);
try {
return this._eTLDService.getBaseDomainFromHost(this._lastLocation.hostname);
} catch (e) {
// If something goes wrong (e.g. hostname is an IP address) just fail back
// to the full domain.
return this._lastLocation.hostname;
}
},
/**
* Update the UI to reflect the specified mode, which should be one of the
* IDENTITY_MODE_* constants.
*/
setMode: function(newMode) {
this._identityBox.setAttribute("mode", newMode);
this.setIdentityMessages(newMode);
// Update the popup too, if it's open
if (!this._identityPopup.hidden)
this.setPopupMessages(newMode);
},
/**
* Set up the messages for the primary identity UI based on the specified mode,
* and the details of the SSL cert, where applicable
*
* @param newMode The newly set identity mode. Should be one of the IDENTITY_MODE_* constants.
*/
setIdentityMessages: function(newMode) {
if (newMode == this.IDENTITY_MODE_DOMAIN_VERIFIED) {
var iData = this.getIdentityData();
// We need a port number for all lookups. If one hasn't been specified, use
// the https default
var lookupHost = this._lastLocation.host;
if (lookupHost.indexOf(':') < 0)
lookupHost += ":443";
// Cache the override service the first time we need to check it
if (!this._overrideService)
this._overrideService = Cc["@mozilla.org/security/certoverride;1"].getService(Ci.nsICertOverrideService);
// Verifier is either the CA Org, for a normal cert, or a special string
// for certs that are trusted because of a security exception.
var tooltip = this._stringBundle.getFormattedString("identity.identified.verifier",
[iData.caOrg]);
// Check whether this site is a security exception. XPConnect does the right
// thing here in terms of converting _lastLocation.port from string to int, but
// the overrideService doesn't like undefined ports, so make sure we have
// something in the default case (bug 432241).
if (this._overrideService.hasMatchingOverride(this._lastLocation.hostname,
(this._lastLocation.port || 443),
iData.cert, {}, {}))
tooltip = this._stringBundle.getString("identity.identified.verified_by_you");
}
else if (newMode == this.IDENTITY_MODE_IDENTIFIED) {
// If it's identified, then we can populate the dialog with credentials
iData = this.getIdentityData();
tooltip = this._stringBundle.getFormattedString("identity.identified.verifier",
[iData.caOrg]);
}
else {
tooltip = this._stringBundle.getString("identity.unknown.tooltip");
}
// Push the appropriate strings out to the UI
this._identityBox.tooltipText = tooltip;
},
/**
* Set up the title and content messages for the identity message popup,
* based on the specified mode, and the details of the SSL cert, where
* applicable
*
* @param newMode The newly set identity mode. Should be one of the IDENTITY_MODE_* constants.
*/
setPopupMessages: function(newMode) {
this._identityPopup.setAttribute("mode", newMode);
this._identityPopupContentBox.className = newMode;
// Set the static strings up front
this._identityPopupEncLabel.textContent = this._staticStrings[newMode].encryption_label;
// Initialize the optional strings to empty values
var supplemental = "";
var verifier = "";
if (newMode == this.IDENTITY_MODE_DOMAIN_VERIFIED) {
var iData = this.getIdentityData();
var host = this.getEffectiveHost();
var owner = this._stringBundle.getString("identity.ownerUnknown2");
verifier = this._identityBox.tooltipText;
supplemental = "";
}
else if (newMode == this.IDENTITY_MODE_IDENTIFIED) {
// If it's identified, then we can populate the dialog with credentials
iData = this.getIdentityData();
host = this.getEffectiveHost();
owner = iData.subjectOrg;
verifier = this._identityBox.tooltipText;
// Build an appropriate supplemental block out of whatever location data we have
if (iData.city)
supplemental += iData.city + "\n";
if (iData.state && iData.country)
supplemental += this._stringBundle.getFormattedString("identity.identified.state_and_country",
[iData.state, iData.country]);
else if (iData.state) // State only
supplemental += iData.state;
else if (iData.country) // Country only
supplemental += iData.country;
}
else {
// These strings will be hidden in CSS anyhow
host = "";
owner = "";
}
// Push the appropriate strings out to the UI
this._identityPopupContentHost.textContent = host;
this._identityPopupContentOwner.textContent = owner;
this._identityPopupContentSupp.textContent = supplemental;
this._identityPopupContentVerif.textContent = verifier;
// Update the search engines results
BrowserSearch.updatePageSearchEngines();
},
show: function ih_show() {
// dismiss any dialog which hide the identity popup
while (BrowserUI.activeDialog)
BrowserUI.activeDialog.close();
this._identityPopup.hidden = false;
this._identityPopup.top = BrowserUI.toolbarH;
this._identityPopup.focus();
this._identityBox.setAttribute("open", "true");
// Update the popup strings
this.setPopupMessages(this._identityBox.getAttribute("mode") || this.IDENTITY_MODE_UNKNOWN);
BrowserUI.pushPopup(this, [this._identityPopup, this._identityBox]);
BrowserUI.lockToolbar();
},
hide: function ih_hide() {
this._identityPopup.hidden = true;
this._identityBox.removeAttribute("open");
BrowserUI.unlockToolbar();
},
/**
* Click handler for the identity-box element in primary chrome.
*/
handleIdentityButtonEvent: function(event) {
event.stopPropagation();
if ((event.type == "click" && event.button != 0) ||
(event.type == "keypress" && event.charCode != KeyEvent.DOM_VK_SPACE &&
event.keyCode != KeyEvent.DOM_VK_RETURN))
return; // Left click, space or enter only
if (this._identityPopup.hidden)
this.show();
else
this.hide();
}
};
var gIdentityHandler;
/**
* Returns the singleton instance of the identity handler class. Should always be
* used instead of referencing the global variable directly or creating new instances
*/
function getIdentityHandler() {
if (!gIdentityHandler)
gIdentityHandler = new IdentityHandler();
return gIdentityHandler;
}
/**
* Handler for blocked popups, triggered by DOMUpdatePageReport events in browser.xml
*/
const gPopupBlockerObserver = {
_kIPM: Ci.nsIPermissionManager,
2009-08-07 14:59:06 -07:00
onUpdatePageReport: function onUpdatePageReport(aEvent)
{
var cBrowser = Browser.selectedBrowser;
if (aEvent.originalTarget != cBrowser)
return;
if (!cBrowser.pageReport)
return;
let pm = Cc["@mozilla.org/permissionmanager;1"].getService(Ci.nsIPermissionManager);
let result = pm.testExactPermission(Browser.selectedBrowser.currentURI, "popup");
if (result == Ci.nsIPermissionManager.DENY_ACTION)
return;
// Only show the notification again if we've not already shown it. Since
// notifications are per-browser, we don't need to worry about re-adding
// it.
if (!cBrowser.pageReport.reported) {
if(gPrefService.getBoolPref("privacy.popups.showBrowserMessage")) {
var bundle_browser = document.getElementById("bundle_browser");
var brandBundle = document.getElementById("bundle_brand");
var brandShortName = brandBundle.getString("brandShortName");
var message;
var popupCount = cBrowser.pageReport.length;
if (popupCount > 1)
message = bundle_browser.getFormattedString("popupWarningMultiple", [brandShortName, popupCount]);
else
message = bundle_browser.getFormattedString("popupWarning", [brandShortName]);
var notificationBox = Browser.getNotificationBox();
var notification = notificationBox.getNotificationWithValue("popup-blocked");
if (notification) {
notification.label = message;
}
else {
var buttons = [
{
label: bundle_browser.getString("popupButtonAllowOnce"),
accessKey: null,
callback: function() { gPopupBlockerObserver.showPopupsForSite(); }
},
{
label: bundle_browser.getString("popupButtonAlwaysAllow2"),
accessKey: null,
callback: function() { gPopupBlockerObserver.allowPopupsForSite(); }
},
{
label: bundle_browser.getString("popupButtonNeverWarn2"),
accessKey: null,
callback: function() { gPopupBlockerObserver.denyPopupsForSite(); }
}
];
const priority = notificationBox.PRIORITY_WARNING_MEDIUM;
notificationBox.appendNotification(message, "popup-blocked",
"",
priority, buttons);
}
}
// Record the fact that we've reported this blocked popup, so we don't
// show it again.
cBrowser.pageReport.reported = true;
}
},
allowPopupsForSite: function allowPopupsForSite(aEvent) {
var currentURI = Browser.selectedBrowser.currentURI;
var pm = Cc["@mozilla.org/permissionmanager;1"].getService(this._kIPM);
pm.add(currentURI, "popup", this._kIPM.ALLOW_ACTION);
Browser.getNotificationBox().removeCurrentNotification();
},
denyPopupsForSite: function denyPopupsForSite(aEvent) {
var currentURI = Browser.selectedBrowser.currentURI;
var pm = Cc["@mozilla.org/permissionmanager;1"].getService(this._kIPM);
pm.add(currentURI, "popup", this._kIPM.DENY_ACTION);
Browser.getNotificationBox().removeCurrentNotification();
},
showPopupsForSite: function showPopupsForSite() {
let uri = Browser.selectedBrowser.currentURI;
let pageReport = Browser.selectedBrowser.pageReport;
if (pageReport) {
for (let i = 0; i < pageReport.length; ++i) {
var popupURIspec = pageReport[i].popupWindowURI.spec;
// Sometimes the popup URI that we get back from the pageReport
// isn't useful (for instance, netscape.com's popup URI ends up
// being "http://www.netscape.com", which isn't really the URI of
// the popup they're trying to show). This isn't going to be
// useful to the user, so we won't create a menu item for it.
if (popupURIspec == "" || popupURIspec == "about:blank" ||
popupURIspec == uri.spec)
continue;
let popupFeatures = pageReport[i].popupWindowFeatures;
let popupName = pageReport[i].popupWindowName;
Browser.addTab(popupURIspec, false);
}
}
}
};
const gXPInstallObserver = {
2009-08-07 14:59:06 -07:00
observe: function xpi_observer(aSubject, aTopic, aData)
{
var brandBundle = document.getElementById("bundle_brand");
var browserBundle = document.getElementById("bundle_browser");
switch (aTopic) {
case "xpinstall-install-blocked":
var installInfo = aSubject.QueryInterface(Ci.nsIXPIInstallInfo);
var host = installInfo.originatingURI.host;
var brandShortName = brandBundle.getString("brandShortName");
var notificationName, messageString, buttons;
if (!gPrefService.getBoolPref("xpinstall.enabled")) {
notificationName = "xpinstall-disabled";
if (gPrefService.prefIsLocked("xpinstall.enabled")) {
messageString = browserBundle.getString("xpinstallDisabledMessageLocked");
buttons = [];
}
else {
messageString = browserBundle.getFormattedString("xpinstallDisabledMessage",
[brandShortName, host]);
buttons = [{
label: browserBundle.getString("xpinstallDisabledButton"),
accessKey: browserBundle.getString("xpinstallDisabledButton.accesskey"),
popup: null,
callback: function editPrefs() {
gPrefService.setBoolPref("xpinstall.enabled", true);
return false;
}
}];
}
}
else {
notificationName = "xpinstall";
messageString = browserBundle.getFormattedString("xpinstallPromptWarning",
[brandShortName, host]);
buttons = [{
label: browserBundle.getString("xpinstallPromptAllowButton"),
accessKey: browserBundle.getString("xpinstallPromptAllowButton.accesskey"),
popup: null,
callback: function() {
2009-03-19 20:36:02 -07:00
// Kick off the xpinstall
var mgr = Cc["@mozilla.org/xpinstall/install-manager;1"].createInstance(Ci.nsIXPInstallManager);
mgr.initManagerWithInstallInfo(installInfo);
return false;
}
}];
}
var nBox = Browser.getNotificationBox();
if (!nBox.getNotificationWithValue(notificationName)) {
const priority = nBox.PRIORITY_WARNING_MEDIUM;
const iconURL = "chrome://mozapps/skin/update/update.png";
nBox.appendNotification(messageString, notificationName, iconURL, priority, buttons);
}
break;
}
}
};
const gSessionHistoryObserver = {
observe: function sho_observe(subject, topic, data) {
if (topic != "browser:purge-session-history")
return;
let back = document.getElementById("cmd_back");
back.setAttribute("disabled", "true");
let forward = document.getElementById("cmd_forward");
forward.setAttribute("disabled", "true");
let urlbar = document.getElementById("urlbar-edit");
if (urlbar) {
// Clear undo history of the URL bar
urlbar.editor.transactionManager.clear();
}
}
};
var MemoryObserver = {
observe: function() {
let memory = Cc["@mozilla.org/xpcom/memory-service;1"].getService(Ci.nsIMemory);
do {
Browser.windowUtils.garbageCollect();
} while (memory.isLowMemory() && Browser.sacrificeTab());
}
};
#ifdef WINCE
// Windows Mobile does not resize the window automatically when the soft
// keyboard is displayed. Maemo does resize the window.
var SoftKeyboardObserver = {
observe: function sko_observe(subject, topic, data) {
if (topic === "softkb-change") {
// The rect passed to us is the space available to our window, so
// let's use it to resize the main window
let rect = JSON.parse(data);
if (rect) {
let height = rect.bottom - rect.top;
let width = rect.right - rect.left;
let popup = document.getElementById("popup_autocomplete");
popup.height = height - BrowserUI.toolbarH;
popup.width = width;
}
}
}
};
#endif
function getNotificationBox(aWindow) {
return Browser.getNotificationBox();
}
function importDialog(src, arguments) {
// load the dialog with a synchronous XHR
let xhr = Cc["@mozilla.org/xmlextras/xmlhttprequest;1"].createInstance();
xhr.open("GET", src, false);
xhr.overrideMimeType("text/xml");
xhr.send(null);
if (!xhr.responseXML)
return null;
let doc = xhr.responseXML.documentElement;
var dialog = null;
// we need to insert before select-container if we want it to show correctly
let selectContainer = document.getElementById("select-container");
let parent = selectContainer.parentNode;
// create a full-screen semi-opaque box as a background
let back = document.createElement("box");
back.setAttribute("class", "modal-block");
dialog = back.appendChild(document.importNode(doc, true));
parent.insertBefore(back, selectContainer);
dialog.arguments = arguments;
return dialog;
}
function showDownloadsManager(aWindowContext, aID, aReason) {
BrowserUI.show(UIMODE_PANEL);
BrowserUI.switchPane("downloads-container");
// TODO: select the download with aID
}
var AlertsHelper = {
_timeoutID: -1,
_listener: null,
_cookie: "",
_clickable: false,
showAlertNotification: function ah_show(aImageURL, aTitle, aText, aTextClickable, aCookie, aListener) {
this._clickable = aTextClickable || false;
this._listener = aListener || null;
this._cookie = aCookie || "";
document.getElementById("alerts-image").setAttribute("src", aImageURL);
document.getElementById("alerts-title").value = aTitle;
document.getElementById("alerts-text").textContent = aText;
let container = document.getElementById("alerts-container");
container.hidden = false;
let rect = container.getBoundingClientRect();
container.top = window.innerHeight - (rect.height + 20);
container.left = window.innerWidth - (rect.width + 20);
let timeout = gPrefService.getIntPref("alerts.totalOpenTime");
let self = this;
this._timeoutID = setTimeout(function() { self._timeoutAlert(); }, timeout);
},
_timeoutAlert: function ah__timeoutAlert() {
this._timeoutID = -1;
let container = document.getElementById("alerts-container");
container.hidden = true;
if (this._listener)
this._listener.observe(null, "alertfinished", this._cookie);
// TODO: add slide to UI
},
click: function ah_click(aEvent) {
if (this._clickable && this._listener)
this._listener.observe(null, "alertclickcallback", this._cookie);
if (this._timeoutID != -1) {
clearTimeout(this._timeoutID);
this._timeoutAlert();
}
}
};
var HelperAppDialog = {
_launcher: null,
show: function had_show(aLauncher) {
this._launcher = aLauncher;
document.getElementById("helperapp-target").value = this._launcher.suggestedFileName;
if (!this._launcher.MIMEInfo.hasDefaultHandler)
document.getElementById("helperapp-open").disabled = true;
let toolbar = document.getElementById("toolbar-main");
let top = toolbar.top + toolbar.boxObject.height;
let container = document.getElementById("helperapp-container");
container.hidden = false;
let rect = container.getBoundingClientRect();
container.top = top < 0 ? 0 : top;
container.left = (window.innerWidth - rect.width) / 2;
},
save: function had_save() {
this._launcher.saveToDisk(null, false);
this.close();
},
open: function had_open() {
this._launcher.launchWithApplication(null, false);
this.close();
},
close: function had_close() {
document.getElementById("helperapp-target").value = "";
let container = document.getElementById("helperapp-container");
container.hidden = true;
}
};
function ProgressController(tab) {
this._tab = tab;
// Properties used to cache security state used to update the UI
this.state = null;
this._hostChanged = false; // onLocationChange will flip this bit
}
ProgressController.prototype = {
get browser() {
return this._tab.browser;
},
2009-08-03 20:32:17 -07:00
onStateChange: function onStateChange(aWebProgress, aRequest, aStateFlags, aStatus) {
if (aStateFlags & Ci.nsIWebProgressListener.STATE_IS_NETWORK) {
if (aStateFlags & Ci.nsIWebProgressListener.STATE_START)
this._networkStart();
else if (aStateFlags & Ci.nsIWebProgressListener.STATE_STOP)
this._networkStop();
} else if (aStateFlags & Ci.nsIWebProgressListener.STATE_IS_DOCUMENT) {
if (aStateFlags & Ci.nsIWebProgressListener.STATE_STOP)
this._documentStop();
}
},
/** This method is called to indicate progress changes for the currently loading page. */
onProgressChange: function(aWebProgress, aRequest, aCurSelf, aMaxSelf, aCurTotal, aMaxTotal) {
},
/** This method is called to indicate a change to the current location. */
onLocationChange: function(aWebProgress, aRequest, aLocationURI) {
let location = aLocationURI ? aLocationURI.spec : "";
let selectedBrowser = Browser.selectedBrowser;
this._hostChanged = true;
if (this._tab == Browser.selectedTab) {
BrowserUI.updateURI();
// We're about to have new page content, to scroll the content area
// to the top so the new paints will draw correctly.
Browser.scrollContentToTop();
}
},
/**
* This method is called to indicate a status changes for the currently
* loading page. The message is already formatted for display.
*/
onStatusChange: function(aWebProgress, aRequest, aStatus, aMessage) {
},
/** This method is called when the security state of the browser changes. */
onSecurityChange: function(aWebProgress, aRequest, aState) {
// Don't need to do anything if the data we use to update the UI hasn't changed
if (this.state == aState && !this._hostChanged)
return;
this._hostChanged = false;
this.state = aState;
if (this._tab == Browser.selectedTab) {
getIdentityHandler().checkIdentity();
}
},
QueryInterface: function(aIID) {
if (aIID.equals(Ci.nsIWebProgressListener) ||
aIID.equals(Ci.nsISupportsWeakReference) ||
aIID.equals(Ci.nsISupports))
return this;
throw Components.results.NS_ERROR_NO_INTERFACE;
},
2009-08-03 20:32:17 -07:00
_networkStart: function _networkStart() {
this._tab.startLoading();
if (this._tab == Browser.selectedTab)
BrowserUI.update(TOOLBARSTATE_LOADING);
// broadcast a URLChanged message for consumption by InputHandler
let event = document.createEvent("Events");
event.initEvent("URLChanged", true, false);
this.browser.dispatchEvent(event);
},
2009-07-22 06:12:02 -07:00
_networkStop: function _networkStop() {
this._tab.endLoading();
if (this._tab == Browser.selectedTab) {
BrowserUI.update(TOOLBARSTATE_LOADED);
this.browser.docShell.isOffScreenBrowser = true;
}
this._tab.updateThumbnail();
},
_documentStop: function() {
// translate any phone numbers
Browser.translatePhoneNumbers();
if (this._tab == Browser.selectedTab && !BrowserUI.isAutoCompleteOpen()) {
// focus the dom window
if (this.browser.currentURI.spec != "about:blank")
this.browser.contentWindow.focus();
}
}
};
function Tab() {
this._id = null;
this._browser = null;
this._browserViewportState = null;
this._state = null;
this._listener = null;
this._loading = false;
this._chromeTab = null;
// Set to 0 since new tabs that have not been viewed yet are good tabs to
// toss if app needs more memory.
this.lastSelected = 0;
2009-07-17 16:17:57 -07:00
this.create();
}
Tab.prototype = {
get browser() {
return this._browser;
},
get browserViewportState() {
return this._browserViewportState;
},
get chromeTab() {
return this._chromeTab;
},
/**
* Throttles redraws to once every second while loading the page, zooming to fit page if
* user hasn't started zooming.
*/
_resizeAndPaint: function() {
let bv = Browser._browserView;
if (this == Browser.selectedTab) {
// !!! --- RESIZE HACK BEGIN -----
bv.simulateMozAfterSizeChange();
// !!! --- RESIZE HACK END -----
let restoringPage = (this._state != null);
if (!this._browserViewportState.zoomChanged && !restoringPage) {
// Only fit page if user hasn't started zooming around and this is a page that
// isn't being restored.
bv.zoomToPage();
}
}
bv.commitBatchOperation();
if (this._loading) {
// kick ourselves off 2s later while we're still loading
2009-08-13 12:34:03 -07:00
bv.beginBatchOperation();
this._loadingTimeout = setTimeout(Util.bind(this._resizeAndPaint, this), 2000);
} else {
delete this._loadingTimeout;
}
},
/** Returns tab's identity state for updating security UI. */
getIdentityState: function() {
return this._listener.state;
},
startLoading: function() {
//if (this._loading)
// dump("!!! Already loading this tab, please file a bug\n");
this._loading = true;
this._browserViewportState.zoomChanged = false;
if (!this._loadingTimeout) {
if (this == Browser.selectedTab) {
Browser._browserView.beginBatchOperation();
}
this._loadingTimeout = setTimeout(Util.bind(this._resizeAndPaint, this), 2000);
}
},
endLoading: function() {
//if (!this._loading)
// dump("!!! Already finished loading this tab, please file a bug\n");
this._loading = false;
clearTimeout(this._loadingTimeout);
// in order to ensure we commit our current batch,
// we need to run this function here
this._resizeAndPaint();
// if this tab was sacrificed previously, restore its state
this.restoreState();
},
isLoading: function() {
return this._loading;
},
2009-07-17 16:17:57 -07:00
load: function(uri) {
this._browser.setAttribute("src", uri);
},
create: function() {
// Initialize a viewport state for BrowserView
this._browserViewportState = BrowserView.Util.createBrowserViewportState();
this._chromeTab = document.getElementById("tabs").addTab();
2009-07-17 16:17:57 -07:00
this._createBrowser();
},
destroy: function() {
this._destroyBrowser();
document.getElementById("tabs").removeTab(this._chromeTab);
this._chromeTab = null;
},
/** Create browser if it doesn't already exist. */
ensureBrowserExists: function() {
if (!this._browser) {
this._createBrowser();
this.browser.contentDocument.location = this._state._url;
}
},
2009-07-17 16:17:57 -07:00
_createBrowser: function() {
if (this._browser)
throw "Browser already exists";
// Create the browser using the current width the dynamically size the height
let scaledHeight = kDefaultBrowserWidth * (window.innerHeight / window.innerWidth);
let browser = this._browser = document.createElement("browser");
2009-07-17 16:17:57 -07:00
2009-07-22 06:12:02 -07:00
browser.setAttribute("style", "overflow: -moz-hidden-unscrollable; visibility: hidden; width: " + kDefaultBrowserWidth + "px; height: " + scaledHeight + "px;");
browser.setAttribute("type", "content");
// Append the browser to the document, which should start the page load
document.getElementById("browsers").appendChild(browser);
2009-07-22 06:12:02 -07:00
// stop about:blank from loading
browser.stop();
// Attach a separate progress listener to the browser
this._listener = new ProgressController(this);
browser.addProgressListener(this._listener);
},
_destroyBrowser: function() {
if (this._browser) {
document.getElementById("browsers").removeChild(this._browser);
this._browser = null;
}
},
/** Serializes as much state as possible of the current content. */
saveState: function() {
let state = { };
var browser = this._browser;
var doc = browser.contentDocument;
state._url = doc.location.href;
state._scroll = BrowserView.Util.getContentScrollValues(this.browser);
if (doc instanceof HTMLDocument) {
var tags = ["input", "textarea", "select"];
for (var t = 0; t < tags.length; t++) {
var elements = doc.getElementsByTagName(tags[t]);
for (var e = 0; e < elements.length; e++) {
var element = elements[e];
var id;
if (element.id)
id = "#" + element.id;
else if (element.name)
id = "$" + element.name;
if (id)
state[id] = element.value;
}
}
}
this._state = state;
},
/** Restores serialized content from saveState. */
restoreState: function() {
let state = this._state;
if (!state)
return;
let doc = this._browser.contentDocument;
for (var item in state) {
var elem = null;
if (item.charAt(0) == "#") {
elem = doc.getElementById(item.substring(1));
} else if (item.charAt(0) == "$") {
var list = doc.getElementsByName(item.substring(1));
if (list.length)
elem = list[0];
}
if (elem)
elem.value = state[item];
}
this.browser.contentWindow.scrollX = state._scroll[0];
this.browser.contentWindow.scrollY = state._scroll[1];
this._state = null;
},
updateThumbnail: function() {
if (!this._browser)
return;
let browserView = (Browser.selectedBrowser == this._browser) ? Browser._browserView : null;
this._chromeTab.updateThumbnail(this._browser, browserView);
},
toString: function() {
return "[Tab " + (this._browser ? this._browser.contentDocument.location.toString() : "(no browser)") + "]";
}
};