1116 lines
38 KiB
JavaScript
Executable File
1116 lines
38 KiB
JavaScript
Executable File
# -*- Mode: C++; tab-width: 8; 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 the Feed Writer.
|
|
#
|
|
# The Initial Developer of the Original Code is Google Inc.
|
|
# Portions created by the Initial Developer are Copyright (C) 2006
|
|
# the Initial Developer. All Rights Reserved.
|
|
#
|
|
# Contributor(s):
|
|
# Ben Goodger <beng@google.com>
|
|
# Jeff Walden <jwalden+code@mit.edu>
|
|
# Asaf Romano <mano@mozilla.com>
|
|
# Robert Sayre <sayrer@gmail.com>
|
|
# Michael Ventnor <m.ventnor@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 ***** */
|
|
|
|
const Cc = Components.classes;
|
|
const Ci = Components.interfaces;
|
|
const Cr = Components.results;
|
|
|
|
function LOG(str) {
|
|
var prefB =
|
|
Cc["@mozilla.org/preferences-service;1"].
|
|
getService(Ci.nsIPrefBranch);
|
|
|
|
var shouldLog = false;
|
|
try {
|
|
shouldLog = prefB.getBoolPref("feeds.log");
|
|
}
|
|
catch (ex) {
|
|
}
|
|
|
|
if (shouldLog)
|
|
dump("*** Feeds: " + str + "\n");
|
|
}
|
|
|
|
/**
|
|
* Wrapper function for nsIIOService::newURI.
|
|
* @param aURLSpec
|
|
* The URL string from which to create an nsIURI.
|
|
* @returns an nsIURI object, or null if the creation of the URI failed.
|
|
*/
|
|
function makeURI(aURLSpec, aCharset) {
|
|
var ios = Cc["@mozilla.org/network/io-service;1"].
|
|
getService(Ci.nsIIOService);
|
|
try {
|
|
return ios.newURI(aURLSpec, aCharset, null);
|
|
} catch (ex) { }
|
|
|
|
return null;
|
|
}
|
|
|
|
|
|
const XML_NS = "http://www.w3.org/XML/1998/namespace"
|
|
const HTML_NS = "http://www.w3.org/1999/xhtml";
|
|
const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
|
|
const TYPE_MAYBE_FEED = "application/vnd.mozilla.maybe.feed";
|
|
const URI_BUNDLE = "chrome://browser/locale/feeds/subscribe.properties";
|
|
|
|
const PREF_SELECTED_APP = "browser.feeds.handlers.application";
|
|
const PREF_SELECTED_WEB = "browser.feeds.handlers.webservice";
|
|
const PREF_SELECTED_ACTION = "browser.feeds.handler";
|
|
const PREF_SELECTED_READER = "browser.feeds.handler.default";
|
|
const PREF_SHOW_FIRST_RUN_UI = "browser.feeds.showFirstRunUI";
|
|
|
|
const FW_CLASSID = Components.ID("{49bb6593-3aff-4eb3-a068-2712c28bd58e}");
|
|
const FW_CLASSNAME = "Feed Writer";
|
|
const FW_CONTRACTID = "@mozilla.org/browser/feeds/result-writer;1";
|
|
|
|
const TITLE_ID = "feedTitleText";
|
|
const SUBTITLE_ID = "feedSubtitleText";
|
|
|
|
const NH_CONTRACTID = "@mozilla.org/browser/nav-history-service;1";
|
|
const FAV_CONTRACTID = "@mozilla.org/browser/favicon-service;1";
|
|
|
|
function FeedWriter() {
|
|
}
|
|
FeedWriter.prototype = {
|
|
_getPropertyAsBag: function FW__getPropertyAsBag(container, property) {
|
|
return container.fields.getProperty(property).
|
|
QueryInterface(Ci.nsIPropertyBag2);
|
|
},
|
|
|
|
_getPropertyAsString: function FW__getPropertyAsString(container, property) {
|
|
try {
|
|
return container.fields.getPropertyAsAString(property);
|
|
}
|
|
catch (e) {
|
|
}
|
|
return "";
|
|
},
|
|
|
|
_setContentText: function FW__setContentText(id, text) {
|
|
var element = this._document.getElementById(id);
|
|
while (element.hasChildNodes())
|
|
element.removeChild(element.firstChild);
|
|
element.appendChild(this._document.createTextNode(text));
|
|
},
|
|
|
|
/**
|
|
* Safely sets the href attribute on an anchor tag, providing the URI
|
|
* specified can be loaded according to rules.
|
|
* @param element
|
|
* The element to set a URI attribute on
|
|
* @param attribute
|
|
* The attribute of the element to set the URI to, e.g. href or src
|
|
* @param uri
|
|
* The URI spec to set as the href
|
|
*/
|
|
_safeSetURIAttribute:
|
|
function FW__safeSetURIAttribute(element, attribute, uri) {
|
|
var secman =
|
|
Cc["@mozilla.org/scriptsecuritymanager;1"].
|
|
getService(Ci.nsIScriptSecurityManager);
|
|
const flags = Ci.nsIScriptSecurityManager.DISALLOW_INHERIT_PRINCIPAL;
|
|
try {
|
|
secman.checkLoadURIStr(this._window.location.href, uri, flags);
|
|
// checkLoadURIStr will throw if the link URI should not be loaded per
|
|
// the rules specified in |flags|, so we'll never "linkify" the link...
|
|
element.setAttribute(attribute, uri);
|
|
}
|
|
catch (e) {
|
|
// Not allowed to load this link because secman.checkLoadURIStr threw
|
|
}
|
|
},
|
|
|
|
__faviconService: null,
|
|
get _faviconService() {
|
|
if (!this.__faviconService) {
|
|
this.__faviconService =
|
|
Cc[FAV_CONTRACTID].getService(Ci.nsIFaviconService);
|
|
}
|
|
|
|
return this.__faviconService;
|
|
},
|
|
|
|
__bundle: null,
|
|
get _bundle() {
|
|
if (!this.__bundle) {
|
|
this.__bundle = Cc["@mozilla.org/intl/stringbundle;1"].
|
|
getService(Ci.nsIStringBundleService).
|
|
createBundle(URI_BUNDLE);
|
|
}
|
|
return this.__bundle;
|
|
},
|
|
|
|
_getFormattedString: function FW__getFormattedString(key, params) {
|
|
return this._bundle.formatStringFromName(key, params, params.length);
|
|
},
|
|
|
|
_getString: function FW__getString(key) {
|
|
return this._bundle.GetStringFromName(key);
|
|
},
|
|
|
|
/* Magic helper methods to be used instead of xbl properties */
|
|
_getSelectedItemFromMenulist: function FW__getSelectedItemFromList(aList) {
|
|
var node = aList.firstChild.firstChild;
|
|
while (node) {
|
|
if (node.localName == "menuitem" && node.getAttribute("selected") == "true")
|
|
return node;
|
|
|
|
node = node.nextSibling;
|
|
}
|
|
|
|
return null;
|
|
},
|
|
|
|
_setCheckboxCheckedState: function FW__setCheckboxCheckedState(aCheckbox, aValue) {
|
|
// see checkbox.xml
|
|
var change = (aValue != (aCheckbox.getAttribute('checked') == 'true'));
|
|
if (aValue)
|
|
aCheckbox.setAttribute('checked', 'true');
|
|
else
|
|
aCheckbox.removeAttribute('checked');
|
|
|
|
if (change) {
|
|
var event = this._document.createEvent('Events');
|
|
event.initEvent('CheckboxStateChange', true, true);
|
|
aCheckbox.dispatchEvent(event);
|
|
}
|
|
},
|
|
|
|
// For setting and getting the file expando property, we need to keep a
|
|
// reference to explict XPCNativeWrappers around the associated menuitems
|
|
_selectedApplicationItemWrapped: null,
|
|
get selectedApplicationItemWrapped() {
|
|
if (!this._selectedApplicationItemWrapped) {
|
|
this._selectedApplicationItemWrapped =
|
|
XPCNativeWrapper(this._document.getElementById("selectedAppMenuItem"));
|
|
}
|
|
|
|
return this._selectedApplicationItemWrapped;
|
|
},
|
|
|
|
_defaultSystemReaderItemWrapped: null,
|
|
get defaultSystemReaderItemWrapped() {
|
|
if (!this._defaultSystemReaderItemWrapped) {
|
|
// Unlike the selected application item, this might not exist at all,
|
|
// see _initSubscriptionUI
|
|
var menuItem = this._document.getElementById("defaultHandlerMenuItem");
|
|
if (menuItem)
|
|
this._defaultSystemReaderItemWrapped = XPCNativeWrapper(menuItem);
|
|
}
|
|
|
|
return this._defaultSystemReaderItemWrapped;
|
|
},
|
|
|
|
/**
|
|
* Returns a date suitable for displaying in the feed preview.
|
|
* If the date cannot be parsed, the return value is "false".
|
|
* @param dateString
|
|
* A date as extracted from a feed entry. (entry.updated)
|
|
*/
|
|
_parseDate: function FW__parseDate(dateString) {
|
|
// Convert the date into the user's local time zone
|
|
dateObj = new Date(dateString);
|
|
|
|
// Make sure the date we're given is valid.
|
|
if (!dateObj.getTime())
|
|
return false;
|
|
|
|
var dateService = Cc["@mozilla.org/intl/scriptabledateformat;1"].
|
|
getService(Ci.nsIScriptableDateFormat);
|
|
return dateService.FormatDateTime("", dateService.dateFormatLong, dateService.timeFormatNoSeconds,
|
|
dateObj.getFullYear(), dateObj.getMonth()+1, dateObj.getDate(),
|
|
dateObj.getHours(), dateObj.getMinutes(), dateObj.getSeconds());
|
|
},
|
|
|
|
/**
|
|
* Writes the feed title into the preview document.
|
|
* @param container
|
|
* The feed container
|
|
*/
|
|
_setTitleText: function FW__setTitleText(container) {
|
|
if (container.title) {
|
|
this._setContentText(TITLE_ID, container.title.plainText());
|
|
this._document.title = container.title.plainText();
|
|
}
|
|
|
|
var feed = container.QueryInterface(Ci.nsIFeed);
|
|
if (feed && feed.subtitle)
|
|
this._setContentText(SUBTITLE_ID, container.subtitle.plainText());
|
|
},
|
|
|
|
/**
|
|
* Writes the title image into the preview document if one is present.
|
|
* @param container
|
|
* The feed container
|
|
*/
|
|
_setTitleImage: function FW__setTitleImage(container) {
|
|
try {
|
|
var parts = container.image;
|
|
|
|
// Set up the title image (supplied by the feed)
|
|
var feedTitleImage = this._document.getElementById("feedTitleImage");
|
|
this._safeSetURIAttribute(feedTitleImage, "src",
|
|
parts.getPropertyAsAString("url"));
|
|
|
|
// Set up the title image link
|
|
var feedTitleLink = this._document.getElementById("feedTitleLink");
|
|
|
|
var titleText =
|
|
this._getFormattedString("linkTitleTextFormat",
|
|
[parts.getPropertyAsAString("title")]);
|
|
feedTitleLink.setAttribute("title", titleText);
|
|
this._safeSetURIAttribute(feedTitleLink, "href",
|
|
parts.getPropertyAsAString("link"));
|
|
|
|
// Fix the margin on the main title, so that the image doesn't run over
|
|
// the underline
|
|
var feedTitleText = this._document.getElementById("feedTitleText");
|
|
var titleImageWidth = parseInt(parts.getPropertyAsAString("width")) + 15;
|
|
feedTitleText.style.marginRight = titleImageWidth + "px";
|
|
}
|
|
catch (e) {
|
|
LOG("Failed to set Title Image (this is benign): " + e);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Writes all entries contained in the feed.
|
|
* @param container
|
|
* The container of entries in the feed
|
|
*/
|
|
_writeFeedContent: function FW__writeFeedContent(container) {
|
|
// Build the actual feed content
|
|
var feedContent = this._document.getElementById("feedContent");
|
|
var feed = container.QueryInterface(Ci.nsIFeed);
|
|
|
|
for (var i = 0; i < feed.items.length; ++i) {
|
|
var entry = feed.items.queryElementAt(i, Ci.nsIFeedEntry);
|
|
entry.QueryInterface(Ci.nsIFeedContainer);
|
|
|
|
var entryContainer = this._document.createElementNS(HTML_NS, "div");
|
|
entryContainer.className = "entry";
|
|
|
|
// If the entry has a title, make it a link
|
|
if (entry.title) {
|
|
var a = this._document.createElementNS(HTML_NS, "a");
|
|
a.appendChild(this._document.createTextNode(entry.title.plainText()));
|
|
|
|
// Entries are not required to have links, so entry.link can be null.
|
|
if (entry.link)
|
|
this._safeSetURIAttribute(a, "href", entry.link.spec);
|
|
|
|
var title = this._document.createElementNS(HTML_NS, "h3");
|
|
title.appendChild(a);
|
|
entryContainer.appendChild(title);
|
|
|
|
var lastUpdated = this._parseDate(entry.updated);
|
|
if (lastUpdated) {
|
|
var dateDiv = this._document.createElementNS(HTML_NS, "div");
|
|
dateDiv.setAttribute("class", "lastUpdated");
|
|
title.appendChild(dateDiv);
|
|
dateDiv.textContent = lastUpdated;
|
|
}
|
|
}
|
|
|
|
var body = this._document.createElementNS(HTML_NS, "div");
|
|
var summary = entry.summary || entry.content;
|
|
var docFragment = null;
|
|
if (summary) {
|
|
|
|
if (summary.base)
|
|
body.setAttributeNS(XML_NS, "base", summary.base.spec);
|
|
else
|
|
LOG("no base?");
|
|
docFragment = summary.createDocumentFragment(body);
|
|
if (docFragment)
|
|
body.appendChild(docFragment);
|
|
|
|
// If the entry doesn't have a title, append a # permalink
|
|
// See http://scripting.com/rss.xml for an example
|
|
if (!entry.title && entry.link) {
|
|
var a = this._document.createElementNS(HTML_NS, "a");
|
|
a.appendChild(this._document.createTextNode("#"));
|
|
this._safeSetURIAttribute(a, "href", entry.link.spec);
|
|
body.appendChild(this._document.createTextNode(" "));
|
|
body.appendChild(a);
|
|
}
|
|
|
|
}
|
|
body.className = "feedEntryContent";
|
|
entryContainer.appendChild(body);
|
|
feedContent.appendChild(entryContainer);
|
|
var clearDiv = this._document.createElementNS(HTML_NS, "div");
|
|
clearDiv.style.clear = "both";
|
|
feedContent.appendChild(clearDiv);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Gets a valid nsIFeedContainer object from the parsed nsIFeedResult.
|
|
* Displays error information if there was one.
|
|
* @param result
|
|
* The parsed feed result
|
|
* @returns A valid nsIFeedContainer object containing the contents of
|
|
* the feed.
|
|
*/
|
|
_getContainer: function FW__getContainer(result) {
|
|
var feedService =
|
|
Cc["@mozilla.org/browser/feeds/result-service;1"].
|
|
getService(Ci.nsIFeedResultService);
|
|
|
|
try {
|
|
var result =
|
|
feedService.getFeedResult(this._getOriginalURI(this._window));
|
|
}
|
|
catch (e) {
|
|
LOG("Subscribe Preview: feed not available?!");
|
|
}
|
|
|
|
if (result.bozo) {
|
|
LOG("Subscribe Preview: feed result is bozo?!");
|
|
}
|
|
|
|
try {
|
|
var container = result.doc;
|
|
}
|
|
catch (e) {
|
|
LOG("Subscribe Preview: no result.doc? Why didn't the original reload?");
|
|
return null;
|
|
}
|
|
return container;
|
|
},
|
|
|
|
/**
|
|
* Get the human-readable display name of a file. This could be the
|
|
* application name.
|
|
* @param file
|
|
* A nsIFile to look up the name of
|
|
* @returns The display name of the application represented by the file.
|
|
*/
|
|
_getFileDisplayName: function FW__getFileDisplayName(file) {
|
|
#ifdef XP_WIN
|
|
if (file instanceof Ci.nsILocalFileWin) {
|
|
try {
|
|
return file.getVersionInfoField("FileDescription");
|
|
}
|
|
catch (e) {
|
|
}
|
|
}
|
|
#endif
|
|
#ifdef XP_MACOSX
|
|
var lfm = file.QueryInterface(Ci.nsILocalFileMac);
|
|
try {
|
|
return lfm.bundleDisplayName;
|
|
}
|
|
catch (e) {
|
|
// fall through to the file name
|
|
}
|
|
#endif
|
|
var ios =
|
|
Cc["@mozilla.org/network/io-service;1"].
|
|
getService(Ci.nsIIOService);
|
|
var url = ios.newFileURI(file).QueryInterface(Ci.nsIURL);
|
|
return url.fileName;
|
|
},
|
|
|
|
/**
|
|
* Get moz-icon url for a file
|
|
* @param file
|
|
* A nsIFile object for which the moz-icon:// is returned
|
|
* @returns moz-icon url of the given file as a string
|
|
*/
|
|
_getFileIconURL: function FW__getFileIconURL(file) {
|
|
var ios = Cc["@mozilla.org/network/io-service;1"].
|
|
getService(Components.interfaces.nsIIOService);
|
|
var fph = ios.getProtocolHandler("file")
|
|
.QueryInterface(Ci.nsIFileProtocolHandler);
|
|
var urlSpec = fph.getURLSpecFromFile(file);
|
|
return "moz-icon://" + urlSpec + "?size=16";
|
|
},
|
|
|
|
/**
|
|
* Helper method to set the selected application and system default
|
|
* reader menuitems details from a file object
|
|
* @param aMenuItem
|
|
* The menuitem on which the attributes should be set
|
|
* @param aFile
|
|
* The menuitem's associated file
|
|
*/
|
|
_initMenuItemWithFile: function(aMenuItem, aFile) {
|
|
aMenuItem.setAttribute("label", this._getFileDisplayName(aFile));
|
|
aMenuItem.setAttribute("image", this._getFileIconURL(aFile));
|
|
aMenuItem.file = aFile;
|
|
},
|
|
|
|
/**
|
|
* Displays a prompt from which the user may choose a (client) feed reader.
|
|
* @return - true if a feed reader was selected, false otherwise.
|
|
*/
|
|
_chooseClientApp: function FW__chooseClientApp() {
|
|
try {
|
|
var fp = Cc["@mozilla.org/filepicker;1"].createInstance(Ci.nsIFilePicker);
|
|
fp.init(this._window,
|
|
this._getString("chooseApplicationDialogTitle"),
|
|
Ci.nsIFilePicker.modeOpen);
|
|
fp.appendFilters(Ci.nsIFilePicker.filterApps);
|
|
|
|
if (fp.show() == Ci.nsIFilePicker.returnOK) {
|
|
var selectedApp = fp.file;
|
|
if (selectedApp) {
|
|
// XXXben - we need to compare this with the running instance executable
|
|
// just don't know how to do that via script...
|
|
// XXXmano TBD: can probably add this to nsIShellService
|
|
#ifdef XP_WIN
|
|
#expand if (fp.file.leafName != "__MOZ_APP_NAME__.exe") {
|
|
#else
|
|
#ifdef XP_MACOSX
|
|
#expand if (fp.file.leafName != "__MOZ_APP_DISPLAYNAME__.app") {
|
|
#else
|
|
#expand if (fp.file.leafName != "__MOZ_APP_NAME__-bin") {
|
|
#endif
|
|
#endif
|
|
var selectedAppMenuItem = this.selectedApplicationItemWrapped;
|
|
this._initMenuItemWithFile(selectedAppMenuItem, selectedApp);
|
|
|
|
// Show and select the selected application menuitem
|
|
selectedAppMenuItem.hidden = false;
|
|
selectedAppMenuItem.doCommand();
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch(ex) { }
|
|
|
|
return false;
|
|
},
|
|
|
|
_setAlwaysUseCheckedState: function FW__setAlwaysUseCheckedState() {
|
|
var checkbox = this._document.getElementById("alwaysUse");
|
|
if (checkbox) {
|
|
var alwaysUse = false;
|
|
try {
|
|
var prefs = Cc["@mozilla.org/preferences-service;1"].
|
|
getService(Ci.nsIPrefBranch);
|
|
if (prefs.getCharPref(PREF_SELECTED_ACTION) != "ask")
|
|
alwaysUse = true;
|
|
}
|
|
catch(ex) { }
|
|
this._setCheckboxCheckedState(checkbox, alwaysUse);
|
|
}
|
|
},
|
|
|
|
_setAlwaysUseLabel: function FW__setAlwaysUseLabel() {
|
|
var checkbox = this._document.getElementById("alwaysUse");
|
|
if (checkbox) {
|
|
var handlersMenuList = this._document.getElementById("handlersMenuList");
|
|
if (handlersMenuList) {
|
|
var handlerName = this._getSelectedItemFromMenulist(handlersMenuList)
|
|
.getAttribute("label");
|
|
checkbox.setAttribute("label", this._getFormattedString("alwaysUse", [handlerName]));
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* See nsIDOMEventListener
|
|
*/
|
|
handleEvent: function(event) {
|
|
// see comments in the write method
|
|
event = new XPCNativeWrapper(event);
|
|
if (event.target.ownerDocument != this._document) {
|
|
LOG("FeedWriter.handleEvent: Someone passed the feed writer as a listener to the events of another document!");
|
|
return;
|
|
}
|
|
|
|
if (event.type == "command") {
|
|
switch (event.target.id) {
|
|
case "subscribeButton":
|
|
this.subscribe();
|
|
break;
|
|
case "chooseApplicationMenuItem":
|
|
/* Bug 351263: Make sure to not steal focus if the "Choose
|
|
* Application" item is being selected with the keyboard. We do this
|
|
* by ignoring command events while the dropdown is closed (user
|
|
* arrowing through the combobox), but handling them while the
|
|
* combobox dropdown is open (user pressed enter when an item was
|
|
* selected). If we don't show the filepicker here, it will be shown
|
|
* when clicking "Subscribe Now".
|
|
*/
|
|
if (this._document.getElementById("handlersMenuList")
|
|
.getAttribute("open") == "true") {
|
|
if (!this._chooseClientApp()) {
|
|
// Select the (per-prefs) selected handler if no application was
|
|
// selected
|
|
this._setSelectedHandler();
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
this._setAlwaysUseLabel();
|
|
}
|
|
}
|
|
},
|
|
|
|
_setSelectedHandler: function FW__setSelectedHandler() {
|
|
var prefs =
|
|
Cc["@mozilla.org/preferences-service;1"].
|
|
getService(Ci.nsIPrefBranch);
|
|
|
|
var handler = "bookmarks";
|
|
try {
|
|
handler = prefs.getCharPref(PREF_SELECTED_READER);
|
|
}
|
|
catch (ex) { }
|
|
|
|
switch (handler) {
|
|
case "web": {
|
|
var handlersMenuList = this._document.getElementById("handlersMenuList");
|
|
if (handlersMenuList) {
|
|
var url = prefs.getComplexValue(PREF_SELECTED_WEB, Ci.nsISupportsString).data;
|
|
var handlers =
|
|
handlersMenuList.getElementsByAttribute("webhandlerurl", url);
|
|
if (handlers.length == 0) {
|
|
LOG("FeedWriter._setSelectedHandler: selected web handler isn't in the menulist")
|
|
return;
|
|
}
|
|
|
|
handlers[0].doCommand();
|
|
}
|
|
break;
|
|
}
|
|
case "client": {
|
|
var selectedAppMenuItem = this.selectedApplicationItemWrapped;
|
|
if (selectedAppMenuItem) {
|
|
try {
|
|
var selectedApp = prefs.getComplexValue(PREF_SELECTED_APP,
|
|
Ci.nsILocalFile);
|
|
} catch(ex) { }
|
|
|
|
if (selectedApp) {
|
|
this._initMenuItemWithFile(selectedAppMenuItem, selectedApp);
|
|
selectedAppMenuItem.hidden = false;
|
|
selectedAppMenuItem.doCommand();
|
|
|
|
// Only show the default reader menuitem if the default reader
|
|
// isn't the selected application
|
|
var defaultHandlerMenuItem = this.defaultSystemReaderItemWrapped;
|
|
if (defaultHandlerMenuItem) {
|
|
defaultHandlerMenuItem.hidden =
|
|
defaultHandlerMenuItem.file.path == selectedApp.path;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
case "bookmarks":
|
|
default: {
|
|
var liveBookmarksMenuItem =
|
|
this._document.getElementById("liveBookmarksMenuItem");
|
|
if (liveBookmarksMenuItem)
|
|
liveBookmarksMenuItem.doCommand();
|
|
}
|
|
}
|
|
},
|
|
|
|
_initSubscriptionUI: function FW__initSubscriptionUI() {
|
|
var handlersMenuPopup =
|
|
this._document.getElementById("handlersMenuPopup");
|
|
if (!handlersMenuPopup)
|
|
return;
|
|
|
|
// Last-selected application
|
|
var selectedApp;
|
|
menuItem = this._document.createElementNS(XUL_NS, "menuitem");
|
|
menuItem.id = "selectedAppMenuItem";
|
|
menuItem.className = "menuitem-iconic";
|
|
menuItem.setAttribute("handlerType", "client");
|
|
handlersMenuPopup.appendChild(menuItem);
|
|
|
|
var selectedApplicationItem = this.selectedApplicationItemWrapped;
|
|
try {
|
|
var prefs = Cc["@mozilla.org/preferences-service;1"].
|
|
getService(Ci.nsIPrefBranch);
|
|
selectedApp = prefs.getComplexValue(PREF_SELECTED_APP,
|
|
Ci.nsILocalFile);
|
|
|
|
if (selectedApp.exists()) {
|
|
this._initMenuItemWithFile(selectedApplicationItem, selectedApp);
|
|
}
|
|
else {
|
|
// Hide the menuitem if the last selected application doesn't exist
|
|
selectedApplicationItem.hidden = true;
|
|
}
|
|
}
|
|
catch(ex) {
|
|
// Hide the menuitem until an application is selected
|
|
selectedApplicationItem.hidden = true;
|
|
}
|
|
|
|
// List the default feed reader
|
|
var defaultReader = null;
|
|
try {
|
|
var defaultReader = Cc["@mozilla.org/browser/shell-service;1"].
|
|
getService(Ci.nsIShellService).defaultFeedReader;
|
|
menuItem = this._document.createElementNS(XUL_NS, "menuitem");
|
|
menuItem.id = "defaultHandlerMenuItem";
|
|
menuItem.className = "menuitem-iconic";
|
|
menuItem.setAttribute("handlerType", "client");
|
|
handlersMenuPopup.appendChild(menuItem);
|
|
|
|
var defaultSystemReaderItem = this.defaultSystemReaderItemWrapped;
|
|
this._initMenuItemWithFile(defaultSystemReaderItem, defaultReader);
|
|
|
|
// Hide the default reader item if it points to the same application
|
|
// as the last-selected application
|
|
if (selectedApp && selectedApp.path == defaultReader.path)
|
|
defaultSystemReaderItem.hidden = true;
|
|
}
|
|
catch(ex) { /* no default reader */ }
|
|
|
|
// "Choose Application..." menuitem
|
|
menuItem = this._document.createElementNS(XUL_NS, "menuitem");
|
|
menuItem.id = "chooseApplicationMenuItem";
|
|
menuItem.setAttribute("label", this._getString("chooseApplicationMenuItem"));
|
|
handlersMenuPopup.appendChild(menuItem);
|
|
|
|
// separator
|
|
handlersMenuPopup.appendChild(this._document.createElementNS(XUL_NS,
|
|
"menuseparator"));
|
|
|
|
var historySvc = Cc[NH_CONTRACTID].getService(Ci.nsINavHistoryService);
|
|
historySvc.addObserver(this, false);
|
|
|
|
// List of web handlers
|
|
var wccr =
|
|
Cc["@mozilla.org/embeddor.implemented/web-content-handler-registrar;1"].
|
|
getService(Ci.nsIWebContentConverterService);
|
|
var handlers = wccr.getContentHandlers(TYPE_MAYBE_FEED, {});
|
|
if (handlers.length != 0) {
|
|
for (var i = 0; i < handlers.length; ++i) {
|
|
menuItem = this._document.createElementNS(XUL_NS, "menuitem");
|
|
menuItem.className = "menuitem-iconic";
|
|
menuItem.setAttribute("label", handlers[i].name);
|
|
menuItem.setAttribute("handlerType", "web");
|
|
menuItem.setAttribute("webhandlerurl", handlers[i].uri);
|
|
handlersMenuPopup.appendChild(menuItem);
|
|
|
|
// For privacy reasons we cannot set the image attribute directly
|
|
// to the icon url, see Bug 358878
|
|
var uri = makeURI(handlers[i].uri);
|
|
if (!this._setFaviconForWebReader(uri, menuItem)) {
|
|
if (uri && /^https?/.test(uri.scheme)) {
|
|
var iconURL = makeURI(uri.prePath + "/favicon.ico");
|
|
this._faviconService.setAndLoadFaviconForPage(uri, iconURL, true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
this._setSelectedHandler();
|
|
|
|
// "Always use..." checkbox initial state
|
|
this._setAlwaysUseCheckedState();
|
|
this._setAlwaysUseLabel();
|
|
|
|
// We update the "Always use.." checkbox label whenever the selected item
|
|
// in the list is changed
|
|
handlersMenuPopup.addEventListener("command", this, false);
|
|
|
|
// Set up the "Subscribe Now" button
|
|
this._document
|
|
.getElementById("subscribeButton")
|
|
.addEventListener("command", this, false);
|
|
|
|
// first-run ui
|
|
var showFirstRunUI = true;
|
|
try {
|
|
showFirstRunUI = prefs.getBoolPref(PREF_SHOW_FIRST_RUN_UI);
|
|
}
|
|
catch (ex) { }
|
|
if (showFirstRunUI) {
|
|
var feedHeader = this._document.getElementById("feedHeader");
|
|
if (feedHeader)
|
|
feedHeader.setAttribute("firstrun", "true");
|
|
|
|
prefs.setBoolPref(PREF_SHOW_FIRST_RUN_UI, false);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Returns the original URI object of the feed and ensures that this
|
|
* component is only ever invoked from the preview document.
|
|
* @param aWindow
|
|
* The window of the document invoking the BrowserFeedWriter
|
|
*/
|
|
_getOriginalURI: function FW__getOriginalURI(aWindow) {
|
|
var chan =
|
|
aWindow.QueryInterface(Ci.nsIInterfaceRequestor).
|
|
getInterface(Ci.nsIWebNavigation).
|
|
QueryInterface(Ci.nsIDocShell).currentDocumentChannel;
|
|
|
|
const SUBSCRIBE_PAGE_URI = "chrome://browser/content/feeds/subscribe.xhtml";
|
|
var uri = makeURI(SUBSCRIBE_PAGE_URI);
|
|
var resolvedURI = Cc["@mozilla.org/chrome/chrome-registry;1"].
|
|
getService(Ci.nsIChromeRegistry).
|
|
convertChromeURL(uri);
|
|
|
|
if (resolvedURI.equals(chan.URI))
|
|
return chan.originalURI;
|
|
|
|
return null;
|
|
},
|
|
|
|
_window: null,
|
|
_document: null,
|
|
_feedURI: null,
|
|
|
|
/**
|
|
* See nsIFeedWriter
|
|
*/
|
|
init: function FW_init(aWindow) {
|
|
// Explicitly wrap |window| in an XPCNativeWrapper to make sure
|
|
// it's a real native object! This will throw an exception if we
|
|
// get a non-native object.
|
|
var window = new XPCNativeWrapper(aWindow);
|
|
this._feedURI = this._getOriginalURI(window);
|
|
if (!this._feedURI)
|
|
return;
|
|
|
|
this._window = window;
|
|
this._document = window.document;
|
|
|
|
LOG("Subscribe Preview: feed uri = " + this._window.location.href);
|
|
|
|
// Set up the subscription UI
|
|
this._initSubscriptionUI();
|
|
var prefs = Cc["@mozilla.org/preferences-service;1"].
|
|
getService(Ci.nsIPrefBranch2);
|
|
prefs.addObserver(PREF_SELECTED_ACTION, this, false);
|
|
prefs.addObserver(PREF_SELECTED_READER, this, false);
|
|
prefs.addObserver(PREF_SELECTED_WEB, this, false);
|
|
prefs.addObserver(PREF_SELECTED_APP, this, false);
|
|
},
|
|
|
|
/**
|
|
* See nsIFeedWriter
|
|
*/
|
|
writeContent: function FW_writeContent() {
|
|
if (!this._window)
|
|
return;
|
|
|
|
try {
|
|
// Set up the feed content
|
|
var container = this._getContainer();
|
|
if (!container)
|
|
return;
|
|
|
|
this._setTitleText(container);
|
|
this._setTitleImage(container);
|
|
this._writeFeedContent(container);
|
|
}
|
|
finally {
|
|
this._removeFeedFromCache();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* See nsIFeedWriter
|
|
*/
|
|
close: function FW_close() {
|
|
this._document = null;
|
|
this._window = null;
|
|
var prefs =
|
|
Cc["@mozilla.org/preferences-service;1"].
|
|
getService(Ci.nsIPrefBranch2);
|
|
prefs.removeObserver(PREF_SELECTED_ACTION, this);
|
|
prefs.removeObserver(PREF_SELECTED_READER, this);
|
|
prefs.removeObserver(PREF_SELECTED_WEB, this);
|
|
prefs.removeObserver(PREF_SELECTED_APP, this);
|
|
this._removeFeedFromCache();
|
|
|
|
var historySvc = Cc[NH_CONTRACTID].getService(Ci.nsINavHistoryService);
|
|
historySvc.removeObserver(this);
|
|
},
|
|
|
|
_removeFeedFromCache: function FW__removeFeedFromCache() {
|
|
if (this._feedURI) {
|
|
var feedService =
|
|
Cc["@mozilla.org/browser/feeds/result-service;1"].
|
|
getService(Ci.nsIFeedResultService);
|
|
feedService.removeFeedResult(this._feedURI);
|
|
this._feedURI = null;
|
|
}
|
|
},
|
|
|
|
subscribe: function FW_subscribe() {
|
|
// Subscribe to the feed using the selected handler and save prefs
|
|
var prefs =
|
|
Cc["@mozilla.org/preferences-service;1"].
|
|
getService(Ci.nsIPrefBranch);
|
|
var defaultHandler = "reader";
|
|
var useAsDefault = this._document.getElementById("alwaysUse")
|
|
.getAttribute("checked");
|
|
|
|
var handlersMenuList = this._document.getElementById("handlersMenuList");
|
|
var selectedItem = this._getSelectedItemFromMenulist(handlersMenuList);
|
|
|
|
// Show the file picker before subscribing if the
|
|
// choose application menuitem was choosen using the keyboard
|
|
if (selectedItem.id == "chooseApplicationMenuItem") {
|
|
if (!this._chooseClientApp())
|
|
return;
|
|
|
|
selectedItem = this._getSelectedItemFromMenulist(handlersMenuList);
|
|
}
|
|
|
|
if (selectedItem.hasAttribute("webhandlerurl")) {
|
|
var webURI = selectedItem.getAttribute("webhandlerurl");
|
|
prefs.setCharPref(PREF_SELECTED_READER, "web");
|
|
|
|
var supportsString = Cc["@mozilla.org/supports-string;1"].
|
|
createInstance(Ci.nsISupportsString);
|
|
supportsString.data = webURI;
|
|
prefs.setComplexValue(PREF_SELECTED_WEB, Ci.nsISupportsString,
|
|
supportsString);
|
|
|
|
var wccr =
|
|
Cc["@mozilla.org/embeddor.implemented/web-content-handler-registrar;1"].
|
|
getService(Ci.nsIWebContentConverterService);
|
|
var handler = wccr.getWebContentHandlerByURI(TYPE_MAYBE_FEED, webURI);
|
|
if (handler) {
|
|
if (useAsDefault)
|
|
wccr.setAutoHandler(TYPE_MAYBE_FEED, handler);
|
|
|
|
this._window.location.href =
|
|
handler.getHandlerURI(this._window.location.href);
|
|
}
|
|
}
|
|
else {
|
|
switch (selectedItem.id) {
|
|
case "selectedAppMenuItem":
|
|
prefs.setCharPref(PREF_SELECTED_READER, "client");
|
|
prefs.setComplexValue(PREF_SELECTED_APP, Ci.nsILocalFile,
|
|
this.selectedApplicationItemWrapped.file);
|
|
break;
|
|
case "defaultHandlerMenuItem":
|
|
prefs.setCharPref(PREF_SELECTED_READER, "client");
|
|
prefs.setComplexValue(PREF_SELECTED_APP, Ci.nsILocalFile,
|
|
this.defaultSystemReaderItemWrapped.file);
|
|
break;
|
|
case "liveBookmarksMenuItem":
|
|
defaultHandler = "bookmarks";
|
|
prefs.setCharPref(PREF_SELECTED_READER, "bookmarks");
|
|
break;
|
|
}
|
|
var feedService = Cc["@mozilla.org/browser/feeds/result-service;1"].
|
|
getService(Ci.nsIFeedResultService);
|
|
|
|
// Pull the title and subtitle out of the document
|
|
var feedTitle = this._document.getElementById(TITLE_ID).textContent;
|
|
var feedSubtitle =
|
|
this._document.getElementById(SUBTITLE_ID).textContent;
|
|
feedService.addToClientReader(this._window.location.href,
|
|
feedTitle, feedSubtitle);
|
|
}
|
|
|
|
// If "Always use..." is checked, we should set PREF_SELECTED_ACTION
|
|
// to either "reader" (If a web reader or if an application is selected),
|
|
// or to "bookmarks" (if the live bookmarks option is selected).
|
|
// Otherwise, we should set it to "ask"
|
|
if (useAsDefault)
|
|
prefs.setCharPref(PREF_SELECTED_ACTION, defaultHandler);
|
|
else
|
|
prefs.setCharPref(PREF_SELECTED_ACTION, "ask");
|
|
},
|
|
|
|
/**
|
|
* See nsIObserver
|
|
*/
|
|
observe: function FW_observe(subject, topic, data) {
|
|
if (!this._window) {
|
|
// this._window is null unless this.write was called with a trusted
|
|
// window object.
|
|
return;
|
|
}
|
|
|
|
if (topic == "nsPref:changed") {
|
|
switch (data) {
|
|
case PREF_SELECTED_READER:
|
|
case PREF_SELECTED_WEB:
|
|
case PREF_SELECTED_APP:
|
|
this._setSelectedHandler();
|
|
break;
|
|
case PREF_SELECTED_ACTION:
|
|
this._setAlwaysUseCheckedState();
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Sets the icon for the given web-reader item in the readers menu
|
|
* if the favicon-service has the necessary icon stored.
|
|
* @param aURI
|
|
* the reader URI.
|
|
* @param aMenuItem
|
|
* the reader item in the readers menulist.
|
|
* @return true if the icon was set, false otherwise.
|
|
*/
|
|
_setFaviconForWebReader:
|
|
function FW__setFaviconForWebReader(aURI, aMenuItem) {
|
|
var faviconsSvc = this._faviconService;
|
|
var faviconURL = null;
|
|
try {
|
|
faviconURL = faviconsSvc.getFaviconForPage(aURI);
|
|
}
|
|
catch(ex) { }
|
|
|
|
if (faviconURL) {
|
|
var mimeType = { };
|
|
var bytes = faviconsSvc.getFaviconData(faviconURL, mimeType,
|
|
{ /* dataLen */ });
|
|
if (bytes) {
|
|
var dataURI = "data:" + mimeType.value + ";" + "base64," +
|
|
btoa(String.fromCharCode.apply(null, bytes));
|
|
aMenuItem.setAttribute("image", dataURI);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* See nsINavHistoryService
|
|
*/
|
|
onPageChanged: function FW_onPageChanged(aURI, aWhat, aValue) {
|
|
if (aWhat == Ci.nsINavHistoryObserver.ATTRIBUTE_FAVICON) {
|
|
// Go through the readers menu and look for the corresponding
|
|
// reader menu-item for the page if any.
|
|
var spec = aURI.spec;
|
|
var handlersMenulist = this._document.getElementById("handlersMenuList");
|
|
var possibleHandlers = handlersMenulist.firstChild.childNodes;
|
|
for (var i=0; i < possibleHandlers.length ; i++) {
|
|
if (possibleHandlers[i].getAttribute("webhandlerurl") == spec) {
|
|
this._setFaviconForWebReader(aURI, possibleHandlers[i]);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
onBeginUpdateBatch: function() { },
|
|
onEndUpdateBatch: function() { },
|
|
onVisit: function() { },
|
|
onTitleChanged: function() { },
|
|
onDeleteURI: function() { },
|
|
onClearHistory: function() { },
|
|
onPageExpired: function() { },
|
|
|
|
/**
|
|
* See nsIClassInfo
|
|
*/
|
|
getInterfaces: function WCCR_getInterfaces(countRef) {
|
|
var interfaces =
|
|
[Ci.nsIFeedWriter, Ci.nsIClassInfo, Ci.nsISupports];
|
|
countRef.value = interfaces.length;
|
|
return interfaces;
|
|
},
|
|
getHelperForLanguage: function WCCR_getHelperForLanguage(language) {
|
|
return null;
|
|
},
|
|
contractID: FW_CONTRACTID,
|
|
classDescription: FW_CLASSNAME,
|
|
classID: FW_CLASSID,
|
|
implementationLanguage: Ci.nsIProgrammingLanguage.JAVASCRIPT,
|
|
flags: Ci.nsIClassInfo.DOM_OBJECT,
|
|
|
|
QueryInterface: function FW_QueryInterface(iid) {
|
|
if (iid.equals(Ci.nsIFeedWriter) ||
|
|
iid.equals(Ci.nsIClassInfo) ||
|
|
iid.equals(Ci.nsIDOMEventListener) ||
|
|
iid.equals(Ci.nsIObserver) ||
|
|
iid.equals(Ci.nsINavHistoryObserver) ||
|
|
iid.equals(Ci.nsISupports))
|
|
return this;
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
}
|
|
};
|
|
|
|
var Module = {
|
|
QueryInterface: function M_QueryInterface(iid) {
|
|
if (iid.equals(Ci.nsIModule) ||
|
|
iid.equals(Ci.nsISupports))
|
|
return this;
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
},
|
|
|
|
getClassObject: function M_getClassObject(cm, cid, iid) {
|
|
if (!iid.equals(Ci.nsIFactory))
|
|
throw Cr.NS_ERROR_NOT_IMPLEMENTED;
|
|
|
|
if (cid.equals(FW_CLASSID))
|
|
return new GenericComponentFactory(FeedWriter);
|
|
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
},
|
|
|
|
registerSelf: function M_registerSelf(cm, file, location, type) {
|
|
var cr = cm.QueryInterface(Ci.nsIComponentRegistrar);
|
|
|
|
cr.registerFactoryLocation(FW_CLASSID, FW_CLASSNAME, FW_CONTRACTID,
|
|
file, location, type);
|
|
|
|
var catman =
|
|
Cc["@mozilla.org/categorymanager;1"].
|
|
getService(Ci.nsICategoryManager);
|
|
catman.addCategoryEntry("JavaScript global constructor",
|
|
"BrowserFeedWriter", FW_CONTRACTID, true, true);
|
|
},
|
|
|
|
unregisterSelf: function M_unregisterSelf(cm, location, type) {
|
|
var cr = cm.QueryInterface(Ci.nsIComponentRegistrar);
|
|
cr.unregisterFactoryLocation(FW_CLASSID, location);
|
|
},
|
|
|
|
canUnload: function M_canUnload(cm) {
|
|
return true;
|
|
}
|
|
};
|
|
|
|
function NSGetModule(cm, file) {
|
|
return Module;
|
|
}
|
|
|
|
#include ../../../../toolkit/content/debug.js
|
|
#include GenericFactory.js
|