2014-03-27 01:03:42 -07:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
this.EXPORTED_SYMBOLS = ["DirectoryLinksProvider"];
|
|
|
|
|
|
|
|
const Ci = Components.interfaces;
|
|
|
|
const Cc = Components.classes;
|
|
|
|
const Cu = Components.utils;
|
2014-05-09 08:24:30 -07:00
|
|
|
const XMLHttpRequest =
|
|
|
|
Components.Constructor("@mozilla.org/xmlextras/xmlhttprequest;1", "nsIXMLHttpRequest");
|
2014-03-27 01:03:42 -07:00
|
|
|
|
|
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
|
|
|
|
|
|
XPCOMUtils.defineLazyModuleGetter(this, "NetUtil",
|
|
|
|
"resource://gre/modules/NetUtil.jsm");
|
2014-05-09 08:24:30 -07:00
|
|
|
XPCOMUtils.defineLazyModuleGetter(this, "OS",
|
|
|
|
"resource://gre/modules/osfile.jsm")
|
|
|
|
XPCOMUtils.defineLazyModuleGetter(this, "Promise",
|
|
|
|
"resource://gre/modules/Promise.jsm");
|
2014-03-27 01:03:42 -07:00
|
|
|
|
2014-05-09 08:24:30 -07:00
|
|
|
// The filename where directory links are stored locally
|
|
|
|
const DIRECTORY_LINKS_FILE = "directoryLinks.json";
|
2014-03-27 01:03:42 -07:00
|
|
|
|
|
|
|
// The preference that tells whether to match the OS locale
|
|
|
|
const PREF_MATCH_OS_LOCALE = "intl.locale.matchOS";
|
|
|
|
|
|
|
|
// The preference that tells what locale the user selected
|
|
|
|
const PREF_SELECTED_LOCALE = "general.useragent.locale";
|
|
|
|
|
|
|
|
// The preference that tells where to obtain directory links
|
2014-05-28 17:00:32 -07:00
|
|
|
const PREF_DIRECTORY_SOURCE = "browser.newtabpage.directorySource";
|
2014-03-27 01:03:42 -07:00
|
|
|
|
|
|
|
// The frecency of a directory link
|
|
|
|
const DIRECTORY_FRECENCY = 1000;
|
|
|
|
|
2014-03-31 01:51:22 -07:00
|
|
|
const LINK_TYPES = Object.freeze([
|
|
|
|
"sponsored",
|
|
|
|
"affiliate",
|
|
|
|
"organic",
|
|
|
|
]);
|
|
|
|
|
2014-03-27 01:03:42 -07:00
|
|
|
/**
|
|
|
|
* Singleton that serves as the provider of directory links.
|
|
|
|
* Directory links are a hard-coded set of links shown if a user's link
|
|
|
|
* inventory is empty.
|
|
|
|
*/
|
|
|
|
let DirectoryLinksProvider = {
|
|
|
|
|
|
|
|
__linksURL: null,
|
|
|
|
|
2014-05-28 17:00:32 -07:00
|
|
|
_observers: [],
|
2014-03-27 01:03:42 -07:00
|
|
|
|
2014-05-09 08:24:30 -07:00
|
|
|
get _observedPrefs() Object.freeze({
|
2014-03-27 01:03:42 -07:00
|
|
|
linksURL: PREF_DIRECTORY_SOURCE,
|
|
|
|
matchOSLocale: PREF_MATCH_OS_LOCALE,
|
|
|
|
prefSelectedLocale: PREF_SELECTED_LOCALE,
|
|
|
|
}),
|
|
|
|
|
|
|
|
get _linksURL() {
|
|
|
|
if (!this.__linksURL) {
|
|
|
|
try {
|
2014-05-09 08:24:30 -07:00
|
|
|
this.__linksURL = Services.prefs.getCharPref(this._observedPrefs["linksURL"]);
|
2014-03-27 01:03:42 -07:00
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
Cu.reportError("Error fetching directory links url from prefs: " + e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this.__linksURL;
|
|
|
|
},
|
|
|
|
|
2014-05-09 08:24:30 -07:00
|
|
|
/**
|
|
|
|
* Gets the currently selected locale for display.
|
|
|
|
* @return the selected locale or "en-US" if none is selected
|
|
|
|
*/
|
|
|
|
get locale() {
|
|
|
|
let matchOS;
|
|
|
|
try {
|
|
|
|
matchOS = Services.prefs.getBoolPref(PREF_MATCH_OS_LOCALE);
|
|
|
|
}
|
|
|
|
catch (e) {}
|
|
|
|
|
|
|
|
if (matchOS) {
|
|
|
|
return Services.locale.getLocaleComponentForUserAgent();
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
let locale = Services.prefs.getComplexValue(PREF_SELECTED_LOCALE,
|
|
|
|
Ci.nsIPrefLocalizedString);
|
|
|
|
if (locale) {
|
|
|
|
return locale.data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (e) {}
|
|
|
|
|
|
|
|
try {
|
|
|
|
return Services.prefs.getCharPref(PREF_SELECTED_LOCALE);
|
|
|
|
}
|
|
|
|
catch (e) {}
|
|
|
|
|
|
|
|
return "en-US";
|
|
|
|
},
|
|
|
|
|
2014-03-31 01:51:22 -07:00
|
|
|
get linkTypes() LINK_TYPES,
|
|
|
|
|
2014-03-27 01:03:42 -07:00
|
|
|
observe: function DirectoryLinksProvider_observe(aSubject, aTopic, aData) {
|
|
|
|
if (aTopic == "nsPref:changed") {
|
2014-05-09 08:24:30 -07:00
|
|
|
if (aData == this._observedPrefs["linksURL"]) {
|
2014-03-27 01:03:42 -07:00
|
|
|
delete this.__linksURL;
|
|
|
|
}
|
2014-05-28 17:00:32 -07:00
|
|
|
this._callObservers("onManyLinksChanged");
|
2014-03-27 01:03:42 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_addPrefsObserver: function DirectoryLinksProvider_addObserver() {
|
2014-05-09 08:24:30 -07:00
|
|
|
for (let pref in this._observedPrefs) {
|
|
|
|
let prefName = this._observedPrefs[pref];
|
2014-03-27 01:03:42 -07:00
|
|
|
Services.prefs.addObserver(prefName, this, false);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_removePrefsObserver: function DirectoryLinksProvider_removeObserver() {
|
2014-05-09 08:24:30 -07:00
|
|
|
for (let pref in this._observedPrefs) {
|
|
|
|
let prefName = this._observedPrefs[pref];
|
2014-03-27 01:03:42 -07:00
|
|
|
Services.prefs.removeObserver(prefName, this);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2014-05-28 16:59:30 -07:00
|
|
|
/**
|
|
|
|
* Fetches the current set of directory links.
|
|
|
|
* @param aCallback a callback that is provided a set of links.
|
|
|
|
*/
|
|
|
|
_fetchLinks: function DirectoryLinksProvider_fetchLinks(aCallback) {
|
|
|
|
try {
|
|
|
|
NetUtil.asyncFetch(this._linksURL, (aInputStream, aResult, aRequest) => {
|
|
|
|
let output;
|
|
|
|
if (Components.isSuccessCode(aResult)) {
|
|
|
|
try {
|
|
|
|
let json = NetUtil.readInputStreamToString(aInputStream,
|
|
|
|
aInputStream.available(),
|
|
|
|
{charset: "UTF-8"});
|
|
|
|
let locale = this.locale;
|
|
|
|
output = JSON.parse(json)[locale];
|
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
Cu.reportError(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Cu.reportError(new Error("the fetch of " + this._linksURL + "was unsuccessful"));
|
|
|
|
}
|
|
|
|
aCallback(output || []);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
Cu.reportError(e);
|
|
|
|
aCallback([]);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2014-05-09 08:24:30 -07:00
|
|
|
_fetchAndCacheLinks: function DirectoryLinksProvider_fetchAndCacheLinks(uri) {
|
|
|
|
let deferred = Promise.defer();
|
|
|
|
let xmlHttp = new XMLHttpRequest();
|
|
|
|
xmlHttp.overrideMimeType("application/json");
|
|
|
|
|
|
|
|
let self = this;
|
|
|
|
xmlHttp.onload = function(aResponse) {
|
|
|
|
let json = this.responseText;
|
|
|
|
if (this.status && this.status != 200) {
|
|
|
|
json = "{}";
|
|
|
|
}
|
2014-05-28 17:00:32 -07:00
|
|
|
let directoryLinksFilePath = OS.Path.join(OS.Constants.Path.localProfileDir, DIRECTORY_LINKS_FILE);
|
|
|
|
OS.File.writeAtomic(directoryLinksFilePath, json, {tmpPath: directoryLinksFilePath + ".tmp"})
|
2014-05-09 08:24:30 -07:00
|
|
|
.then(() => {
|
|
|
|
deferred.resolve();
|
2014-05-28 17:00:32 -07:00
|
|
|
self._callObservers("onManyLinksChanged");
|
2014-05-09 08:24:30 -07:00
|
|
|
},
|
|
|
|
() => {
|
|
|
|
deferred.reject("Error writing uri data in profD.");
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
xmlHttp.onerror = function(e) {
|
|
|
|
deferred.reject("Fetching " + uri + " results in error code: " + e.target.status);
|
|
|
|
};
|
|
|
|
|
|
|
|
try {
|
|
|
|
xmlHttp.open('POST', uri);
|
|
|
|
xmlHttp.send(JSON.stringify({ locale: this.locale }));
|
|
|
|
} catch (e) {
|
|
|
|
deferred.reject("Error fetching " + uri);
|
|
|
|
Cu.reportError(e);
|
|
|
|
}
|
|
|
|
return deferred.promise;
|
|
|
|
},
|
|
|
|
|
2014-03-27 01:03:42 -07:00
|
|
|
/**
|
|
|
|
* Gets the current set of directory links.
|
|
|
|
* @param aCallback The function that the array of links is passed to.
|
|
|
|
*/
|
|
|
|
getLinks: function DirectoryLinksProvider_getLinks(aCallback) {
|
2014-05-28 16:59:30 -07:00
|
|
|
this._fetchLinks(rawLinks => {
|
2014-03-27 01:03:42 -07:00
|
|
|
// all directory links have a frecency of DIRECTORY_FRECENCY
|
|
|
|
aCallback(rawLinks.map((link, position) => {
|
|
|
|
link.frecency = DIRECTORY_FRECENCY;
|
|
|
|
link.lastVisitDate = rawLinks.length - position;
|
|
|
|
return link;
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
init: function DirectoryLinksProvider_init() {
|
|
|
|
this._addPrefsObserver();
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the object to its pre-init state
|
|
|
|
*/
|
|
|
|
reset: function DirectoryLinksProvider_reset() {
|
|
|
|
delete this.__linksURL;
|
|
|
|
this._removePrefsObserver();
|
|
|
|
this._removeObservers();
|
|
|
|
},
|
|
|
|
|
|
|
|
addObserver: function DirectoryLinksProvider_addObserver(aObserver) {
|
2014-05-28 17:00:32 -07:00
|
|
|
this._observers.push(aObserver);
|
2014-03-27 01:03:42 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
_callObservers: function DirectoryLinksProvider__callObservers(aMethodName, aArg) {
|
|
|
|
for (let obs of this._observers) {
|
|
|
|
if (typeof(obs[aMethodName]) == "function") {
|
|
|
|
try {
|
|
|
|
obs[aMethodName](this, aArg);
|
|
|
|
} catch (err) {
|
|
|
|
Cu.reportError(err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_removeObservers: function() {
|
2014-05-28 17:00:32 -07:00
|
|
|
while (this._observers.length) {
|
|
|
|
this._observers.pop();
|
|
|
|
}
|
2014-03-27 01:03:42 -07:00
|
|
|
}
|
|
|
|
};
|