mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
2441 lines
79 KiB
JavaScript
2441 lines
79 KiB
JavaScript
// -*- 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");
|
|
|
|
function getBrowser() {
|
|
return Browser.selectedBrowser;
|
|
}
|
|
|
|
const kDefaultTextZoom = 1.2;
|
|
const kDefaultBrowserWidth = 1024;
|
|
|
|
function debug() {
|
|
let bv = Browser._browserView;
|
|
let tc = bv._tileManager._tileCache;
|
|
let scrollbox = document.getElementById("tile-container-container")
|
|
.boxObject
|
|
.QueryInterface(Components.interfaces.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("tile-container-container");
|
|
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;
|
|
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 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:
|
|
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 : [],
|
|
_browsers : [],
|
|
_selectedTab : null,
|
|
windowUtils: window.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
.getInterface(Ci.nsIDOMWindowUtils),
|
|
contentScrollbox: null,
|
|
contentScrollboxScroller: null,
|
|
controlsScrollbox: null,
|
|
controlsScrollboxScroller: null,
|
|
|
|
startup: function() {
|
|
var self = this;
|
|
|
|
dump("begin startup\n");
|
|
|
|
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 = this._createContentCustomClicker(bv);
|
|
|
|
/* vertically scrolling box that contains tiles and the urlbar */
|
|
let contentScrollbox = this.contentScrollbox = document.getElementById("tile-container-container");
|
|
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("scrollbox");
|
|
this.controlsScrollboxScroller = controlsScrollbox.boxObject.QueryInterface(Ci.nsIScrollBoxObject);
|
|
controlsScrollbox.customDragger = {
|
|
allowRealtimeDownUp: true,
|
|
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
|
|
bv.beginBatchOperation();
|
|
|
|
function resizeHandler(e) {
|
|
|
|
if (e.target != window)
|
|
return;
|
|
|
|
dump(window.innerWidth + "," + window.innerHeight + "\n");
|
|
// 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();
|
|
|
|
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");
|
|
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 scaledH = (kDefaultBrowserWidth * (h / w));
|
|
for (let i=0; i<browsers.length; i++) {
|
|
let browserStyle = browsers[i].style;
|
|
browserStyle.height = scaledH + 'px';
|
|
}
|
|
}
|
|
|
|
Browser.hideSidebars();
|
|
|
|
bv.onAfterVisibleMove();
|
|
bv.zoomToPage();
|
|
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);
|
|
|
|
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
|
|
|
|
// XXX hook up memory-pressure notification to clear out tab browsers
|
|
//os.addObserver(function(subject, topic, data) self.destroyEarliestBrowser(), "memory-pressure", false);
|
|
|
|
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;
|
|
|
|
// 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 */
|
|
// 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]) {
|
|
var whereURI = null;
|
|
|
|
try {
|
|
// Try to access the commandline
|
|
var cmdLine = window.arguments[0].QueryInterface(Ci.nsICommandLine);
|
|
|
|
try {
|
|
// Check for and use a default homepage
|
|
whereURI = gPrefService.getCharPref("browser.startup.homepage");
|
|
} catch (e) {}
|
|
|
|
// 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) {}
|
|
|
|
if (whereURI)
|
|
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);
|
|
}
|
|
|
|
bv.commitBatchOperation();
|
|
|
|
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");
|
|
#ifdef WINCE
|
|
os.removeObserver(SoftKeyboardObserver, "softkb-change");
|
|
#endif
|
|
|
|
window.controllers.removeController(this);
|
|
window.controllers.removeController(BrowserUI);
|
|
},
|
|
|
|
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._browsers;
|
|
},
|
|
|
|
scrollContentToTop: function scrollContentToTop() {
|
|
this.contentScrollboxScroller.scrollTo(0, 0);
|
|
this._browserView.onAfterVisibleMove();
|
|
},
|
|
|
|
hideSidebars: function scrollSidebarsOffscreen() {
|
|
let container = document.getElementById("tile-container");
|
|
let containerBCR = container.getBoundingClientRect();
|
|
|
|
let dx = containerBCR.left;
|
|
if (dx < 0)
|
|
dx = Math.min(containerBCR.right - window.innerWidth, 0);
|
|
|
|
this.controlsScrollboxScroller.scrollBy(Math.round(dx), 0);
|
|
|
|
Browser.contentScrollbox.customDragger.scrollingOuterX = false; // XXX ugh.
|
|
|
|
this._browserView.onAfterVisibleMove();
|
|
},
|
|
|
|
/**
|
|
* 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);
|
|
this._browsers.push(newTab.browser);
|
|
|
|
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);
|
|
this._browsers.splice(tabIndex, 1);
|
|
|
|
// redraw the tabs
|
|
for (let t = tabIndex; t < this._tabs.length; t++)
|
|
this._tabs[t].updateThumbnail();
|
|
},
|
|
|
|
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;
|
|
|
|
let firstTab = this._selectedTab == null;
|
|
this._selectedTab = tab;
|
|
|
|
bv.beginBatchOperation();
|
|
|
|
bv.setBrowser(tab.browser, tab.browserViewportState, false);
|
|
bv.forceContainerResize();
|
|
|
|
// XXX these should probably be computed less hackily so they don't
|
|
// potentially break if we change something in browser.xul
|
|
let offY = Math.round(document.getElementById("toolbar-container").getBoundingClientRect().height);
|
|
let restoreX = Math.max(0, tab.browserViewportState.visibleX);
|
|
let restoreY = Math.max(0, tab.browserViewportState.visibleY) + offY;
|
|
|
|
dump('Switch tab scrolls to: ' + restoreX
|
|
+ ', ' + restoreY + '\n');
|
|
|
|
Browser.contentScrollboxScroller.scrollTo(restoreX, restoreY);
|
|
|
|
document.getElementById("tabs").selectedItem = tab.chromeTab;
|
|
|
|
if (!firstTab) {
|
|
let webProgress = this.selectedBrowser.webProgress;
|
|
let securityUI = this.selectedBrowser.securityUI;
|
|
|
|
try {
|
|
tab._listener.onLocationChange(webProgress, null, tab.browser.currentURI);
|
|
if (securityUI)
|
|
tab._listener.onSecurityChange(webProgress, null, securityUI.state);
|
|
} catch (e) {
|
|
// don't inhibit other listeners or following code
|
|
Components.utils.reportError(e);
|
|
}
|
|
|
|
let event = document.createEvent("Events");
|
|
event.initEvent("TabSelect", true, false);
|
|
tab.chromeTab.dispatchEvent(event);
|
|
}
|
|
|
|
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);
|
|
},
|
|
|
|
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})/;
|
|
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
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 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:neterror\?e=nssBadCert/.test(errorDoc.documentURI)) {
|
|
if (ot == errorDoc.getElementById('exceptionDialogButton')) {
|
|
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);
|
|
}
|
|
}
|
|
},
|
|
|
|
_createContentCustomClicker: function _createContentCustomClicker(browserView) {
|
|
// XXX we probably shouldn't generate this dynamically like this, but
|
|
// actually make it a prototype somewhere and instantiate it and such...
|
|
|
|
function dispatchContentClick(browser, x, y) {
|
|
let cwu = BrowserView.Util.getBrowserDOMWindowUtils(browser);
|
|
let scrollX = { value: 0 };
|
|
let scrollY = { value: 0 };
|
|
cwu.getScrollXY(false, scrollX, scrollY);
|
|
cwu.sendMouseEvent("mousedown", x - scrollX.value, y - scrollY.value, 0, 1, 0, true);
|
|
cwu.getScrollXY(false, scrollX, scrollY);
|
|
cwu.sendMouseEvent("mouseup", x - scrollX.value, y - scrollY.value, 0, 1, 0, true);
|
|
}
|
|
|
|
return {
|
|
singleClick: function singleClick(cX, cY) {
|
|
let browser = browserView.getBrowser();
|
|
if (browser) {
|
|
let [x, y] = Browser.transformClientToBrowser(cX, cY);
|
|
dispatchContentClick(browser, x, y);
|
|
}
|
|
},
|
|
|
|
doubleClick: function doubleClick(cX1, cY1, cX2, cY2) {
|
|
if (!Browser.zoomToPoint(cX2, cY2))
|
|
Browser.zoomFromPoint(cX2, cY2);
|
|
},
|
|
|
|
toString: function toString() {
|
|
return "[ContentCustomClicker] { }";
|
|
}
|
|
};
|
|
},
|
|
|
|
/**
|
|
* 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;
|
|
|
|
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) {
|
|
document.getElementById("toolbar-moveable-container").top = 0;
|
|
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.002 && ritevis <= 0.002) {
|
|
document.getElementById("toolbar-moveable-container").top = "";
|
|
this.floatedWhileDragging = false;
|
|
return true;
|
|
}
|
|
return false;
|
|
},
|
|
|
|
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;
|
|
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);
|
|
|
|
/*
|
|
* Animation code for zooming to zoom to the correct place
|
|
*
|
|
let callback = {
|
|
scroller: scroller,
|
|
sx: (dx > 0) ? 1 : -1,
|
|
sy: (dy > 0) ? 1 : -1,
|
|
dx: dx,
|
|
dy: dy,
|
|
notify: function kineticTimerCallback(timer) {
|
|
if (this.dx == 0 && this.dy == 0) {
|
|
timer.cancel();
|
|
return;
|
|
}
|
|
Browser.contentScrollbox.customDragger.dragMove(this.sx, this.sy, this.scroller);
|
|
|
|
if (this.dx != 0) this.dx -= this.sx;
|
|
if (this.dy != 0) this.dy -= this.sy;
|
|
}
|
|
};
|
|
|
|
this._zoomTimer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
|
|
//initialize our timer with updateInterval
|
|
this._zoomTimer.initWithCallback(callback,
|
|
2,
|
|
this._zoomTimer.TYPE_REPEATING_PRECISE);
|
|
*/
|
|
|
|
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;
|
|
|
|
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);
|
|
},
|
|
|
|
/**
|
|
* return element at client coordinates of browser, returns null if
|
|
* there's no active browser or if no element can be found
|
|
*/
|
|
elementFromPoint: function elementFromPoint(x, y) {
|
|
Util.dumpLn("*** elementFromPoint: page ", x, ",", y);
|
|
|
|
let browser = this._browserView.getBrowser();
|
|
if (!browser) return null;
|
|
|
|
let cwu = BrowserView.Util.getBrowserDOMWindowUtils(browser);
|
|
|
|
let scrollX = { value: 0 }, scrollY = { value: 0 };
|
|
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)) {
|
|
let frameWin = elem.ownerDocument.defaultView;
|
|
let frameUtils = frameWin.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindowUtils);
|
|
frameUtils.getScrollXY(false, scrollX, scrollY);
|
|
|
|
x = x - elem.offsetLeft + scrollX.value;
|
|
y = y - elem.offsetTop + scrollY.value;
|
|
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.allowRealtimeDownUp = true;
|
|
|
|
this.scrollingOuterX = false;
|
|
this.bv = browserView;
|
|
this.floatedWhileDragging = false;
|
|
this.draggedFrame = null;
|
|
};
|
|
|
|
Browser.MainDragger.prototype = {
|
|
|
|
dragStart: function dragStart(clientX, clientY, target, scroller) {
|
|
this.draggedFrame = null;
|
|
|
|
if (this._targetIsContent(target)) {
|
|
// since we're dealing with content, look to see if user has started
|
|
// a drag while over a IFRAME/FRAME element
|
|
let [x, y] = Browser.transformClientToBrowser(clientX, clientY);
|
|
let element = Browser.elementFromPoint(x, y);
|
|
if (element && element.ownerDocument != Browser.selectedBrowser.contentDocument) {
|
|
Util.dumpLn("*** dragStart got element ", element, " ownerDoc ", element.ownerDocument,
|
|
" selectedBrowser.contentDoc ", Browser.selectedBrowser.contentDocument);
|
|
this.draggedFrame = element.ownerDocument.defaultView;
|
|
}
|
|
}
|
|
|
|
this.bv.pauseRendering();
|
|
},
|
|
|
|
dragStop: function dragStop(dx, dy, scroller) {
|
|
let dx = this.dragMove(dx, dy, scroller, true);
|
|
|
|
dx += this.dragMove(Browser.snapSidebars(), 0, scroller, true);
|
|
|
|
/* XXX
|
|
* Set scrollingOuterX to be true when the sidebars are open.
|
|
* We should really just take a look at our geometry at this point
|
|
* and determine this ourselves rather than relying on our helper
|
|
* functions to give us this information
|
|
*/
|
|
this.scrollingOuterX = !Browser.tryUnfloatToolbar();
|
|
|
|
this.bv.resumeRendering();
|
|
|
|
return (dx != 0) || (dy != 0);
|
|
},
|
|
|
|
dragMove: function dragMove(dx, dy, scroller, doReturnDX) {
|
|
let outrv = 0;
|
|
|
|
if (this._panFrame(dx, dy)) // first see if we need to adjust internal IFRAME/FRAME
|
|
return true;
|
|
|
|
//dump("enter drag move\n");
|
|
if (this.scrollingOuterX) {
|
|
let odx = 0;
|
|
let ody = 0;
|
|
|
|
let snappedX = false;
|
|
if (dx > 0) {
|
|
let contentleft = Math.round(Browser.contentScrollbox.getBoundingClientRect().left);
|
|
odx = (contentleft > 0) ? Math.min(contentleft, dx) : dx;
|
|
snappedX = (contentleft > 0 && odx == contentleft);
|
|
} else if (dx < 0) {
|
|
let contentright = Math.round(Browser.contentScrollbox.getBoundingClientRect().right);
|
|
let w = window.innerWidth;
|
|
odx = (contentright < w) ? Math.max(contentright - w, dx) : dx;
|
|
snappedX = (contentright < w && odx == (contentright - w));
|
|
}
|
|
|
|
//dump(" odx, ody, snappedX: " + odx + " " + ody + " " + snappedX + "\n");
|
|
|
|
if (odx) {
|
|
outrv = this.outerDragMove(odx, ody, Browser.controlsScrollboxScroller, doReturnDX);
|
|
}
|
|
|
|
if (snappedX || ody != dy) {
|
|
if (snappedX)
|
|
this.scrollingOuterX = false;
|
|
|
|
dx -= odx;
|
|
dy -= ody;
|
|
} else {
|
|
return outrv;
|
|
}
|
|
}
|
|
//dump(" scrolling inner x\n");
|
|
|
|
this.bv.onBeforeVisibleMove(dx, dy);
|
|
|
|
let [x0, y0] = Browser.getScrollboxPosition(scroller);
|
|
scroller.scrollBy(dx, dy);
|
|
let [x1, y1] = Browser.getScrollboxPosition(scroller);
|
|
|
|
let realdx = x1 - x0;
|
|
let realdy = y1 - y0;
|
|
|
|
this.bv.onAfterVisibleMove();
|
|
|
|
//dump(" dx, realdx: " + dx + " " + realdx + "\n");
|
|
|
|
if (realdx != dx) {
|
|
let restdx = dx - realdx;
|
|
|
|
this.scrollingOuterX = true;
|
|
this.dragMove(restdx, 0, scroller, doReturnDX);
|
|
}
|
|
|
|
return (doReturnDX) ? (outrv + realdx) : (outrv || realdx != 0 || realdy != 0);
|
|
},
|
|
|
|
outerDragMove: function outerDragMove(dx, dy, scroller, doReturnDX) {
|
|
this.bv.onBeforeVisibleMove(dx, dy);
|
|
|
|
Browser.tryFloatToolbar(dx, dy);
|
|
|
|
let [x0, y0] = Browser.getScrollboxPosition(scroller);
|
|
scroller.scrollBy(dx, dy);
|
|
let [x1, y1] = Browser.getScrollboxPosition(scroller);
|
|
|
|
let realdx = x1 - x0;
|
|
let realdy = y1 - y0;
|
|
|
|
this.bv.onAfterVisibleMove();
|
|
|
|
return (doReturnDX) ? realdx : (realdx != 0 || realdy != 0);
|
|
},
|
|
|
|
_panFrame: function _panFrame(dx, dy) {
|
|
if (this.draggedFrame === null)
|
|
return false;
|
|
|
|
if (dx == 0 && dy == 0)
|
|
return true;
|
|
|
|
let panned = false;
|
|
let elem = this.draggedFrame;
|
|
|
|
// top-level window will have itself as its parent, so stop
|
|
// there to allow canvasbrowser/widgetstack to pan instead
|
|
// of doing scrolling
|
|
while (elem && elem !== elem.parent.document.defaultView) {
|
|
let windowUtils = elem.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindowUtils);
|
|
|
|
let origX = {}, origY = {};
|
|
windowUtils.getScrollXY(false, origX, origY);
|
|
|
|
elem.scrollBy(dx, dy);
|
|
|
|
let newX = {}, newY = {};
|
|
windowUtils.getScrollXY(false, newX, newY);
|
|
|
|
panned = (origX.value != newX.value) || (origY.value != newY.value);
|
|
|
|
if (panned) {
|
|
// get critical area to redraw after we move frame
|
|
// NOTE: may need to rate limit these for performance
|
|
this.bv.renderNow();
|
|
break;
|
|
}
|
|
|
|
elem = elem.parent.document.defaultView;
|
|
}
|
|
|
|
return panned;
|
|
},
|
|
|
|
_targetIsContent: function _targetIsContent(target) {
|
|
let tileBox = document.getElementById("tile-container");
|
|
while (target) {
|
|
if (target === window)
|
|
return false;
|
|
if (target === tileBox)
|
|
return true;
|
|
|
|
target = target.parentNode;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
};
|
|
|
|
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 = BrowserUI.newTab().browser.contentWindow;
|
|
else
|
|
newWindow = aOpener ? aOpener.top : browser.contentWindow;
|
|
}
|
|
|
|
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);
|
|
}
|
|
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);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 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. Intended to
|
|
* be called by onSecurityChange
|
|
*
|
|
* @param PRUint32 state
|
|
* @param JS Object location that mirrors an nsLocation (i.e. has .host and
|
|
* .hostname and .port)
|
|
*/
|
|
checkIdentity: function(state, location) {
|
|
var currentStatus = getBrowser().securityUI
|
|
.QueryInterface(Ci.nsISSLStatusProvider)
|
|
.SSLStatus;
|
|
this._lastStatus = currentStatus;
|
|
this._lastLocation = location;
|
|
|
|
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;
|
|
},
|
|
|
|
show: function ih_show() {
|
|
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]);
|
|
},
|
|
|
|
hide: function ih_hide() {
|
|
this._identityPopup.hidden = true;
|
|
|
|
this._identityBox.removeAttribute("open");
|
|
|
|
BrowserUI.popPopup();
|
|
},
|
|
|
|
/**
|
|
* 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,
|
|
|
|
onUpdatePageReport: function onUpdatePageReport(aEvent)
|
|
{
|
|
var cBrowser = Browser.selectedBrowser;
|
|
if (aEvent.originalTarget != cBrowser)
|
|
return;
|
|
|
|
if (!cBrowser.pageReport)
|
|
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.toggleAllowPopupsForSite(); }
|
|
},
|
|
{
|
|
label: bundle_browser.getString("popupButtonNeverWarn2"),
|
|
accessKey: null,
|
|
callback: function() { gPopupBlockerObserver.dontShowMessage(); }
|
|
}
|
|
];
|
|
|
|
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;
|
|
}
|
|
},
|
|
|
|
toggleAllowPopupsForSite: function toggleAllowPopupsForSite(aEvent)
|
|
{
|
|
var currentURI = Browser.selectedBrowser.webNavigation.currentURI;
|
|
var pm = Cc["@mozilla.org/permissionmanager;1"].getService(this._kIPM);
|
|
pm.add(currentURI, "popup", this._kIPM.ALLOW_ACTION);
|
|
|
|
Browser.getNotificationBox().removeCurrentNotification();
|
|
},
|
|
|
|
dontShowMessage: function dontShowMessage()
|
|
{
|
|
var showMessage = gPrefService.getBoolPref("privacy.popups.showBrowserMessage");
|
|
gPrefService.setBoolPref("privacy.popups.showBrowserMessage", !showMessage);
|
|
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 = {
|
|
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() {
|
|
// 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();
|
|
}
|
|
}
|
|
};
|
|
|
|
#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;
|
|
window.resizeTo(width, height);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
#endif
|
|
|
|
function getNotificationBox(aWindow) {
|
|
return Browser.getNotificationBox();
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
ProgressController.prototype = {
|
|
get browser() {
|
|
return this._tab.browser;
|
|
},
|
|
|
|
onStateChange: function onStateChange(aWebProgress, aRequest, aStateFlags, aStatus) {
|
|
// ignore notification that aren't about the main document (iframes, etc)
|
|
if (aWebProgress.DOMWindow != this._tab.browser.contentWindow)
|
|
return;
|
|
|
|
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) {
|
|
// XXX this code is not multiple-tab friendly.
|
|
var location = aLocationURI ? aLocationURI.spec : "";
|
|
let selectedBrowser = Browser.selectedBrowser;
|
|
let lastURI = selectedBrowser.lastURI;
|
|
|
|
//don't do anything for about:blank or about:firstrun on first display
|
|
if (!lastURI && (location == "about:blank" || location == "about:firstrun" ))
|
|
return;
|
|
|
|
this._hostChanged = true;
|
|
|
|
// This code here does not compare uris exactly when determining
|
|
// whether or not the message(s) should be hidden since the message
|
|
// may be prematurely hidden when an install is invoked by a click
|
|
// on a link that looks like this:
|
|
//
|
|
// <a href="#" onclick="return install();">Install Foo</a>
|
|
//
|
|
// - which fires a onLocationChange message to uri + '#'...
|
|
selectedBrowser.lastURI = aLocationURI;
|
|
if (lastURI) {
|
|
var oldSpec = lastURI.spec;
|
|
var oldIndexOfHash = oldSpec.indexOf("#");
|
|
if (oldIndexOfHash != -1)
|
|
oldSpec = oldSpec.substr(0, oldIndexOfHash);
|
|
var newSpec = location;
|
|
var newIndexOfHash = newSpec.indexOf("#");
|
|
if (newIndexOfHash != -1)
|
|
newSpec = newSpec.substr(0, newSpec.indexOf("#"));
|
|
if (newSpec != oldSpec) {
|
|
// Remove all the notifications, except for those which want to
|
|
// persist across the first location change.
|
|
|
|
// XXX
|
|
// var nBox = Browser.getNotificationBox();
|
|
// nBox.removeTransientNotifications();
|
|
}
|
|
}
|
|
|
|
if (aWebProgress.DOMWindow == selectedBrowser.contentWindow) {
|
|
BrowserUI.setURI();
|
|
|
|
// 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) {
|
|
},
|
|
|
|
_networkStart: function _networkStart() {
|
|
this._tab.startLoading();
|
|
|
|
if (Browser.selectedBrowser == this.browser) {
|
|
BrowserUI.update(TOOLBARSTATE_LOADING);
|
|
|
|
// We increase the default text size to make the text more readable when
|
|
// the page is zoomed out
|
|
if (this.browser.markupDocumentViewer.textZoom != kDefaultTextZoom)
|
|
this.browser.markupDocumentViewer.textZoom = kDefaultTextZoom;
|
|
}
|
|
|
|
// broadcast a URLChanged message for consumption by InputHandler
|
|
let event = document.createEvent("Events");
|
|
event.initEvent("URLChanged", true, false);
|
|
this.browser.dispatchEvent(event);
|
|
},
|
|
|
|
_networkStop: function _networkStop() {
|
|
this._tab.endLoading();
|
|
|
|
if (Browser.selectedBrowser == this.browser) {
|
|
BrowserUI.update(TOOLBARSTATE_LOADED);
|
|
this.browser.docShell.isOffScreenBrowser = true;
|
|
}
|
|
|
|
this._tab.updateThumbnail();
|
|
},
|
|
|
|
_documentStop: function() {
|
|
// translate any phone numbers
|
|
Browser.translatePhoneNumbers();
|
|
|
|
if (Browser.selectedBrowser == this.browser) {
|
|
// focus the dom window
|
|
if (this.browser.currentURI.spec != "about:blank")
|
|
this.browser.contentWindow.focus();
|
|
}
|
|
},
|
|
|
|
// Properties used to cache security state used to update the UI
|
|
_state: null,
|
|
_host: undefined,
|
|
_hostChanged: false, // onLocationChange will flip this bit
|
|
|
|
// 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._state = aState;
|
|
|
|
try {
|
|
this._host = getBrowser().contentWindow.location.host;
|
|
}
|
|
catch(ex) {
|
|
this._host = null;
|
|
}
|
|
|
|
this._hostChanged = false;
|
|
|
|
// Don't pass in the actual location object, since it can cause us to
|
|
// hold on to the window object too long. Just pass in the fields we
|
|
// care about. (bug 424829)
|
|
var location = getBrowser().contentWindow.location;
|
|
var locationObj = {};
|
|
try {
|
|
locationObj.host = location.host;
|
|
locationObj.hostname = location.hostname;
|
|
locationObj.port = location.port;
|
|
}
|
|
catch (ex) {
|
|
// Can sometimes throw if the URL being visited has no host/hostname,
|
|
// e.g. about:blank. The _state for these pages means we won't need these
|
|
// properties anyways, though.
|
|
}
|
|
getIdentityHandler().checkIdentity(this._state, locationObj);
|
|
},
|
|
|
|
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;
|
|
}
|
|
};
|
|
|
|
|
|
function Tab() {
|
|
this._id = null;
|
|
this._browser = null;
|
|
this._browserViewportState = null;
|
|
this._state = null;
|
|
this._listener = null;
|
|
this._loading = false;
|
|
this._chromeTab = null;
|
|
this.create();
|
|
}
|
|
|
|
Tab.prototype = {
|
|
get browser() {
|
|
return this._browser;
|
|
},
|
|
|
|
get browserViewportState() {
|
|
return this._browserViewportState;
|
|
},
|
|
|
|
get chromeTab() {
|
|
return this._chromeTab;
|
|
},
|
|
|
|
|
|
_resizeAndPaint: function() {
|
|
let bv = Browser._browserView;
|
|
|
|
if (this == Browser.selectedTab) {
|
|
// !!! --- RESIZE HACK BEGIN -----
|
|
bv.simulateMozAfterSizeChange();
|
|
// !!! --- RESIZE HACK END -----
|
|
|
|
bv.zoomToPage();
|
|
}
|
|
bv.commitBatchOperation();
|
|
|
|
if (this._loading) {
|
|
// kick ourselves off 2s later while we're still loading
|
|
bv.beginBatchOperation();
|
|
this._loadingTimeout = setTimeout(Util.bind(this._resizeAndPaint, this), 2000);
|
|
} else {
|
|
delete this._loadingTimeout;
|
|
}
|
|
},
|
|
|
|
startLoading: function() {
|
|
if (this._loading)
|
|
dump("!!! Already loading this tab, please file a bug\n");
|
|
|
|
this._loading = true;
|
|
|
|
if (!this._loadingTimeout) {
|
|
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();
|
|
},
|
|
|
|
isLoading: function() {
|
|
return this._loading;
|
|
},
|
|
|
|
load: function(uri) {
|
|
this._browser.setAttribute("src", uri);
|
|
},
|
|
|
|
create: function() {
|
|
this._chromeTab = document.createElement("richlistitem");
|
|
this._chromeTab.setAttribute("type", "documenttab");
|
|
document.getElementById("tabs").addTab(this._chromeTab);
|
|
|
|
this._createBrowser();
|
|
},
|
|
|
|
destroy: function() {
|
|
this._destroyBrowser();
|
|
document.getElementById("tabs").removeTab(this._chromeTab);
|
|
this._chromeTab = null;
|
|
},
|
|
|
|
_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");
|
|
|
|
browser.setAttribute("style", "overflow: -moz-hidden-unscrollable; visibility: hidden; width: " + kDefaultBrowserWidth + "px; height: " + scaledHeight + "px;");
|
|
browser.setAttribute("type", "content");
|
|
|
|
// Attach the popup contextmenu
|
|
let container = document.getElementById("tile-container");
|
|
browser.setAttribute("contextmenu", container.getAttribute("contextmenu"));
|
|
let autocompletepopup = container.getAttribute("autocompletepopup");
|
|
if (autocompletepopup)
|
|
browser.setAttribute("autocompletepopup", autocompletepopup);
|
|
|
|
// Append the browser to the document, which should start the page load
|
|
document.getElementById("browsers").appendChild(browser);
|
|
|
|
// stop about:blank from loading
|
|
browser.stop();
|
|
|
|
// Initialize a viewport state for BrowserView
|
|
let initVis = Browser.getVisibleRect();
|
|
initVis.x = 0;
|
|
initVis.y = 0;
|
|
this._browserViewportState = BrowserView.Util.createBrowserViewportState(browser, initVis);
|
|
|
|
// Attach a separate progress listener to the browser
|
|
this._listener = new ProgressController(this);
|
|
browser.addProgressListener(this._listener);
|
|
},
|
|
|
|
_destroyBrowser: function() {
|
|
document.getElementById("browsers").removeChild(this._browser);
|
|
this._browser = null;
|
|
},
|
|
|
|
saveState: function() {
|
|
let state = { };
|
|
|
|
this._url = browser.contentWindow.location.toString();
|
|
var browser = this.getBrowserForDisplay(display);
|
|
var doc = browser.contentDocument;
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
|
|
state._scrollX = browser.contentWindow.scrollX;
|
|
state._scrollY = browser.contentWindow.scrollY;
|
|
|
|
this._state = state;
|
|
},
|
|
|
|
restoreState: function() {
|
|
let state = this._state;
|
|
if (!state)
|
|
return;
|
|
|
|
let doc = this._browser.contentDocument;
|
|
for (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.scrollTo(state._scrollX, state._scrollY);
|
|
},
|
|
|
|
updateThumbnail: function() {
|
|
if (!this._browser)
|
|
return;
|
|
|
|
// XXX draw from the tiles in to the source
|
|
let srcCanvas = (Browser.selectedBrowser == this._browser) ? document.getElementById("browser-canvas") : null;
|
|
this._chromeTab.updateThumbnail(this._browser, srcCanvas);
|
|
}
|
|
};
|