2007-03-22 10:30:00 -07:00
|
|
|
# -*- Mode: C++; tab-width: 4; 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) 1999
|
|
|
|
# the Initial Developer. All Rights Reserved.
|
|
|
|
#
|
|
|
|
# Contributor(s):
|
|
|
|
# Stephen Lamm <slamm@netscape.com>
|
|
|
|
# Robert John Churchill <rjc@netscape.com>
|
|
|
|
# David Hyatt <hyatt@mozilla.org>
|
|
|
|
# Christopher A. Aillon <christopher@aillon.com>
|
|
|
|
# Myk Melez <myk@mozilla.org>
|
|
|
|
# Pamela Greene <pamg.bugs@gmail.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 *****
|
|
|
|
|
|
|
|
/*
|
|
|
|
* No magic constructor behaviour, as is de rigeur for XPCOM.
|
|
|
|
* If you must perform some initialization, and it could possibly fail (even
|
|
|
|
* due to an out-of-memory condition), you should use an Init method, which
|
|
|
|
* can convey failure appropriately (thrown exception in JS,
|
|
|
|
* NS_FAILED(nsresult) return in C++).
|
|
|
|
*
|
|
|
|
* In JS, you can actually cheat, because a thrown exception will cause the
|
|
|
|
* CreateInstance call to fail in turn, but not all languages are so lucky.
|
|
|
|
* (Though ANSI C++ provides exceptions, they are verboten in Mozilla code
|
|
|
|
* for portability reasons -- and even when you're building completely
|
|
|
|
* platform-specific code, you can't throw across an XPCOM method boundary.)
|
|
|
|
*/
|
|
|
|
|
|
|
|
const DEBUG = false; /* set to false to suppress debug messages */
|
|
|
|
|
|
|
|
const SIDEBAR_CONTRACTID = "@mozilla.org/sidebar;1";
|
|
|
|
const SIDEBAR_CID = Components.ID("{22117140-9c6e-11d3-aaf1-00805f8a4905}");
|
|
|
|
const nsISupports = Components.interfaces.nsISupports;
|
|
|
|
const nsIFactory = Components.interfaces.nsIFactory;
|
|
|
|
const nsISidebar = Components.interfaces.nsISidebar;
|
|
|
|
const nsISidebarExternal = Components.interfaces.nsISidebarExternal;
|
|
|
|
const nsIClassInfo = Components.interfaces.nsIClassInfo;
|
|
|
|
|
|
|
|
// File extension for Sherlock search plugin description files
|
|
|
|
const SHERLOCK_FILE_EXT_REGEXP = /\.src$/i;
|
|
|
|
|
|
|
|
function nsSidebar()
|
|
|
|
{
|
|
|
|
const PROMPTSERVICE_CONTRACTID = "@mozilla.org/embedcomp/prompt-service;1";
|
|
|
|
const nsIPromptService = Components.interfaces.nsIPromptService;
|
|
|
|
this.promptService =
|
|
|
|
Components.classes[PROMPTSERVICE_CONTRACTID].getService(nsIPromptService);
|
|
|
|
|
|
|
|
const SEARCHSERVICE_CONTRACTID = "@mozilla.org/browser/search-service;1";
|
|
|
|
const nsIBrowserSearchService = Components.interfaces.nsIBrowserSearchService;
|
|
|
|
this.searchService =
|
|
|
|
Components.classes[SEARCHSERVICE_CONTRACTID].getService(nsIBrowserSearchService);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsSidebar.prototype.nc = "http://home.netscape.com/NC-rdf#";
|
|
|
|
|
|
|
|
function sidebarURLSecurityCheck(url)
|
|
|
|
{
|
|
|
|
if (!/^(https?:|ftp:)/i.test(url)) {
|
|
|
|
Components.utils.reportError("Invalid argument passed to window.sidebar.addPanel: Unsupported panel URL." );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* decorate prototype to provide ``class'' methods and property accessors */
|
|
|
|
nsSidebar.prototype.addPanel =
|
|
|
|
function (aTitle, aContentURL, aCustomizeURL)
|
|
|
|
{
|
|
|
|
debug("addPanel(" + aTitle + ", " + aContentURL + ", " +
|
|
|
|
aCustomizeURL + ")");
|
|
|
|
|
|
|
|
return this.addPanelInternal(aTitle, aContentURL, aCustomizeURL, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsSidebar.prototype.addPersistentPanel =
|
|
|
|
function(aTitle, aContentURL, aCustomizeURL)
|
|
|
|
{
|
|
|
|
debug("addPersistentPanel(" + aTitle + ", " + aContentURL + ", " +
|
|
|
|
aCustomizeURL + ")\n");
|
|
|
|
|
|
|
|
return this.addPanelInternal(aTitle, aContentURL, aCustomizeURL, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsSidebar.prototype.addPanelInternal =
|
|
|
|
function (aTitle, aContentURL, aCustomizeURL, aPersist)
|
|
|
|
{
|
|
|
|
var WINMEDSVC = Components.classes['@mozilla.org/appshell/window-mediator;1']
|
|
|
|
.getService(Components.interfaces.nsIWindowMediator);
|
|
|
|
var win = WINMEDSVC.getMostRecentWindow( "navigator:browser" );
|
|
|
|
|
|
|
|
if (!sidebarURLSecurityCheck(aContentURL))
|
|
|
|
return;
|
|
|
|
|
|
|
|
var uri = null;
|
|
|
|
var ioService = Components.classes["@mozilla.org/network/io-service;1"]
|
|
|
|
.getService(Components.interfaces.nsIIOService);
|
|
|
|
try {
|
|
|
|
uri = ioService.newURI(aContentURL, null, null);
|
|
|
|
}
|
|
|
|
catch(ex) { return; }
|
|
|
|
|
2008-03-10 10:49:31 -07:00
|
|
|
win.PlacesUtils.showMinimalAddBookmarkUI(uri, aTitle, null, null, true, true);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nsSidebar.prototype.validateSearchEngine =
|
|
|
|
function (engineURL, iconURL)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// Make sure we're using HTTP, HTTPS, or FTP.
|
|
|
|
if (! /^(https?|ftp):\/\//i.test(engineURL))
|
|
|
|
throw "Unsupported search engine URL";
|
|
|
|
|
|
|
|
// Make sure we're using HTTP, HTTPS, or FTP and refering to a
|
|
|
|
// .gif/.jpg/.jpeg/.png/.ico file for the icon.
|
|
|
|
if (iconURL &&
|
|
|
|
! /^(https?|ftp):\/\/.+\.(gif|jpg|jpeg|png|ico)$/i.test(iconURL))
|
|
|
|
throw "Unsupported search icon URL.";
|
|
|
|
}
|
|
|
|
catch(ex)
|
|
|
|
{
|
|
|
|
debug(ex);
|
|
|
|
Components.utils.reportError("Invalid argument passed to window.sidebar.addSearchEngine: " + ex);
|
|
|
|
|
|
|
|
var searchBundle = srGetStrBundle("chrome://browser/locale/search.properties");
|
|
|
|
var brandBundle = srGetStrBundle("chrome://branding/locale/brand.properties");
|
|
|
|
var brandName = brandBundle.GetStringFromName("brandShortName");
|
|
|
|
var title = searchBundle.GetStringFromName("error_invalid_engine_title");
|
|
|
|
var msg = searchBundle.formatStringFromName("error_invalid_engine_msg",
|
|
|
|
[brandName], 1);
|
|
|
|
var ww = Components.classes["@mozilla.org/embedcomp/window-watcher;1"].
|
|
|
|
getService(Components.interfaces.nsIWindowWatcher);
|
|
|
|
ww.getNewPrompter(null).alert(title, msg);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The suggestedTitle and suggestedCategory parameters are ignored, but remain
|
|
|
|
// for backward compatibility.
|
|
|
|
nsSidebar.prototype.addSearchEngine =
|
|
|
|
function (engineURL, iconURL, suggestedTitle, suggestedCategory)
|
|
|
|
{
|
|
|
|
debug("addSearchEngine(" + engineURL + ", " + iconURL + ", " +
|
|
|
|
suggestedCategory + ", " + suggestedTitle + ")");
|
|
|
|
|
|
|
|
if (!this.validateSearchEngine(engineURL, iconURL))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// OpenSearch files will likely be far more common than Sherlock files, and
|
|
|
|
// have less consistent suffixes, so we assume that ".src" is a Sherlock
|
|
|
|
// (text) file, and anything else is OpenSearch (XML).
|
|
|
|
var dataType;
|
|
|
|
if (SHERLOCK_FILE_EXT_REGEXP.test(engineURL))
|
|
|
|
dataType = Components.interfaces.nsISearchEngine.DATA_TEXT;
|
|
|
|
else
|
|
|
|
dataType = Components.interfaces.nsISearchEngine.DATA_XML;
|
|
|
|
|
|
|
|
this.searchService.addEngine(engineURL, dataType, iconURL, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// This function exists largely to implement window.external.AddSearchProvider(),
|
|
|
|
// to match other browsers' APIs. The capitalization, although nonstandard here,
|
|
|
|
// is therefore important.
|
|
|
|
nsSidebar.prototype.AddSearchProvider =
|
|
|
|
function (aDescriptionURL)
|
|
|
|
{
|
|
|
|
// Get the favicon URL for the current page, or our best guess at the current
|
|
|
|
// page since we don't have easy access to the active document. Most search
|
|
|
|
// engines will override this with an icon specified in the OpenSearch
|
|
|
|
// description anyway.
|
|
|
|
var WINMEDSVC = Components.classes['@mozilla.org/appshell/window-mediator;1']
|
|
|
|
.getService(Components.interfaces.nsIWindowMediator);
|
|
|
|
var win = WINMEDSVC.getMostRecentWindow("navigator:browser");
|
|
|
|
var browser = win.document.getElementById("content");
|
|
|
|
var iconURL = "";
|
|
|
|
if (browser.shouldLoadFavIcon(browser.selectedBrowser.currentURI))
|
|
|
|
iconURL = win.gProxyFavIcon.getAttribute("src");
|
|
|
|
|
|
|
|
if (!this.validateSearchEngine(aDescriptionURL, iconURL))
|
|
|
|
return;
|
|
|
|
|
|
|
|
const typeXML = Components.interfaces.nsISearchEngine.DATA_XML;
|
|
|
|
this.searchService.addEngine(aDescriptionURL, typeXML, iconURL, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// This function exists to implement window.external.IsSearchProviderInstalled(),
|
|
|
|
// for compatibility with other browsers. It will return an integer value
|
|
|
|
// indicating whether the given engine is installed for the current user.
|
|
|
|
// However, it is currently stubbed out due to security/privacy concerns
|
|
|
|
// stemming from difficulties in determining what domain issued the request.
|
|
|
|
// See bug 340604 and
|
|
|
|
// http://msdn.microsoft.com/workshop/author/dhtml/reference/methods/issearchproviderinstalled.asp .
|
|
|
|
// XXX Implement this!
|
|
|
|
nsSidebar.prototype.IsSearchProviderInstalled =
|
|
|
|
function (aSearchURL)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsSidebar.prototype.addMicrosummaryGenerator =
|
|
|
|
function (generatorURL)
|
|
|
|
{
|
|
|
|
debug("addMicrosummaryGenerator(" + generatorURL + ")");
|
|
|
|
|
2007-08-09 15:07:42 -07:00
|
|
|
if (!/^https?:/i.test(generatorURL))
|
|
|
|
return;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
var stringBundle = srGetStrBundle("chrome://browser/locale/sidebar/sidebar.properties");
|
|
|
|
var titleMessage = stringBundle.GetStringFromName("addMicsumGenConfirmTitle");
|
|
|
|
var dialogMessage = stringBundle.formatStringFromName("addMicsumGenConfirmText", [generatorURL], 1);
|
|
|
|
|
|
|
|
if (!this.promptService.confirm(null, titleMessage, dialogMessage))
|
|
|
|
return;
|
|
|
|
|
|
|
|
var ioService = Components.classes["@mozilla.org/network/io-service;1"].
|
|
|
|
getService(Components.interfaces.nsIIOService);
|
|
|
|
var generatorURI = ioService.newURI(generatorURL, null, null);
|
|
|
|
|
|
|
|
var microsummaryService = Components.classes["@mozilla.org/microsummary/service;1"].
|
|
|
|
getService(Components.interfaces.nsIMicrosummaryService);
|
|
|
|
if (microsummaryService)
|
|
|
|
microsummaryService.addGenerator(generatorURI);
|
|
|
|
}
|
|
|
|
|
|
|
|
// property of nsIClassInfo
|
|
|
|
nsSidebar.prototype.flags = nsIClassInfo.DOM_OBJECT;
|
|
|
|
|
|
|
|
// property of nsIClassInfo
|
|
|
|
nsSidebar.prototype.classDescription = "Sidebar";
|
|
|
|
|
|
|
|
// method of nsIClassInfo
|
|
|
|
nsSidebar.prototype.getInterfaces = function(count) {
|
|
|
|
var interfaceList = [nsISidebar, nsISidebarExternal, nsIClassInfo];
|
|
|
|
count.value = interfaceList.length;
|
|
|
|
return interfaceList;
|
|
|
|
}
|
|
|
|
|
|
|
|
// method of nsIClassInfo
|
|
|
|
nsSidebar.prototype.getHelperForLanguage = function(count) {return null;}
|
|
|
|
|
|
|
|
nsSidebar.prototype.QueryInterface =
|
|
|
|
function (iid) {
|
|
|
|
if (!iid.equals(nsISidebar) &&
|
|
|
|
!iid.equals(nsISidebarExternal) &&
|
|
|
|
!iid.equals(nsIClassInfo) &&
|
|
|
|
!iid.equals(nsISupports))
|
|
|
|
throw Components.results.NS_ERROR_NO_INTERFACE;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
var sidebarModule = new Object();
|
|
|
|
|
|
|
|
sidebarModule.registerSelf =
|
|
|
|
function (compMgr, fileSpec, location, type)
|
|
|
|
{
|
|
|
|
debug("registering (all right -- a JavaScript module!)");
|
|
|
|
compMgr = compMgr.QueryInterface(Components.interfaces.nsIComponentRegistrar);
|
|
|
|
|
|
|
|
compMgr.registerFactoryLocation(SIDEBAR_CID,
|
|
|
|
"Sidebar JS Component",
|
|
|
|
SIDEBAR_CONTRACTID,
|
|
|
|
fileSpec,
|
|
|
|
location,
|
|
|
|
type);
|
|
|
|
const CATMAN_CONTRACTID = "@mozilla.org/categorymanager;1";
|
|
|
|
const nsICategoryManager = Components.interfaces.nsICategoryManager;
|
|
|
|
var catman = Components.classes[CATMAN_CONTRACTID].
|
|
|
|
getService(nsICategoryManager);
|
|
|
|
|
|
|
|
const JAVASCRIPT_GLOBAL_PROPERTY_CATEGORY = "JavaScript global property";
|
|
|
|
catman.addCategoryEntry(JAVASCRIPT_GLOBAL_PROPERTY_CATEGORY,
|
|
|
|
"sidebar",
|
|
|
|
SIDEBAR_CONTRACTID,
|
|
|
|
true,
|
|
|
|
true);
|
|
|
|
|
|
|
|
catman.addCategoryEntry(JAVASCRIPT_GLOBAL_PROPERTY_CATEGORY,
|
|
|
|
"external",
|
|
|
|
SIDEBAR_CONTRACTID,
|
|
|
|
true,
|
|
|
|
true);
|
|
|
|
}
|
|
|
|
|
|
|
|
sidebarModule.getClassObject =
|
|
|
|
function (compMgr, cid, iid) {
|
|
|
|
if (!cid.equals(SIDEBAR_CID))
|
|
|
|
throw Components.results.NS_ERROR_NO_INTERFACE;
|
|
|
|
|
|
|
|
if (!iid.equals(Components.interfaces.nsIFactory))
|
|
|
|
throw Components.results.NS_ERROR_NOT_IMPLEMENTED;
|
|
|
|
|
|
|
|
return sidebarFactory;
|
|
|
|
}
|
|
|
|
|
|
|
|
sidebarModule.canUnload =
|
|
|
|
function(compMgr)
|
|
|
|
{
|
|
|
|
debug("Unloading component.");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* factory object */
|
|
|
|
var sidebarFactory = new Object();
|
|
|
|
|
|
|
|
sidebarFactory.createInstance =
|
|
|
|
function (outer, iid) {
|
|
|
|
debug("CI: " + iid);
|
|
|
|
if (outer != null)
|
|
|
|
throw Components.results.NS_ERROR_NO_AGGREGATION;
|
|
|
|
|
|
|
|
return (new nsSidebar()).QueryInterface(iid);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* entrypoint */
|
|
|
|
function NSGetModule(compMgr, fileSpec) {
|
|
|
|
return sidebarModule;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static functions */
|
|
|
|
if (DEBUG)
|
|
|
|
debug = function (s) { dump("-*- sidebar component: " + s + "\n"); }
|
|
|
|
else
|
|
|
|
debug = function (s) {}
|
|
|
|
|
|
|
|
var strBundleService = null;
|
|
|
|
function srGetStrBundle(path)
|
|
|
|
{
|
|
|
|
var strBundle = null;
|
|
|
|
if (!strBundleService) {
|
|
|
|
try {
|
|
|
|
strBundleService =
|
|
|
|
Components.classes["@mozilla.org/intl/stringbundle;1"].getService();
|
|
|
|
strBundleService =
|
|
|
|
strBundleService.QueryInterface(Components.interfaces.nsIStringBundleService);
|
|
|
|
} catch (ex) {
|
|
|
|
dump("\n--** strBundleService failed: " + ex + "\n");
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
strBundle = strBundleService.createBundle(path);
|
|
|
|
if (!strBundle) {
|
|
|
|
dump("\n--** strBundle createInstance failed **--\n");
|
|
|
|
}
|
|
|
|
return strBundle;
|
|
|
|
}
|