/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* ***** 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 * * 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; function toggleOfflineStatus() { var checkfunc; try { checkfunc = document.getElementById("offline-status").getAttribute('checkfunc'); } catch (ex) { checkfunc = null; } var ioService = Components.classes[kIOServiceProgID] .getService(Components.interfaces.nsIIOService); if (checkfunc) { if (!eval(checkfunc)) { // the pre-offline check function returned false, so don't go offline return; } } 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 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"); } else { networkProxyNo.removeAttribute("disabled"); networkProxyWpad.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]; 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; var bundle = srGetStrBundle("chrome://communicator/locale/utilityOverlay.properties"); panel.setAttribute("tooltiptext", bundle.GetStringFromName(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"); } var bundle = srGetStrBundle("chrome://communicator/locale/utilityOverlay.properties"); if (offline) { broadcaster.setAttribute("offline", "true"); broadcaster.setAttribute("checked", "true"); panel.removeAttribute("context"); panel.setAttribute("tooltiptext", bundle.GetStringFromName("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", bundle.GetStringFromName(onlineTooltip)); } } var goPrefWindow = 0; 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(containerID, paneURL, itemID) { var resizable; var pref = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefBranch); try { // We are resizable ONLY if in box debugging mode, because in // this special debug mode it is often impossible to see the // content of the debug panel in order to disable debug mode. resizable = pref.getBoolPref("xul.debug.box"); } catch (e) { resizable = false; } //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 { var resizability = resizable ? "yes" : "no"; var features = "chrome,titlebar,resizable=" + resizability; openDialog("chrome://communicator/content/pref/pref.xul","PrefWindow", features, paneURL, containerID, itemID); } } 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'); } } 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 ) openTopWin(url); } //No longer needed. Rip this out since we are using openTopWin function goHelpMenu( url ) { /* note that this chrome url should probably change to not have all of the navigator controls */ /* also, do we want to limit the number of help windows that can be spawned? */ window.openDialog( getBrowserURL(), "_blank", "chrome,all,dialog=no", 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 openTopWin( url ) { /* 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; // xlate the URL if necessary if (url.indexOf("urn:") == 0) { url = xlateURL(url); // does RDF urn expansion } // avoid loading "", since this loads a directory listing if (url == "") { url = "about:blank"; } var topWindowOfType = getTopWin(); if ( topWindowOfType ) { topWindowOfType.focus(); topWindowOfType.loadURI(url); return topWindowOfType; } return window.openDialog( getBrowserURL(), "_blank", "chrome,all,dialog=no", url ); } function goAboutDialog() { var defaultAboutState = false; try { var pref = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefBranch); defaultAboutState = pref.getBoolPref("browser.show_about_as_stupid_modal_window"); } catch(e) { defaultAboutState = false; } if( defaultAboutState ) window.openDialog("chrome://global/content/about.xul", "About", "modal,chrome,resizable=yes,height=450,width=550"); else window.openDialog( getBrowserURL(), "_blank", "chrome,all,dialog=no", 'about:' ); } // 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'); } // function that extracts the filename from a url function extractFileNameFromUrl(urlstr) { if (!urlstr) return null; // For "http://foo/bar/cheese.jpg", return "cheese.jpg". // For "imap://user@host.com:143/fetch>UID>/INBOX>951?part=1.2&type=image/gif&filename=foo.jpeg", return "foo.jpeg". // The 2nd url (ie, "imap://...") is generated for inline images by MimeInlineImage_parse_begin() in mimeiimg.cpp. var lastSlash = urlstr.slice(urlstr.lastIndexOf( "/" )+1); if (lastSlash) { var nameIndex = lastSlash.lastIndexOf( "filename=" ); if (nameIndex != -1) return (lastSlash.slice(nameIndex+9)); else return lastSlash; } return null; } // 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 whitespace. text = text.replace( /^\s+/, "" ); // Strip trailing whitespace. text = text.replace( /\s+$/, "" ); // Compress remaining whitespace. text = text.replace( /\s+/g, " " ); return text; } 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) { 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); } 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).replace(/^\s+|\s+$/g, ""); 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, "_"); } // autoscroll var gStartX; var gStartY; var gCurrX; var gCurrY; var gScrollingView; var gAutoScrollTimer; var gIgnoreMouseEvents; var gAutoScrollPopup = null; var gAccumulatedScrollErrorX; var gAccumulatedScrollErrorY; function startScrolling(event) { var pref = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefBranch); if (gScrollingView || event.button != 1) return; if (!pref.getBoolPref("general.autoScroll")) return; if (gScrollingView || event.button != 1) return; if (event.originalTarget instanceof XULElement && ((event.originalTarget.localName == "thumb") || (event.originalTarget.localName == "slider") || (event.originalTarget.localName == "scrollbarbutton"))) return; if (!gAutoScrollPopup) { const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"; gAutoScrollPopup = document.createElementNS(XUL_NS, "popup"); gAutoScrollPopup.id = "autoscroller"; if (screen.colorDepth > 8) { // need high colour for transparency // Exclude second-rate platforms gAutoScrollPopup.setAttribute("transparent", !/BeOS|Mac|OS\/2|Photon/.test(navigator.appVersion)); // Enable translucency on W2K+ (but not WinNT4.0) gAutoScrollPopup.setAttribute("translucent", /^Windows NT /.test(navigator.oscpu)); } gAutoScrollPopup.addEventListener("popuphidden", stopScrolling, true); document.documentElement.appendChild(gAutoScrollPopup); } gScrollingView = event.originalTarget.ownerDocument.defaultView; var scrollingDir; if (gScrollingView.scrollMaxX > 0) { gAutoScrollPopup.setAttribute("scrolldir", gScrollingView.scrollMaxY > 0 ? "NSEW" : "EW"); } else if (gScrollingView.scrollMaxY > 0) { gAutoScrollPopup.setAttribute("scrolldir", "NS"); } else { gScrollingView = null; // abort scrolling return; } document.popupNode = null; gAutoScrollPopup.showPopup(document.documentElement, event.screenX, event.screenY, "popup", null, null); gIgnoreMouseEvents = true; gStartX = event.screenX; gStartY = event.screenY; gCurrX = event.screenX; gCurrY = event.screenY; addEventListener('mousemove', handleMouseMove, true); addEventListener('mouseup', handleMouseUpDown, true); addEventListener('mousedown', handleMouseUpDown, true); addEventListener('contextmenu', handleMouseUpDown, true); gAutoScrollTimer = setInterval(autoScrollLoop, 10); gAccumulatedScrollErrorX = 0; gAccumulatedScrollErrorY = 0; } function handleMouseMove(event) { gCurrX = event.screenX; gCurrY = event.screenY; } function roundToZero(num) { if (num > 0) return Math.floor(num); return Math.ceil(num); } function accelerate(curr, start) { var speed = 12; var val = (curr - start) / speed; if (val > 1) return val * Math.sqrt(val) - 1; if (val < -1) return val * Math.sqrt(-val) + 1; return 0; } function autoScrollLoop() { var x = accelerate(gCurrX, gStartX); var y = accelerate(gCurrY, gStartY); if (x || y) gIgnoreMouseEvents = false; var desiredScrollX = gAccumulatedScrollErrorX + x; var actualScrollX = roundToZero(desiredScrollX); gAccumulatedScrollErrorX = (desiredScrollX - actualScrollX); var desiredScrollY = gAccumulatedScrollErrorY + y; var actualScrollY = roundToZero(desiredScrollY); gAccumulatedScrollErrorY = (desiredScrollY - actualScrollY); gScrollingView.scrollBy(actualScrollX, actualScrollY); } function handleMouseUpDown(event) { if (!gIgnoreMouseEvents) gAutoScrollPopup.hidePopup(); gIgnoreMouseEvents = false; } function stopScrolling() { if (gScrollingView) { gScrollingView = null; removeEventListener('mousemove', handleMouseMove, true); removeEventListener('mousedown', handleMouseUpDown, true); removeEventListener('mouseup', handleMouseUpDown, true); removeEventListener('contextmenu', handleMouseUpDown, true); clearInterval(gAutoScrollTimer); } }