releases-comm-central/suite/common/utilityOverlay.js

1296 строки
42 KiB
JavaScript

/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* ***** 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.org code.
*
* The Initial Developer of the Original Code is
* Netscape Communications Corporation.
* Portions created by the Initial Developer are Copyright (C) 1998
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Alec Flett <alecf@netscape.com>
*
* Alternatively, the contents of this file may be used under the terms of
* either of 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 ***** */
/**
* Communicator Shared Utility Library
* for shared application glue for the Communicator suite of applications
**/
/*
Note: All Editor/Composer-related methods have been moved to editorApplicationOverlay.js,
so app windows that require those must include editorNavigatorOverlay.xul
*/
/**
* Go into online/offline mode
**/
const kIOServiceProgID = "@mozilla.org/network/io-service;1";
const kObserverServiceProgID = "@mozilla.org/observer-service;1";
const kProxyManual = ["network.proxy.ftp",
"network.proxy.gopher",
"network.proxy.http",
"network.proxy.socks",
"network.proxy.ssl"];
var gShowBiDi = false;
var gUtilityBundle = null;
function toggleOfflineStatus()
{
var checkfunc;
try {
checkfunc = document.getElementById("offline-status").getAttribute('checkfunc');
}
catch (ex) {
checkfunc = null;
}
var ioService = Components.classes[kIOServiceProgID]
.getService(Components.interfaces.nsIIOService2);
if (checkfunc) {
if (!eval(checkfunc)) {
// the pre-offline check function returned false, so don't go offline
return;
}
}
ioService.manageOfflineStatus = false;
ioService.offline = !ioService.offline;
}
function setNetworkStatus(networkProxyType)
{
var prefService = Components.classes["@mozilla.org/preferences-service;1"];
prefService = prefService.getService(Components.interfaces.nsIPrefService);
var prefBranch = prefService.getBranch(null);
try {
prefBranch.setIntPref("network.proxy.type", networkProxyType);
}
catch (ex) {}
}
function InitProxyMenu()
{
var networkProxyNo = document.getElementById("network-proxy-no");
var networkProxyManual = document.getElementById("network-proxy-manual");
var networkProxyPac = document.getElementById("network-proxy-pac");
var networkProxyWpad = document.getElementById("network-proxy-wpad");
var networkProxySystem = document.getElementById("network-proxy-system");
var prefService = Components.classes["@mozilla.org/preferences-service;1"];
prefService = prefService.getService(Components.interfaces.nsIPrefService);
var prefBranch = prefService.getBranch(null);
var proxyLocked = prefBranch.prefIsLocked("network.proxy.type");
if (proxyLocked) {
networkProxyNo.setAttribute("disabled", "true");
networkProxyWpad.setAttribute("disabled", "true");
networkProxySystem.setAttribute("disabled", "true");
}
else {
networkProxyNo.removeAttribute("disabled");
networkProxyWpad.removeAttribute("disabled");
networkProxySystem.removeAttribute("disabled");
}
// If no proxy is configured, disable the menuitems.
// Checking for proxy manual settings.
var proxyManuallyConfigured = false;
for (var i = 0; i < kProxyManual.length; i++) {
if (GetStringPref(kProxyManual[i]) != "") {
proxyManuallyConfigured = true;
break;
}
}
if (proxyManuallyConfigured && !proxyLocked) {
networkProxyManual.removeAttribute("disabled");
}
else {
networkProxyManual.setAttribute("disabled", "true");
}
//Checking for proxy PAC settings.
var proxyAutoConfigured = false;
if (GetStringPref("network.proxy.autoconfig_url") != "")
proxyAutoConfigured = true;
if (proxyAutoConfigured && !proxyLocked) {
networkProxyPac.removeAttribute("disabled");
}
else {
networkProxyPac.setAttribute("disabled", "true");
}
var networkProxyType;
try {
networkProxyType = prefBranch.getIntPref("network.proxy.type");
} catch(e) {}
// The pref value 3 for network.proxy.type is unused to maintain
// backwards compatibility. Treat 3 equally to 0. See bug 115720.
var networkProxyStatus = [networkProxyNo, networkProxyManual, networkProxyPac,
networkProxyNo, networkProxyWpad,
networkProxySystem];
networkProxyStatus[networkProxyType].setAttribute("checked", "true");
}
function setProxyTypeUI()
{
var panel = document.getElementById("offline-status");
if (!panel)
return;
var prefService = Components.classes["@mozilla.org/preferences-service;1"];
prefService = prefService.getService(Components.interfaces.nsIPrefService);
var prefBranch = prefService.getBranch(null);
try {
var networkProxyType = prefBranch.getIntPref("network.proxy.type");
} catch(e) {}
var onlineTooltip = "onlineTooltip" + networkProxyType;
panel.setAttribute("tooltiptext", gUtilityBundle.getString(onlineTooltip));
}
function GetStringPref(name)
{
try {
return pref.getComplexValue(name, Components.interfaces.nsISupportsString).data;
} catch (e) {}
return "";
}
function setOfflineUI(offline)
{
var broadcaster = document.getElementById("Communicator:WorkMode");
var panel = document.getElementById("offline-status");
if (!broadcaster || !panel) return;
// Checking for a preference "network.online", if it's locked, disabling
// network icon and menu item
var prefService = Components.classes["@mozilla.org/preferences-service;1"];
prefService = prefService.getService(Components.interfaces.nsIPrefService);
var prefBranch = prefService.getBranch(null);
var offlineLocked = prefBranch.prefIsLocked("network.online");
if (offlineLocked ) {
broadcaster.setAttribute("disabled","true");
}
if (offline)
{
broadcaster.setAttribute("offline", "true");
broadcaster.setAttribute("checked", "true");
panel.removeAttribute("context");
panel.setAttribute("tooltiptext", gUtilityBundle.getString("offlineTooltip"));
}
else
{
broadcaster.removeAttribute("offline");
broadcaster.removeAttribute("checked");
panel.setAttribute("context", "networkProperties");
try {
var networkProxyType = prefBranch.getIntPref("network.proxy.type");
} catch(e) {}
var onlineTooltip = "onlineTooltip" + networkProxyType;
panel.setAttribute("tooltiptext", gUtilityBundle.getString(onlineTooltip));
}
}
function getBrowserURL() {
try {
var prefs = Components.classes["@mozilla.org/preferences-service;1"]
.getService(Components.interfaces.nsIPrefBranch);
var url = prefs.getCharPref("browser.chromeURL");
if (url)
return url;
} catch(e) {
}
return "chrome://navigator/content/navigator.xul";
}
function goPreferences(paneID)
{
//check for an existing pref window and focus it; it's not application modal
const kWindowMediatorContractID = "@mozilla.org/appshell/window-mediator;1";
const kWindowMediatorIID = Components.interfaces.nsIWindowMediator;
const kWindowMediator = Components.classes[kWindowMediatorContractID]
.getService(kWindowMediatorIID);
var lastPrefWindow = kWindowMediator.getMostRecentWindow("mozilla:preferences");
if (lastPrefWindow)
lastPrefWindow.focus();
else
openDialog("chrome://communicator/content/pref/preferences.xul",
"PrefWindow", "chrome,titlebar,dialog=no,resizable",
paneID);
}
function goToggleToolbar( id, elementID )
{
var toolbar = document.getElementById( id );
var element = document.getElementById( elementID );
if ( toolbar )
{
var attribValue = toolbar.getAttribute("hidden") ;
if ( attribValue == "true" )
{
toolbar.setAttribute("hidden", "false" );
if ( element )
element.setAttribute("checked","true")
}
else
{
toolbar.setAttribute("hidden", "true" );
if ( element )
element.setAttribute("checked","false")
}
document.persist(id, "hidden");
document.persist(elementID, "checked");
if (toolbar.hasAttribute("customindex"))
persistCustomToolbar(toolbar);
}
}
var gCustomizeSheet = false;
function goCustomizeToolbar(toolbox)
{
/* If the toolbox has a method "customizeInit" then call it first.
The optional "customizeDone" method will be invoked by the callback
from the Customize Window so we don't need to take care of that */
if ("customizeInit" in toolbox)
toolbox.customizeInit();
var customizeURL = "chrome://global/content/customizeToolbar.xul";
gCustomizeSheet = getBoolPref("toolbar.customization.usesheet", false);
if (gCustomizeSheet) {
var sheetFrame = document.getElementById("customizeToolbarSheetIFrame");
sheetFrame.hidden = false;
sheetFrame.toolbox = toolbox;
// The document might not have been loaded yet, if this is the first time.
// If it is already loaded, reload it so that the onload initialization
// code re-runs.
if (sheetFrame.getAttribute("src") == customizeURL)
sheetFrame.contentWindow.location.reload();
else
sheetFrame.setAttribute("src", customizeURL);
// XXXmano: there's apparently no better way to get this when the iframe
// is hidden
var sheetWidth = sheetFrame.style.width.match(/([0-9]+)px/)[1];
document.getElementById("customizeToolbarSheetPopup")
.openPopup(toolbox,
"after_start",
(window.innerWidth - sheetWidth) / 2, 0);
return sheetFrame.contentWindow;
}
else {
return window.openDialog(customizeURL,
"",
"chrome,all,dependent",
toolbox);
}
}
function onViewToolbarsPopupShowing(aEvent)
{
var popup = aEvent.target;
// Empty the menu
var deadItems = popup.getElementsByAttribute("toolbarid", "*");
for (let i = deadItems.length - 1; i >= 0; --i)
popup.removeChild(deadItems[i]);
var firstMenuItem = popup.firstChild;
var toolbar = document.popupNode;
while (toolbar.localName != "toolbar")
toolbar = toolbar.parentNode;
var toolbox = toolbar.parentNode;
var toolbars = toolbox.getElementsByAttribute("toolbarname", "*");
for (let i = 0; i < toolbars.length; ++i) {
let bar = toolbars[i];
let menuItem = document.createElement("menuitem");
menuItem.setAttribute("toolbarid", bar.id);
menuItem.setAttribute("type", "checkbox");
menuItem.setAttribute("label", bar.getAttribute("toolbarname"));
menuItem.setAttribute("accesskey", bar.getAttribute("accesskey"));
menuItem.setAttribute("checked", !bar.hidden);
popup.insertBefore(menuItem, firstMenuItem);
}
var mode = toolbar.getAttribute("mode") || "full";
var modePopup = document.getElementById("toolbarmodePopup");
var radio = modePopup.getElementsByAttribute("value", mode);
radio[0].setAttribute("checked", "true");
var small = toolbar.getAttribute("iconsize") == "small";
var smallicons = document.getElementById("toolbarmode-smallicons");
smallicons.setAttribute("checked", small);
smallicons.setAttribute("disabled", mode == "text");
var end = toolbar.getAttribute("labelalign") == "end";
var labelalign = document.getElementById("toolbarmode-labelalign");
labelalign.setAttribute("checked", end);
labelalign.setAttribute("disabled", mode != "full");
var custommode = (toolbar.getAttribute("mode") || "full") !=
(toolbar.getAttribute("defaultmode") ||
toolbox.getAttribute("mode") ||
"full");
var customicon = (toolbar.getAttribute("iconsize") || "large") !=
(toolbar.getAttribute("defaulticonsize") ||
toolbox.getAttribute("iconsize") ||
"large");
var customalign = (toolbar.getAttribute("labelalign") || "bottom") !=
(toolbar.getAttribute("defaultlabelalign") ||
toolbox.getAttribute("labelalign") ||
"bottom");
var custom = custommode || customicon || customalign ||
toolbar.hasAttribute("ignoremodepref");
var defmode = document.getElementById("toolbarmode-default");
defmode.setAttribute("checked", !custom);
defmode.setAttribute("disabled", !custom);
var command = document.getElementById("cmd_CustomizeToolbars");
var menuitem = document.getElementById("customize_toolbars");
menuitem.hidden = !command;
menuitem.previousSibling.hidden = !command;
}
function onViewToolbarCommand(aEvent)
{
var toolbar = aEvent.originalTarget.getAttribute("toolbarid");
var menuitem = document.getElementById(toolbar).getAttribute("togglemenuitem");
goToggleToolbar(toolbar, menuitem);
}
function goSetToolbarState(aEvent)
{
aEvent.stopPropagation();
var toolbar = document.popupNode;
while (toolbar.localName != "toolbar")
toolbar = toolbar.parentNode;
var toolbox = toolbar.parentNode;
var target = aEvent.originalTarget;
var mode = target.value;
var radiogroup = target.getAttribute("name");
var primary = /toolbar-primary/.test(toolbar.getAttribute("class"));
switch (mode) {
case "smallicons":
var size = target.getAttribute("checked") == "true" ? "small" : "large";
toolbar.setAttribute("iconsize", size);
break;
case "end":
var align = target.getAttribute("checked") == "true" ? "end" : "bottom";
toolbar.setAttribute("labelalign", align);
break;
case "default":
toolbar.setAttribute("mode", toolbar.getAttribute("defaultmode") ||
toolbox.getAttribute("mode"));
toolbar.setAttribute("iconsize", toolbar.getAttribute("defaulticonsize") ||
toolbox.getAttribute("iconsize"));
toolbar.setAttribute("labelalign", toolbar.getAttribute("defaultlabelalign") ||
toolbox.getAttribute("labelalign"));
if (primary)
toolbar.removeAttribute("ignoremodepref");
break;
default:
toolbar.setAttribute("mode", mode);
if (primary)
toolbar.setAttribute("ignoremodepref", "true");
break;
}
document.persist(toolbar.id, "mode");
document.persist(toolbar.id, "iconsize");
document.persist(toolbar.id, "labelalign");
if (primary)
document.persist(toolbar.id, "ignoremodepref");
if (toolbar.hasAttribute("customindex"))
persistCustomToolbar(toolbar);
}
function persistCustomToolbar(toolbar)
{
var toolbox = toolbar.parentNode;
var name = toolbar.getAttribute("toolbarname").replace(" ", "_");
var attrs = ["mode", "iconsize", "labelalign", "hidden"];
for (let i = 0; i < attrs.length; i++) {
let value = toolbar.getAttribute(attrs[i]);
let attr = name + attrs[i];
toolbox.toolbarset.setAttribute(attr, value);
document.persist(toolbox.toolbarset.id, attr);
}
}
/* Common Customize Toolbar code */
function toolboxCustomizeInit(menubarID)
{
// Disable the toolbar context menu items
var menubar = document.getElementById(menubarID);
for (let i = 0; i < menubar.childNodes.length; ++i) {
let item = menubar.childNodes[i];
if (item.getAttribute("disabled") != "true") {
item.setAttribute("disabled", "true");
item.setAttribute("saved-disabled", "false");
}
}
var cmd = document.getElementById("cmd_CustomizeToolbars");
cmd.setAttribute("disabled", "true");
}
function toolboxCustomizeDone(menubarID, toolbox, aToolboxChanged)
{
if (gCustomizeSheet) {
var sheetFrame = document.getElementById("customizeToolbarSheetIFrame");
sheetFrame.hidden = true;
document.getElementById("customizeToolbarSheetPopup").hidePopup();
if (content)
content.focus();
else
window.focus();
}
// Re-enable parts of the UI we disabled during the dialog
var menubar = document.getElementById(menubarID);
for (let i = 0; i < menubar.childNodes.length; ++i) {
let item = menubar.childNodes[i];
if (item.hasAttribute("saved-disabled")) {
item.removeAttribute("disabled");
item.removeAttribute("saved-disabled");
}
}
var cmd = document.getElementById("cmd_CustomizeToolbars");
cmd.removeAttribute("disabled");
var toolbars = toolbox.getElementsByAttribute("customindex", "*");
for (let i = 0; i < toolbars.length; ++i) {
persistCustomToolbar(toolbars[i]);
}
}
function toolboxCustomizeChange(toolbox, event)
{
if (event != "reset")
return;
var toolbars = toolbox.getElementsByAttribute("toolbarname", "*");
for (let i = 0; i < toolbars.length; ++i) {
let toolbar = toolbars[i];
toolbar.setAttribute("labelalign",
toolbar.getAttribute("defaultlabelalign") ||
toolbox.getAttribute("labelalign"));
document.persist(toolbar.id, "labelalign");
let primary = /toolbar-primary/.test(toolbar.getAttribute("class"));
if (primary) {
toolbar.removeAttribute("ignoremodepref");
document.persist(toolbar.id, "ignoremodepref");
}
}
}
function goClickThrobber( urlPref )
{
var url;
try {
var pref = Components.classes["@mozilla.org/preferences-service;1"]
.getService(Components.interfaces.nsIPrefBranch);
url = pref.getComplexValue(urlPref, Components.interfaces.nsIPrefLocalizedString).data;
}
catch(e) {
url = null;
}
if ( url )
openUILink(url);
}
function getTopWin()
{
var windowManager = Components.classes['@mozilla.org/appshell/window-mediator;1'].getService();
var windowManagerInterface = windowManager.QueryInterface( Components.interfaces.nsIWindowMediator);
var topWindowOfType = windowManagerInterface.getMostRecentWindow( "navigator:browser" );
if (topWindowOfType) {
return topWindowOfType;
}
return null;
}
function isRestricted( url )
{
try {
const nsIURIFixup = Components.interfaces.nsIURIFixup;
var uriFixup = Components.classes["@mozilla.org/docshell/urifixup;1"]
.getService(nsIURIFixup);
var url = uriFixup.createFixupURI(url, nsIURIFixup.FIXUP_FLAG_NONE);
const URI_INHERITS_SECURITY_CONTEXT =
Components.interfaces.nsIProtocolHandler.URI_INHERITS_SECURITY_CONTEXT;
return Components.classes["@mozilla.org/network/util;1"]
.getService(Components.interfaces.nsINetUtil)
.URIChainHasFlags(url, URI_INHERITS_SECURITY_CONTEXT);
} catch (e) {
return false;
}
}
function openTopWin( url, opener )
{
/* note that this chrome url should probably change to not have
all of the navigator controls, but if we do this we need to have
the option for chrome controls because goClickThrobber() needs to
use this function with chrome controls */
/* also, do we want to
limit the number of help windows that can be spawned? */
if ((url == null) || (url == "")) return null;
// avoid loading "", since this loads a directory listing
if (url == "") {
url = "about:blank";
}
var topWindowOfType = getTopWin();
if ( topWindowOfType )
{
if (!opener || !isRestricted(url))
topWindowOfType.loadURI(url);
else if (topWindowOfType.content == opener.top)
opener.open(url, "_top");
else
topWindowOfType.getBrowser().loadURIWithFlags(url,
Components.interfaces.nsIWebNavigation.LOAD_FLAGS_FROM_EXTERNAL);
topWindowOfType.content.focus();
return topWindowOfType;
}
return window.openDialog( getBrowserURL(), "_blank", "chrome,all,dialog=no", url );
}
function goAbout(aProtocol)
{
const kExistingWindow = Components.interfaces.nsIBrowserDOMWindow.OPEN_CURRENTWINDOW;
const kNewWindow = Components.interfaces.nsIBrowserDOMWindow.OPEN_NEWWINDOW;
var target;
var url = "about:" + (aProtocol || "");
var pref = Components.classes["@mozilla.org/preferences-service;1"]
.getService(Components.interfaces.nsIPrefBranch);
var defaultAboutState = pref.getIntPref("browser.link.open_external");
switch (defaultAboutState) {
case kNewWindow:
target = "window";
break;
case kExistingWindow:
target = "current";
break;
default:
target = "tab";
}
openUILinkIn(url, target);
}
function goReleaseNotes()
{
// get release notes URL from prefs
try {
var formatter = Components.classes["@mozilla.org/toolkit/URLFormatterService;1"]
.getService(Components.interfaces.nsIURLFormatter);
openUILink(formatter.formatURLPref("app.releaseNotesURL"));
}
catch (ex) { dump(ex); }
}
function checkForUpdates()
{
var um = Components.classes["@mozilla.org/updates/update-manager;1"]
.getService(Components.interfaces.nsIUpdateManager);
var prompter = Components.classes["@mozilla.org/updates/update-prompt;1"]
.createInstance(Components.interfaces.nsIUpdatePrompt);
// If there's an update ready to be applied, show the "Update Downloaded"
// UI instead and let the user know they have to restart the browser for
// the changes to be applied.
if (um.activeUpdate && um.activeUpdate.state == "pending")
prompter.showUpdateDownloaded(um.activeUpdate);
else
prompter.checkForUpdates();
}
function updateCheckUpdatesItem()
{
var updates = Components.classes["@mozilla.org/updates/update-service;1"]
.getService(Components.interfaces.nsIApplicationUpdateService);
var um = Components.classes["@mozilla.org/updates/update-manager;1"]
.getService(Components.interfaces.nsIUpdateManager);
// Disable the UI if the update enabled pref has been locked by the
// administrator or if we cannot update for some other reason.
var checkForUpdates = document.getElementById("checkForUpdates");
var canUpdate = updates.canUpdate;
checkForUpdates.setAttribute("disabled", !canUpdate);
if (!canUpdate)
return;
// By default, show "Check for Updates..."
var key = "default";
if (um.activeUpdate) {
switch (um.activeUpdate.state) {
case "downloading":
// If we're downloading an update at present, show the text:
// "Downloading SeaMonkey x.x..." otherwise we're paused, and show
// "Resume Downloading SeaMonkey x.x..."
key = updates.isDownloading ? "downloading" : "resume";
break;
case "pending":
// If we're waiting for the user to restart, show: "Apply Downloaded
// Updates Now..."
key = "pending";
break;
}
}
// If there's an active update, substitute its name into the label
// we show for this item, otherwise display a generic label.
if (um.activeUpdate && um.activeUpdate.name)
checkForUpdates.label = gUtilityBundle.getFormattedString("updatesItem_" + key,
[um.activeUpdate.name]);
else
checkForUpdates.label = gUtilityBundle.getString("updatesItem_" + key + "Fallback");
checkForUpdates.accessKey = gUtilityBundle.getString("updatesItem_" + key + "AccessKey");
if (um.activeUpdate && updates.isDownloading)
checkForUpdates.setAttribute("loading", "true");
else
checkForUpdates.removeAttribute("loading");
}
// update menu items that rely on focus
function goUpdateGlobalEditMenuItems()
{
goUpdateCommand('cmd_undo');
goUpdateCommand('cmd_redo');
goUpdateCommand('cmd_cut');
goUpdateCommand('cmd_copy');
goUpdateCommand('cmd_paste');
goUpdateCommand('cmd_selectAll');
goUpdateCommand('cmd_delete');
if (gShowBiDi)
goUpdateCommand('cmd_switchTextDirection');
}
// update menu items that rely on the current selection
function goUpdateSelectEditMenuItems()
{
goUpdateCommand('cmd_cut');
goUpdateCommand('cmd_copy');
goUpdateCommand('cmd_delete');
goUpdateCommand('cmd_selectAll');
}
// update menu items that relate to undo/redo
function goUpdateUndoEditMenuItems()
{
goUpdateCommand('cmd_undo');
goUpdateCommand('cmd_redo');
}
// update menu items that depend on clipboard contents
function goUpdatePasteMenuItems()
{
goUpdateCommand('cmd_paste');
}
// update Find As You Type menu items, they rely on focus
function goUpdateFindTypeMenuItems()
{
goUpdateCommand('cmd_findTypeText');
goUpdateCommand('cmd_findTypeLinks');
}
// Gather all descendent text under given document node.
function gatherTextUnder(root)
{
var text = "";
var node = root.firstChild;
var depth = 1;
while ( node && depth > 0 ) {
// See if this node is text.
if ( node.nodeType == Node.TEXT_NODE ) {
// Add this text to our collection.
text += " " + node.data;
} else if ( node instanceof HTMLImageElement ) {
// If it has an alt= attribute, add that.
var altText = node.getAttribute( "alt" );
if ( altText && altText != "" ) {
text += " " + altText;
}
}
// Find next node to test.
// First, see if this node has children.
if ( node.hasChildNodes() ) {
// Go to first child.
node = node.firstChild;
depth++;
} else {
// No children, try next sibling.
if ( node.nextSibling ) {
node = node.nextSibling;
} else {
// Last resort is a sibling of an ancestor.
while ( node && depth > 0 ) {
node = node.parentNode;
depth--;
if ( node.nextSibling ) {
node = node.nextSibling;
break;
}
}
}
}
}
// Strip leading and trailing whitespaces,
// then compress remaining whitespaces.
return text.trim().replace(/\s+/g, " ");
}
var offlineObserver = {
observe: function(subject, topic, state) {
// sanity checks
if (topic != "network:offline-status-changed") return;
setOfflineUI(state == "offline");
}
}
var proxyTypeObserver = {
observe: function(subject, topic, state) {
// sanity checks
var ioService = Components.classes[kIOServiceProgID]
.getService(Components.interfaces.nsIIOService);
if (state == "network.proxy.type" && !ioService.offline)
setProxyTypeUI();
}
}
function utilityOnLoad(aEvent)
{
gUtilityBundle = document.getElementById("bundle_utilityOverlay");
var broadcaster = document.getElementById("Communicator:WorkMode");
if (!broadcaster) return;
var observerService = Components.classes[kObserverServiceProgID]
.getService(Components.interfaces.nsIObserverService);
observerService.addObserver(offlineObserver, "network:offline-status-changed", false);
// make sure we remove this observer later
var prefService = Components.classes["@mozilla.org/preferences-service;1"];
prefService = prefService.getService(Components.interfaces.nsIPrefService);
var prefBranch = prefService.getBranch(null);
prefBranch = prefBranch.QueryInterface(Components.interfaces.nsIPrefBranch2);
prefBranch.addObserver("network.proxy.type", proxyTypeObserver, false);
addEventListener("unload", utilityOnUnload, false);
// set the initial state
var ioService = Components.classes[kIOServiceProgID]
.getService(Components.interfaces.nsIIOService);
setOfflineUI(ioService.offline);
// Check for system proxy settings class and show menuitem if present
if ("@mozilla.org/system-proxy-settings;1" in Components.classes &&
document.getElementById("network-proxy-system"))
document.getElementById("network-proxy-system").hidden = false;
}
function utilityOnUnload(aEvent)
{
var observerService = Components.classes[kObserverServiceProgID]
.getService(Components.interfaces.nsIObserverService);
observerService.removeObserver(offlineObserver, "network:offline-status-changed");
var prefService = Components.classes["@mozilla.org/preferences-service;1"];
prefService = prefService.getService(Components.interfaces.nsIPrefService);
var prefBranch = prefService.getBranch(null);
prefBranch = prefBranch.QueryInterface(Components.interfaces.nsIPrefBranch2);
prefBranch.removeObserver("network.proxy.type", proxyTypeObserver);
}
addEventListener("load", utilityOnLoad, false);
function GenerateValidFilename(filename, extension)
{
if (filename) // we have a title; let's see if it's usable
{
// clean up the filename to make it usable and
// then trim whitespace from beginning and end
filename = validateFileName(filename).trim();
if (filename.length > 0)
return filename + extension;
}
return null;
}
function validateFileName(aFileName)
{
var re = /[\/]+/g;
if (navigator.appVersion.indexOf("Windows") != -1) {
re = /[\\\/\|]+/g;
aFileName = aFileName.replace(/[\"]+/g, "'");
aFileName = aFileName.replace(/[\*\:\?]+/g, " ");
aFileName = aFileName.replace(/[\<]+/g, "(");
aFileName = aFileName.replace(/[\>]+/g, ")");
}
else if (navigator.appVersion.indexOf("Macintosh") != -1)
re = /[\:\/]+/g;
return aFileName.replace(re, "_");
}
function focusElement(aElement)
{
if (isElementVisible(aElement))
aElement.focus();
}
function isElementVisible(aElement)
{
// If aElement or a direct or indirect parent is hidden or collapsed,
// height, width or both will be 0.
var bo = aElement.boxObject;
return (bo.height > 0 && bo.width > 0);
}
/**
* Handle command events bubbling up from error page content
* called from oncommand by <browser>s that support error pages
*/
function BrowserOnCommand(event)
{
// Don't trust synthetic events
if (!event.isTrusted)
return;
const ot = event.originalTarget;
const ownerDoc = 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(ownerDoc.documentURI) ||
/^about:certerror\?/.test(ownerDoc.documentURI)) {
if (ot.id == 'exceptionDialogButton') {
var params = { exceptionAdded : false };
try {
const prefBranch =
Components.classes["@mozilla.org/preferences-service;1"]
.getService(Components.interfaces.nsIPrefService)
.getBranch(null);
switch (prefBranch.getIntPref("browser.ssl_override_behavior")) {
case 2 : // Pre-fetch & pre-populate.
params.prefetchCert = true;
// Fall through.
case 1 : // Pre-populate.
params.location = ownerDoc.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)
ownerDoc.location.reload();
}
else if (ot.id == 'getMeOutOfHereButton') {
// Redirect them to a known-functioning page, default start page
var url = "about:blank";
try {
url = pref.getComplexValue("browser.startup.homepage",
Components.interfaces.nsIPrefLocalizedString).data;
} catch(e) {
Components.utils.reportError("Couldn't get homepage pref: " + e);
}
content.location = url;
}
}
}
function popupNotificationMenuShowing(event)
{
var notificationbox = document.popupNode.parentNode.control;
var uri = notificationbox.activeBrowser.currentURI;
var allowPopupsForSite = document.getElementById("allowPopupsForSite");
allowPopupsForSite.notificationbox = notificationbox;
var showPopupManager = document.getElementById("showPopupManager");
// Only offer this menu item for the top window.
// See bug 280536 for problems with frames and iframes.
try {
// uri.host generates an exception on nsISimpleURIs, but we also
// don't want to show this menu item when there is no host.
allowPopupsForSite.hidden = !uri.host;
var allowString = gUtilityBundle.getFormattedString("popupAllow", [uri.host]);
allowPopupsForSite.setAttribute("label", allowString);
showPopupManager.hostport = uri.hostPort;
} catch (ex) {
allowPopupsForSite.hidden = true;
showPopupManager.hostport = "";
}
var separator = document.getElementById("popupNotificationMenuSeparator");
separator.hidden = !createShowPopupsMenu(event.target, notificationbox.activeBrowser);
}
function createShowPopupsMenu(parent, browser)
{
while (parent.lastChild && ("popup" in parent.lastChild))
parent.removeChild(parent.lastChild);
if (!browser)
return false;
var popups = browser.pageReport;
if (!popups)
return false;
for (var i = 0; i < popups.length; i++) {
var popup = popups[i];
var menuitem = document.createElement("menuitem");
var str = gUtilityBundle.getFormattedString("popupMenuShow",
[popup.popupWindowURI.spec]);
menuitem.setAttribute("label", str);
menuitem.popup = popup;
parent.appendChild(menuitem);
}
return true;
}
function popupBlockerMenuCommand(target)
{
if (!("popup" in target))
return;
var popup = target.popup;
var reqWin = popup.requestingWindow;
if (reqWin.document == popup.requestingDocument)
reqWin.open(popup.popupWindowURI.spec, popup.popupWindowName, popup.popupWindowFeatures);
}
function disablePopupBlockerNotifications()
{
pref.setBoolPref("privacy.popups.showBrowserMessage", false);
}
/**
* isValidFeed: checks whether the given data represents a valid feed.
*
* @param aData
* An object representing a feed with title, href and type.
* @param aPrincipal
* The principal of the document, used for security check.
* @param aIsFeed
* Whether this is already a known feed or not, if true only a security
* check will be performed.
*/
function isValidFeed(aData, aPrincipal, aIsFeed)
{
if (!aData || !aPrincipal)
return null;
var type = aData.type.toLowerCase().replace(/^\s+|\s*(?:;.*)?$/g, "");
if (aIsFeed || /^application\/(?:atom|rss)\+xml$/.test(type)) {
try {
urlSecurityCheck(aData.href, aPrincipal,
Components.interfaces.nsIScriptSecurityManager.DISALLOW_INHERIT_PRINCIPAL);
return type || "application/rss+xml";
}
catch(ex) {
}
}
return null;
}
// Used as an onclick handler for UI elements with link-like behavior.
// e.g. onclick="checkForMiddleClick(this, event);"
function checkForMiddleClick(node, event) {
// We should be using the disabled property here instead of the attribute,
// but some elements that this function is used with don't support it (e.g.
// menuitem).
if (node.getAttribute("disabled") == "true")
return; // Do nothing
if (event.button == 1) {
/* Execute the node's oncommand or command.
*
* XXX: we should use node.oncommand(event) once bug 246720 is fixed.
*/
var target = node.hasAttribute("oncommand") ? node :
node.ownerDocument.getElementById(node.getAttribute("command"));
var fn = new Function("event", target.getAttribute("oncommand"));
fn.call(target, event);
// If the middle-click was on part of a menu, close the menu.
// (Menus close automatically with left-click but not with middle-click.)
closeMenus(event.target);
}
}
// Closes all popups that are ancestors of the node.
function closeMenus(node)
{
for (; node; node = node.parentNode) {
if (node instanceof Components.interfaces.nsIDOMXULPopupElement)
node.hidePopup();
}
}
function getBoolPref(prefname, def)
{
try {
var pref = Components.classes["@mozilla.org/preferences-service;1"]
.getService(Components.interfaces.nsIPrefBranch);
return pref.getBoolPref(prefname);
}
catch (er) {
return def;
}
}
// openUILink handles clicks on UI elements that cause URLs to load.
function openUILink(url, e, ignoreButton, ignoreSave, allowKeywordFixup)
{
var where = whereToOpenLink(e, ignoreButton, ignoreSave);
return openUILinkIn(url, where, allowKeywordFixup);
}
/* whereToOpenLink() looks at an event to decide where to open a link.
*
* The event may be a mouse event (click, double-click, middle-click) or keypress event (enter).
*
* The logic for modifiers is as following:
* If browser.tabs.opentabfor.middleclick is true, then Ctrl (or Meta) and middle-click
* open a new tab, depending on Shift and browser.tabs.loadInBackground.
* Otherwise if middlemouse.openNewWindow is true, then Ctrl (or Meta) and middle-click
* open a new window.
* Otherwise if middle-click is pressed then nothing happens.
* Save is Alt or Shift depending on the ui.key.saveLink.shift preference.
* Otherwise if Alt, or Shift, or Ctrl (or Meta) is pressed then nothing happens.
* Otherwise the most recent browser is used for left clicks.
*/
function whereToOpenLink(e, ignoreButton, ignoreSave)
{
if (!e)
return "current";
var shift = e.shiftKey;
var ctrl = e.ctrlKey;
var meta = e.metaKey;
var alt = e.altKey;
// ignoreButton allows "middle-click paste" to use function without always opening in a new window.
var middle = !ignoreButton && e.button == 1;
if (meta || ctrl || middle) {
if (getBoolPref("browser.tabs.opentabfor.middleclick", true))
return shift ? "tabshifted" : "tab";
if (getBoolPref("middlemouse.openNewWindow", true))
return "window";
if (middle)
return null;
}
if (!ignoreSave) {
var saveKey = getBoolPref("ui.key.saveLink.shift", true) ? shift : alt;
if (saveKey)
return "save";
}
if (alt || shift || meta || ctrl)
return null;
return "current";
}
/* openUILinkIn opens a URL in a place specified by the parameter |where|.
*
* |where| can be:
* "current" current tab (if there aren't any browser windows, then in a new window instead)
* "tab" new tab (if there aren't any browser windows, then in a new window instead)
* "tabshifted" same as "tab" but in background if default is to select new tabs, and vice versa
* "window" new window
* "save" save to disk (with no filename hint!)
*
* allowThirdPartyFixup controls whether third party services such as Google's
* I'm Feeling Lucky are allowed to interpret this URL. This parameter may be
* undefined, which is treated as false.
*/
function openUILinkIn(url, where, allowThirdPartyFixup)
{
if (!where || !url)
return null;
if (where == "save") {
saveURL(url, null, null, true);
return null;
}
var w = getTopWin();
const nsIWebNavigation = Components.interfaces.nsIWebNavigation;
var flags = allowThirdPartyFixup ? nsIWebNavigation.LOAD_FLAGS_ALLOW_THIRD_PARTY_FIXUP :
nsIWebNavigation.LOAD_FLAGS_NONE;
if (!w || where == "window") {
return window.openDialog(getBrowserURL(), "_blank", "chrome,all,dialog=no", url,
null, null, flags);
}
var loadInBackground = getBoolPref("browser.tabs.loadInBackground", false);
switch (where) {
case "current":
w.loadURI(url, null, flags);
w.content.focus();
break;
case "tabshifted":
loadInBackground = !loadInBackground;
// fall through
case "tab":
var browser = w.getBrowser();
var tab = browser.addTab(url, null, null, false, flags);
if (!loadInBackground) {
browser.selectedTab = tab;
w.content.focus();
}
break;
}
return w;
}
// This opens the URLs contained in the given array in new tabs
// of the most recent window, creates a new window if necessary.
function openUILinkArrayIn(urlArray, where, allowThirdPartyFixup)
{
if (!where || !urlArray.length)
return null;
if (where == "save") {
for (var i = 0; i < urlArray.length; i++)
saveURL(urlArray[i], null, null, true);
return null;
}
var w = getTopWin();
const nsIWebNavigation = Components.interfaces.nsIWebNavigation;
var flags = allowThirdPartyFixup ? nsIWebNavigation.LOAD_FLAGS_ALLOW_THIRD_PARTY_FIXUP :
nsIWebNavigation.LOAD_FLAGS_NONE;
if (!w || where == "window") {
return window.openDialog(getBrowserURL(), "_blank", "chrome,all,dialog=no",
urlArray.join("\n"), // Pretend that we're a home page group
null, null, flags);
}
var loadInBackground = getBoolPref("browser.tabs.loadInBackground", false);
var browser = w.getBrowser();
switch (where) {
case "current":
w.loadURI(urlArray[0], null, flags);
w.content.focus();
break;
case "tabshifted":
loadInBackground = !loadInBackground;
// fall through
case "tab":
var tab = browser.addTab(urlArray[0], null, null, false, flags);
if (!loadInBackground) {
browser.selectedTab = tab;
w.content.focus();
}
}
for (var i = 1; i < urlArray.length; i++)
browser.addTab(urlArray[i], null, null, false, flags);
return w;
}
function subscribeToFeed(href, event) {
// Just load the feed in the content area to either subscribe or show the
// preview UI
var w = getTopWin();
var charset;
if (w) {
var browser = w.getBrowser();
charset = browser.characterSet;
}
else
// When calling this function without any open navigator window
charset = document.characterSet;
var feedURI = makeURI(href, charset);
// Use the feed scheme so X-Moz-Is-Feed will be set
// The value doesn't matter
if (/^https?/.test(feedURI.scheme))
href = "feed:" + href;
openUILink(href, event, false, true);
}
function subscribeToFeedMiddleClick(href, event) {
if (event.button == 1) {
this.subscribeToFeed(href, event);
// unlike for command events, we have to close the menus manually
closeMenus(event.target);
}
}