diff --git a/README.md b/README.md
new file mode 100644
index 0000000..307b966
--- /dev/null
+++ b/README.md
@@ -0,0 +1,2 @@
+# Search Shield Study
+A shield Study add-on testing the search experience.
diff --git a/chrome.manifest b/chrome.manifest
new file mode 100644
index 0000000..1ebdf15
--- /dev/null
+++ b/chrome.manifest
@@ -0,0 +1 @@
+content unified-urlbar chrome/content/
diff --git a/code/content/Panel.jsm b/chrome/content/Panel.jsm
old mode 100755
new mode 100644
similarity index 80%
rename from code/content/Panel.jsm
rename to chrome/content/Panel.jsm
index e92b5a4..f92f7c0
--- a/code/content/Panel.jsm
+++ b/chrome/content/Panel.jsm
@@ -1,17 +1,25 @@
+"use strict";
+
this.EXPORTED_SYMBOLS = [
- "Panel",
+ "Panel"
];
const EXISTING_FOOTER_ID = "urlbar-search-footer";
const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
+Cu.import("resource://gre/modules/XPCOMUtils.jsm");
Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("chrome://unified-urlbar/content/Telemetry.jsm");
+Cu.import("resource://gre/modules/Preferences.jsm");
+
+XPCOMUtils.defineLazyModuleGetter(this, "UnifiedUrlbar",
+ "chrome://unified-urlbar/content/UnifiedUrlbar.jsm");
this.Panel = function (panelElt) {
this.panelElement = panelElt;
this._initPanelElement();
+ this.panelElement.addEventListener("popupshowing", this);
+ this.panelElement.addEventListener("popuphiding", this);
this._initKeyHandler();
this.urlbar.addEventListener("input", this);
};
@@ -39,11 +47,105 @@ this.Panel.prototype = {
this.urlbar.removeEventListener("input", this);
this.panelElement.removeEventListener("popupshowing", this);
+ this.panelElement.removeEventListener("popuphiding", this);
this.footer.remove();
if (this._existingFooter) {
this._existingFooterParent.appendChild(this._existingFooter);
}
+
+ this._uninitTipElement();
+ },
+
+ _shouldShowHint() {
+ const SHOW_TIP_DEFAULT_COUNT = 5;
+ let tipShownCount = SHOW_TIP_DEFAULT_COUNT;
+ if (Preferences.has("browser.urlbar.experiment.unified-urlbar.tipShownCount")) {
+ tipShownCount = Preferences.get("browser.urlbar.experiment.unified-urlbar.tipShownCount",
+ SHOW_TIP_DEFAULT_COUNT);
+ }
+
+ if (tipShownCount > 0) {
+ Preferences.set("browser.urlbar.experiment.unified-urlbar.tipShownCount",
+ --tipShownCount);
+ }
+
+ return tipShownCount > 0 &&
+ Preferences.get("browser.urlbar.suggest.searches", false);
+ },
+
+ _ensureTipElement() {
+ if (this.tipContainer)
+ return;
+
+ let tipContainer = this.document.createElementNS(XUL_NS, "hbox");
+ tipContainer.id = "urlbar-tip-container";
+ tipContainer.setAttribute("flex", "1");
+ tipContainer.setAttribute("align", "center");
+ this.tipContainer = tipContainer;
+
+ let icon = this.document.createElementNS(XUL_NS, "image");
+ icon.className = "ac-site-icon";
+ tipContainer.appendChild(icon);
+ this.tipIcon = icon;
+
+ let titleBox = this.document.createElementNS(XUL_NS, "hbox");
+ titleBox.id = "urlbar-tip-title";
+ tipContainer.appendChild(titleBox);
+
+ let title = this.document.createElementNS(XUL_NS, "description");
+ title.className = "ac-title-text";
+ title.textContent = "Firefox";
+ titleBox.appendChild(title);
+
+ let tipBox = this.document.createElementNS(XUL_NS, "hbox");
+ tipBox.id = "urlbar-tip-box";
+ tipContainer.setAttribute("align", "center");
+ tipContainer.appendChild(tipBox);
+
+ let tip = this.document.createElementNS(XUL_NS, "description");
+ tip.innerHTML = `💡Tip: Results with a magnifying glass are search suggestions. They might be what you're looking for!😄`;
+ tipBox.appendChild(tip);
+
+ let notification = this.panelElement.searchSuggestionsNotification;
+ for (let child of notification.childNodes) {
+ child.collapsed = true;
+ }
+ notification.setAttribute("tip", "true");
+ notification.appendChild(tipContainer);
+ },
+
+ _uninitTipElement() {
+ if (this.tipContainer) {
+ this.tipContainer.remove();
+ delete this.tipContainer;
+ }
+ let notification = this.panelElement.searchSuggestionsNotification;
+ notification.removeAttribute("tip");
+ for (let child of notification.childNodes) {
+ child.collapsed = false;
+ }
+ this.document.getAnonymousElementByAttribute(this.panelElement, "anonid", "search-suggestions-notification")
+ .style.visibility = "collapse";
+ },
+
+ _updateTip() {
+ if (!this._shouldShowHint()) {
+ if (this.tipContainer) {
+ this._uninitTipElement();
+ }
+ return;
+ }
+
+ this._ensureTipElement();
+
+ let iconStart = this.panelElement.siteIconStart;
+ if (iconStart) {
+ this.tipIcon.style.marginInlineStart = iconStart + "px";
+ }
+ this.document.getAnonymousElementByAttribute(this.panelElement, "anonid", "search-suggestions-notification")
+ .style.visibility = "visible";
+ this.tipContainer.setAttribute("animate", "true");
},
_initPanelElement() {
@@ -71,8 +173,6 @@ this.Panel.prototype = {
hbox.appendChild(this.settingsButton);
this.panelElement.appendChild(footer);
-
- this.panelElement.addEventListener("popupshowing", this);
},
_makeFooter() {
@@ -260,9 +360,6 @@ this.Panel.prototype = {
return;
}
- if (this.selectedButton != this.settingsButton) {
- Telemetry.incrementValue("oneOffButtonSelectedByKeypress");
- }
event.preventDefault();
},
@@ -279,7 +376,7 @@ this.Panel.prototype = {
return;
}
if (this.selectedButton == this.settingsButton) {
- Telemetry.incrementValue("searchSettingsClicked");
+ UnifiedUrlbar.reportTelemetryValue("searchSettingsClicked");
this.window.openPreferences("paneSearch");
} else {
this._doSearchFromButton(this.selectedButton, event);
@@ -300,6 +397,13 @@ this.Panel.prototype = {
this.selectedButton = null;
this._buildButtonList();
this._updateHeader();
+ this._updateTip();
+ },
+
+ _onPopuphiding(event) {
+ if (this.tipContainer) {
+ this.tipContainer.removeAttribute("animate");
+ }
},
_onMouseover(event) {
@@ -312,7 +416,6 @@ this.Panel.prototype = {
!target.classList.contains("dummy")) ||
target.classList.contains("addengine-item") ||
target.classList.contains("search-setting-button")) {
- Telemetry.incrementValue("oneOffButtonSelectedByMouseover");
this.selectedButton = target;
}
},
@@ -344,9 +447,9 @@ this.Panel.prototype = {
let win = button.ownerDocument.defaultView;
if (event instanceof win.KeyboardEvent) {
- Telemetry.incrementValue("searchByReturnKeyOnOneOffButton", { engine } );
+ UnifiedUrlbar.reportTelemetryValue("searchByReturnKeyOnOneOffButton", { engine } );
} else if (event instanceof win.MouseEvent) {
- Telemetry.incrementValue("searchByClickOnOneOffButton", { engine });
+ UnifiedUrlbar.reportTelemetryValue("searchByClickOnOneOffButton", { engine });
}
let query = this.urlbar.controller.searchString;
@@ -367,8 +470,6 @@ this.Panel.prototype = {
this.buttonList.firstChild.remove();
}
- let Preferences =
- Cu.import("resource://gre/modules/Preferences.jsm", {}).Preferences;
let pref = Preferences.get("browser.search.hiddenOneOffs");
let hiddenList = pref ? pref.split(",") : [];
@@ -448,14 +549,6 @@ this.Panel.prototype = {
this.buttonList.appendChild(button);
}
-
- // Add a data point for the engine count if it's changed (or this is the
- // first time reaching here).
- this._engineCount = this._engineCount || 0;
- if (engines.length != this._engineCount) {
- this._engineCount = engines.length;
- Telemetry.setValue("engineCount", engines.length);
- }
},
get numButtons() {
diff --git a/chrome/content/UnifiedUrlbar.jsm b/chrome/content/UnifiedUrlbar.jsm
new file mode 100644
index 0000000..9645b84
--- /dev/null
+++ b/chrome/content/UnifiedUrlbar.jsm
@@ -0,0 +1,398 @@
+"use strict";
+
+this.EXPORTED_SYMBOLS = [
+ "UnifiedUrlbar"
+];
+
+const STYLE_URL = "chrome://unified-urlbar/content/style.css";
+const SEARCH_BAR_WIDGET_ID = "search-container";
+const XHTML_NS = "http://www.w3.org/1999/xhtml";
+
+const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
+Cu.import("resource://gre/modules/XPCOMUtils.jsm");
+Cu.import("resource://gre/modules/Services.jsm");
+Cu.import("resource://gre/modules/Preferences.jsm");
+Cu.import("resource:///modules/CustomizableUI.jsm");
+Cu.import("resource:///modules/BrowserUITelemetry.jsm");
+Cu.import("chrome://unified-urlbar/content/Panel.jsm");
+
+var gOptions = {
+ forceSearchSuggestions: true,
+ addOneOffAndOnboarding: true,
+ removeSearchbar: true,
+};
+
+var gBrowsers = null;
+var gForcedSuggestions = false;
+var gDisabledOneOffs = false;
+
+this.UnifiedUrlbar = Object.freeze({
+ isUserEligible() {
+ // Users having a version of Firefox that includes the NEW one-off buttons
+ // implementation are not eligible for this experiment.
+ let vc = Cc["@mozilla.org/xpcom/version-comparator;1"]
+ .getService(Ci.nsIVersionComparator);
+ let appInfo = Cc["@mozilla.org/xre/app-info;1"]
+ .getService(Ci.nsIXULAppInfo);
+ if (vc.compare(appInfo.version, "51") >= 0) {
+ return false;
+ }
+
+ // Exclude users who already removed the search bar from the UI.
+ if (CustomizableUI.getPlacementOfWidget("search-container") === null) {
+ return false;
+ }
+
+ return true;
+ },
+
+ init(options = gOptions) {
+ if (gBrowsers) {
+ Cu.reportError("UnifiedUrlbar.init() was invoked multiple times?");
+ return;
+ }
+
+ gOptions = options;
+
+ if (("forceSearchSuggestions" in gOptions) && gOptions.forceSearchSuggestions) {
+ forceSearchSuggestions();
+ }
+ if (("addOneOffAndOnboarding" in gOptions) && gOptions.addOneOffAndOnboarding) {
+ disableNewOneOffs();
+ }
+
+ Services.obs.addObserver(this, "autocomplete-did-enter-text", false);
+
+ gBrowsers = new Set();
+ getBrowserWindows();
+ Services.ww.registerNotification(this);
+ },
+
+ destroy() {
+ if (!gBrowsers) {
+ Cu.reportError("UnifiedUrlbar.destroy() was invoked multiple times?");
+ return;
+ }
+
+ restoreSearchSuggestions();
+ restoreNewOneOffs();
+
+ Services.obs.removeObserver(this, "autocomplete-did-enter-text", false);
+
+ Services.ww.unregisterNotification(this);
+ for (let browser of gBrowsers) {
+ browser.destroy();
+ }
+ gBrowsers.clear();
+ gBrowsers = null;
+ },
+
+ observe(subject, topic, data) {
+ if (topic == "autocomplete-did-enter-text") {
+ trackAutocompleteEnter(subject.QueryInterface(Ci.nsIAutoCompleteInput));
+ return;
+ }
+
+ let win = subject.QueryInterface(Ci.nsIDOMWindow);
+ if (!win) {
+ return;
+ }
+
+ if (topic == "domwindowopened") {
+ whenWindowLoaded(win, () => {
+ if (isValidBrowserWindow(win)) {
+ gBrowsers.add(new Browser(win));
+ }
+ });
+ } else if (topic == "domwindowclosed") {
+ for (let browser of gBrowsers) {
+ if (browser.window == win) {
+ browser.destroy();
+ gBrowsers.delete(browser);
+ break;
+ }
+ }
+ }
+ },
+
+ reportTelemetryValue(key, optionalData={}) {
+ reportTelemetryValue(key, optionalData);
+ }
+});
+
+function Browser(win, branch) {
+ this.window = win;
+
+ // Per window toggles.
+ if (("removeSearchbar" in gOptions) && gOptions.removeSearchbar) {
+ this._removeSearchBar();
+ }
+ if (("addOneOffAndOnboarding" in gOptions) && gOptions.addOneOffAndOnboarding) {
+ this._initPanel();
+ }
+
+ win.gBrowser.addProgressListener(this);
+}
+
+Browser.prototype = {
+ get document() {
+ return this.window.document;
+ },
+
+ destroy() {
+ this.window.gBrowser.removeProgressListener(this);
+
+ if (this._styleLink) {
+ this._styleLink.remove();
+ delete this._styleLink;
+ }
+ if (this._panel) {
+ this._panel.destroy();
+ }
+ if (this._searchbarPlacement) {
+ CustomizableUI.addWidgetToArea(SEARCH_BAR_WIDGET_ID,
+ this._searchbarPlacement.area,
+ this._searchbarPlacement.position);
+ }
+ },
+
+ _initPanel() {
+ // Inject style.
+ this._styleLink = this.document.createElementNS(XHTML_NS, "link");
+ this._styleLink.setAttribute("href", STYLE_URL);
+ this._styleLink.setAttribute("rel", "stylesheet");
+ this.document.documentElement.appendChild(this._styleLink);
+
+ let elt = this.document.getElementById("PopupAutoCompleteRichResult");
+ this._panel = new Panel(this.window.gURLBar.popup);
+ },
+
+ _removeSearchBar() {
+ this._searchbarPlacement =
+ CustomizableUI.getPlacementOfWidget(SEARCH_BAR_WIDGET_ID);
+ CustomizableUI.removeWidgetFromArea(SEARCH_BAR_WIDGET_ID);
+ },
+
+ onLocationChange(webProgress, request, uri, flags) {
+ try {
+ if (webProgress.isTopLevel && uri.host) {
+ let host = uri.host.replace(/^www./, "").replace(/^search./, "");
+ if (gEngines.has(host)) {
+ let rv = Services.search.parseSubmissionURL(uri.spec);
+ // HACK: try to not count result pages we generated and subpages.
+ // This is tricky and working until the engines keep same param names.
+ if (rv.engine &&
+ !["hspart=mozilla", // Yahoo tracking
+ "&b=", // Yahoo paging
+ "client=firefox", // Google tracking
+ "&start=", // Google paging
+ "pc=MOZI", // Bing tracking
+ "&first=", // Bing paging
+ ].some(str => uri.path.includes(str))) {
+ reportTelemetryValue("userVisitedEngineResult",
+ { engine: rv.engine });
+ }
+ }
+ }
+ } catch (ex) {}
+ },
+ onProgressChange() {},
+ onSecurityChange() {},
+ onStateChange(webProgress, request, flags, status) {
+ try {
+ if (webProgress.isTopLevel &&
+ flags & Ci.nsIWebProgressListener.STATE_START &&
+ flags & Ci.nsIWebProgressListener.STATE_IS_NETWORK &&
+ (request && (request instanceof Ci.nsIChannel || "URI" in request)) &&
+ request.URI.path == "/") {
+ let host = request.URI.host.replace(/^www./, "").replace(/^search./, "");
+ if (gEngines.has(host)) {
+ reportTelemetryValue("userVisitedEngineHost",
+ { engine: gEngines.get(host) });
+ }
+ }
+ } catch (ex) {}
+ },
+ onStatusChange() {},
+
+ QueryInterface: XPCOMUtils.generateQI([ Ci.nsIWebProgressListener ])
+};
+
+XPCOMUtils.defineLazyGetter(this, "gEngines", () => {
+ let engines = new Map();
+ for (let engineName of [ "Google", "Yahoo", "Bing"]) {
+ let engine = Services.search.getEngineByName(engineName);
+ if (engine) {
+ try {
+ let engineHost = Services.io.newURI(engine.searchForm, null, null).host;
+ engines.set(engineHost.replace(/^www./, "").replace(/^search./, ""),
+ engine);
+ } catch (ex) {}
+ }
+ }
+ return engines;
+});
+
+function forceSearchSuggestions() {
+ // Enable search suggestions for everyone.
+ let suggestions = Preferences.get("browser.urlbar.suggest.searches", false);
+ if (!suggestions) {
+ gForcedSuggestions = true;
+ Preferences.set("browser.urlbar.suggest.searches", true);
+ }
+}
+
+function disableNewOneOffs() {
+ // Disable the one-off buttons NEW implementation, to replce it with ours.
+ let oneoffs = Preferences.get("browser.urlbar.oneOffSearches", false);
+ if (oneoffs) {
+ gDisabledOneOffs = true;
+ Preferences.set("browser.urlbar.oneOffSearches", false);
+ }
+}
+
+function restoreSearchSuggestions() {
+ if (gForcedSuggestions) {
+ Preferences.set("browser.urlbar.suggest.searches", false);
+ gForcedSuggestions = false;
+ }
+}
+
+function restoreNewOneOffs() {
+ if (gDisabledOneOffs) {
+ Preferences.set("browser.urlbar.oneOffSearches", true);
+ gDisabledOneOffs = false;
+ }
+}
+
+function isValidBrowserWindow(win) {
+ return !win.closed && win.toolbar.visible &&
+ win.document.documentElement.getAttribute("windowtype") == "navigator:browser";
+}
+
+function getBrowserWindows() {
+ let wins = Services.ww.getWindowEnumerator();
+ while (wins.hasMoreElements()) {
+ let win = wins.getNext().QueryInterface(Ci.nsIDOMWindow);
+ whenWindowLoaded(win, () => {
+ if (isValidBrowserWindow(win)) {
+ gBrowsers.add(new Browser(win));
+ }
+ });
+ }
+}
+
+function whenWindowLoaded(win, callback) {
+ if (win.document.readyState == "complete") {
+ callback();
+ return;
+ }
+ win.addEventListener("load", function onLoad(event) {
+ if (event.target == win.document) {
+ win.removeEventListener("load", onLoad, true);
+ win.setTimeout(callback, 0);
+ }
+ }, true);
+}
+
+function trackAutocompleteEnter(input) {
+ if (!input || input.id != "urlbar" || input.inPrivateContext) {
+ return;
+ }
+
+ let controller = input.popup.view.QueryInterface(Ci.nsIAutoCompleteController);
+
+ let idx = input.popup.selectedIndex;
+ if (idx == -1) {
+ return;
+ }
+ let value = controller.getValueAt(idx);
+ let action = input._parseActionUrl(value);
+ if (action.type != "searchengine")
+ return;
+
+ if ("searchSuggestion" in action.params) {
+ reportTelemetryValue("searchBySuggestion");
+ } else if ("alias" in action.params) {
+ let engine = Services.search.getEngineByAlias(action.params.alias);
+ if (engine)
+ reportTelemetryValue("searchByAlias", { engine });
+ } else {
+ reportTelemetryValue("searchByDefaultEngine");
+ }
+}
+
+/**
+ * Registers the presence of an event.
+ *
+ * @param eventName The data is logged with this name.
+ */
+function reportTelemetryValue(key, optionalData={}) {
+ // Since we only care about search volume, ignore the rare cases where a
+ // search opens in a new tab (currently onlye CTRL + Go button) and always
+ // set where to "current".
+ function recordOneOff(engine, source="unknown", type="unknown") {
+ let engineId = engine ? engine.identifier ? engine.identifier
+ : "other-" + engine.name
+ : "other";
+ BrowserUITelemetry.countOneoffSearchEvent(`${engineId}.${source}`, type,
+ "current");
+ }
+
+ // Increment "search" / "urlbar" countable.
+ // Since we only care about search volume, don't report details, like the
+ // suggestion index clicked, to countSearchEvent.
+ function recordSearch(engine=Services.search.currentEngine, source) {
+ let isInContentSearch = source.includes("content");
+ if (isInContentSearch) {
+ // Add engine name to the probe name.
+ source += "-" + engine.name.toLowerCase();
+ }
+
+ BrowserUITelemetry.countSearchEvent(source, null);
+
+ if (!isInContentSearch) {
+ let engineId = engine ? engine.identifier ? engine.identifier
+ : "other-" + engine.name
+ : "other";
+ let count = Services.telemetry.getKeyedHistogramById("SEARCH_COUNTS");
+ count.add(`${engineId}.${source}`);
+ }
+ }
+
+ switch(key) {
+ case "userVisitedEngineHost":
+ recordSearch(optionalData.engine, "content");
+ break;
+ case "userVisitedEngineResult":
+ recordSearch(optionalData.engine, "content-result");
+ break;
+ case "searchSettingsClicked":
+ BrowserUITelemetry.countSearchSettingsEvent("urlbar");
+ break;
+ case "searchByReturnKeyOnOneOffButton":
+ recordOneOff(optionalData.engine, "urlbar-oneoff", "key");
+ recordSearch(optionalData.engine, "urlbar");
+ break;
+ case "searchByClickOnOneOffButton":
+ recordOneOff(optionalData.engine, "urlbar-oneoff", "mouse");
+ recordSearch(optionalData.engine, "urlbar");
+ break;
+ case "searchBySuggestions":
+ recordSearch(optionalData.engine, "urlbar-suggestion");
+ break;
+ case "searchByDefaultEngine":
+ // If it's a direct Enter actions it already increments "search" / "urlbar"
+ // countable, so we should only increment oneoff countable for the current
+ // engine (with unknown source, like the searchbar).
+ recordOneOff(Services.search.currentEngine, "urlbar", "unknown");
+ break;
+ case "searchByAlias":
+ // Due to 1313080 aliases are not accounted by default, so do that here.
+ recordOneOff(optionalData.engine, "urlbar", "unknown");
+ recordSearch(optionalData.engine, "urlbar");
+ break;
+ default:
+ Cu.reportError("reportTelemetryValue() got an unknown value");
+ }
+}
diff --git a/code/content/gear.svg b/chrome/content/gear.svg
old mode 100755
new mode 100644
similarity index 100%
rename from code/content/gear.svg
rename to chrome/content/gear.svg
diff --git a/chrome/content/style.css b/chrome/content/style.css
new file mode 100644
index 0000000..9727f38
--- /dev/null
+++ b/chrome/content/style.css
@@ -0,0 +1,166 @@
+.urlbar-header {
+ font-size: 10px;
+ font-weight: normal;
+ margin: 0;
+ padding: 3px 6px;
+ color: #666;
+}
+
+.urlbar-header > label {
+ margin-top: 2px !important;
+ margin-bottom: 2px !important;
+}
+
+.urlbar-current-input > label {
+ margin: 2px 0 !important;
+}
+
+.urlbar-input-value {
+ color: black;
+}
+
+.urlbar-one-offs {
+ margin: 0 0 !important;
+}
+
+.urlbar-engine-one-off-item {
+ -moz-appearance: none;
+ display: inline-block;
+ border: none;
+ min-width: 48px;
+ height: 32px;
+ margin: 0 0;
+ padding: 0 0;
+}
+
+.urlbar-engine-one-off-item:not(.dummy) {
+ background-image: url('');
+ background-repeat: no-repeat;
+ background-position: right center;
+}
+
+.urlbar-engine-one-off-item:not(.last-row) {
+ box-sizing: content-box;
+ border-bottom: 1px solid #ccc;
+}
+
+.urlbar-engine-one-off-item.last-of-row {
+ background-image: none;
+}
+
+.urlbar-engine-one-off-item[selected] {
+ background-color: Highlight;
+ background-image: none;
+}
+
+.urlbar-engine-one-off-item > .button-box > .button-text {
+ display: none;
+}
+
+.urlbar-engine-one-off-item > .button-box > .button-icon {
+ -moz-margin-start: 0;
+ width: 16px;
+ height: 16px;
+}
+
+#urlbar-search-settings2 {
+ list-style-image: url("chrome://unified-urlbar/content/gear.svg#gear");
+ background-position: left center;
+ border-bottom: none;
+}
+
+#urlbar-search-settings2:hover {
+ list-style-image: url("chrome://unified-urlbar/content/gear.svg#gear-inverted");
+ background-color: Highlight;
+}
+
+#urlbar-search-footer2 {
+ border-top: 1px solid hsla(210, 4%, 10%, 0.14);
+ background-color: hsla(210, 4%, 10%, 0.07);
+}
+
+#urlbar-tip-container {
+ font-size: 14px;
+}
+
+#urlbar-tip-box {
+ min-height: 3em;
+ padding: 4px 8px;
+ background-color: #ffeebe;
+ border: 1px solid #ffdf81;
+ border-radius: 4px;
+ color: #7d3500;
+ font: message-box;
+ opacity: 0;
+}
+
+#urlbar-tip-box span.bold {
+ font-weight: bold;
+}
+
+#urlbar-tip-box span.emoji {
+ margin: 0 0.5ch;
+ font-size: 120%;
+}
+
+#urlbar-tip-container > .ac-site-icon {
+ list-style-image: url(chrome://global/skin/icons/autocomplete-search.svg#search-icon);
+}
+
+#urlbar-tip-container[animate="true"] > #urlbar-tip-box {
+ margin-left: 160px;
+ animation-name: buildIn-tip;
+ animation-duration: 250ms;
+ animation-delay: 1000ms;
+ animation-iteration-count: 1;
+ animation-fill-mode: forwards;
+}
+
+#urlbar-tip-container[animate="true"] > .ac-site-icon {
+ transform: scale(0);
+ animation-name: buildIn-grow;
+ animation-duration: 500ms;
+ animation-delay: 750ms;
+ animation-iteration-count: 1;
+ animation-timing-function: ease-in-out;
+ animation-fill-mode: forwards;
+}
+
+#urlbar-tip-container[animate="true"] > #urlbar-tip-title > .ac-title-text {
+ text-overflow: clip;
+}
+
+#urlbar-tip-container[animate="true"] > #urlbar-tip-title {
+ overflow: hidden;
+ max-width: 8ch;
+ width: 0;
+ animation-name: typing;
+ animation-duration: 750ms;
+ animation-delay: 300ms;
+ animation-iteration-count: 1;
+ animation-fill-mode: forwards;
+}
+
+@keyframes buildIn-tip {
+ 0% { margin-left: 160px; opacity: 0; }
+ 100% { margin-left: 0; opacity: 1; }
+}
+
+@keyframes buildIn-grow {
+ 0% { transform: scale(0); }
+ 40% { transform: scale(1.5); }
+ 60% { transform: scale(1); }
+ 80% { transform: scale(1.25); }
+ 100% { transform: scale(1); }
+}
+
+@keyframes typing {
+ from { width: 0; }
+ to { width: 8ch; }
+}
+
+#PopupAutoCompleteRichResult > hbox[anonid="search-suggestions-notification"][tip="true"] {
+ background-image: none;
+ padding-inline-start: 0;
+ padding-inline-end: 0;
+}
diff --git a/code/bootstrap.js b/code/bootstrap.js
deleted file mode 100755
index b6fbc46..0000000
--- a/code/bootstrap.js
+++ /dev/null
@@ -1,79 +0,0 @@
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-Cu.import("resource:///modules/experiments/Experiments.jsm");
-Cu.import("resource:///modules/CustomizableUI.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "Telemetry",
- "chrome://unified-urlbar/content/Telemetry.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "BrowserListener",
- "chrome://unified-urlbar/content/BrowserListener.jsm");
-
-var gStarted = false;
-
-function startup(data, reason) {
- // Seems startup() function is launched twice after install, we're
- // unsure why so far. We only want it to run once.
- if (gStarted) {
- return;
- }
- gStarted = true;
-
- // Workaround until bug 1228359 is fixed.
- //Components.manager.addBootstrappedManifestLocation(data.installPath);
-
- ensureExperimentBranch().then(branch => {
- BrowserListener.init(branch);
- Telemetry.init(branch);
- });
-}
-
-function shutdown(data, reason) {
- // Workaround until bug 1228359 is fixed.
- //Components.manager.removeBootstrappedManifestLocation(data.installPath);
-
- BrowserListener.destroy();
- Telemetry.destroy();
-}
-
-function install(data, reason) {}
-function uninstall(data, reason) {}
-
-/**
- * Ensures that the experiment branch is set and returns it.
- *
- * @return Promise Resolved with the branch.
- */
-function ensureExperimentBranch() {
- return new Promise(resolve => {
- // TESTING CODE
- try {
- let forcedBranch =
- Services.prefs.getCharPref("browser.urlbar.experiment.unified-urlbar.branch");
- resolve(forcedBranch);
- } catch (ex) {}
-
- let experiments = Experiments.instance();
- // This experiment has 3 user groups:
- // * "control" : Users with default search bar setup.
- // No UI changes.
- // * "customized": Users who customized the search bar position.
- // Add one-off buttons.
- // * "unified" : Add one-off search buttons to the location bar and
- // customize away the search bar.
- let branch = experiments.getActiveExperimentBranch();
- if (branch) {
- resolve(branch);
- return;
- }
- let placement = CustomizableUI.getPlacementOfWidget("search-container");
- if (!placement || placement.area != "nav-bar") {
- branch = "customized";
- } else {
- let coinFlip = Math.floor(2 * Math.random());
- branch = coinFlip ? "control" : "unified";
- }
- let id = experiments.getActiveExperimentID();
- experiments.setExperimentBranch(id, branch).then(() => resolve(branch));
- });
-}
diff --git a/code/chrome.manifest b/code/chrome.manifest
deleted file mode 100755
index da37f4e..0000000
--- a/code/chrome.manifest
+++ /dev/null
@@ -1 +0,0 @@
-content unified-urlbar content/
diff --git a/code/content/BrowserListener.jsm b/code/content/BrowserListener.jsm
deleted file mode 100755
index c725556..0000000
--- a/code/content/BrowserListener.jsm
+++ /dev/null
@@ -1,153 +0,0 @@
-this.EXPORTED_SYMBOLS = [
- "BrowserListener",
-];
-
-const STYLE_URL = "chrome://unified-urlbar/content/style.css";
-const SEARCH_BAR_WIDGET_ID = "search-container";
-const XHTML_NS = "http://www.w3.org/1999/xhtml";
-
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource:///modules/CustomizableUI.jsm");
-Cu.import("chrome://unified-urlbar/content/Panel.jsm");
-
-var gBranch = "control";
-var gBrowsers = null;
-
-this.BrowserListener = Object.freeze({
- init(branch) {
- gBranch = branch;
-
- if (gBrowsers) {
- return;
- }
- gBrowsers = new Set();
- getBrowserWindows();
- Services.ww.registerNotification(this);
- },
-
- destroy() {
- if (!gBrowsers) {
- return;
- }
-
- Services.ww.unregisterNotification(this);
- for (let browser of gBrowsers) {
- browser.destroy();
- }
- gBrowsers.clear();
- gBrowsers = null;
- },
-
- observe(subj, topic, data) {
- let win = subj.QueryInterface(Ci.nsIDOMWindow);
- if (!win) {
- return;
- }
-
- if (topic == "domwindowopened") {
- whenWindowLoaded(win, () => {
- if (isValidBrowserWindow(win)) {
- gBrowsers.add(new Browser(win));
- }
- });
- } else if (topic == "domwindowclosed") {
- for (let browser of gBrowsers) {
- if (browser.window == win) {
- browser.destroy();
- gBrowsers.delete(browser);
- break;
- }
- }
- }
- },
-});
-
-function Browser(win) {
- this.window = win;
-
- switch (gBranch) {
- case "unified":
- this._moveSearchBar();
- // Fall through.
- case "customized":
- this._injectStyle();
- this._initPanel();
- break;
- default:
- // Nothing!
- break;
- }
-}
-
-Browser.prototype = {
- get document() {
- return this.window.document;
- },
-
- destroy() {
- if (this._styleLink) {
- this._styleLink.remove();
- delete this._styleLink;
- }
-
- if (this._panel) {
- this._panel.destroy();
- }
-
- if (this._searchbarPlacement) {
- CustomizableUI.addWidgetToArea(SEARCH_BAR_WIDGET_ID,
- this._searchbarPlacement.area,
- this._searchbarPlacement.position);
- }
- },
-
- _injectStyle() {
- this._styleLink = this.document.createElementNS(XHTML_NS, "link");
- this._styleLink.setAttribute("href", STYLE_URL);
- this._styleLink.setAttribute("rel", "stylesheet");
- this.document.documentElement.appendChild(this._styleLink);
- },
-
- _initPanel() {
- let elt = this.document.getElementById("PopupAutoCompleteRichResult");
- this._panel = new Panel(elt);
- },
-
- _moveSearchBar() {
- this._searchbarPlacement =
- CustomizableUI.getPlacementOfWidget(SEARCH_BAR_WIDGET_ID);
- CustomizableUI.removeWidgetFromArea(SEARCH_BAR_WIDGET_ID);
- },
-};
-
-function isValidBrowserWindow(win) {
- return !win.closed && win.toolbar.visible &&
- win.document.documentElement.getAttribute("windowtype") == "navigator:browser";
-}
-
-function getBrowserWindows() {
- let wins = Services.ww.getWindowEnumerator();
- while (wins.hasMoreElements()) {
- let win = wins.getNext().QueryInterface(Ci.nsIDOMWindow);
- whenWindowLoaded(win, () => {
- if (isValidBrowserWindow(win)) {
- gBrowsers.add(new Browser(win));
- }
- });
- }
-}
-
-function whenWindowLoaded(win, callback) {
- if (win.document.readyState == "complete") {
- callback();
- return;
- }
- win.addEventListener("load", function onLoad(event) {
- if (event.target == win.document) {
- win.removeEventListener("load", onLoad, true);
- win.setTimeout(callback, 0);
- }
- }, true);
-}
diff --git a/code/content/Telemetry.jsm b/code/content/Telemetry.jsm
deleted file mode 100755
index d293b81..0000000
--- a/code/content/Telemetry.jsm
+++ /dev/null
@@ -1,137 +0,0 @@
-this.EXPORTED_SYMBOLS = [
- "Telemetry",
-];
-
-const SEARCH_SUGGESTIONS_OPT_IN_CHOICE_PREF =
- "browser.urlbar.userMadeSearchSuggestionsChoice";
-
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-Cu.import("resource://gre/modules/Preferences.jsm");
-Cu.import("resource:///modules/BrowserUITelemetry.jsm");
-Cu.import("resource://gre/modules/Services.jsm");
-
-var gBranch = "control";
-var gValues = new Map();
-
-this.Telemetry = Object.freeze({
-
- init(branch) {
- gBranch = branch;
- addSearchSuggestionsOptInTelemetry();
- Services.obs.addObserver(this, "autocomplete-did-enter-text", false);
- },
-
- destroy() {
- Services.obs.removeObserver(this, "autocomplete-did-enter-text", false);
- Preferences.ignore(SEARCH_SUGGESTIONS_OPT_IN_CHOICE_PREF);
- gValues.clear();
- },
-
- observe: function(aSubject, aTopic, aData) {
- let input = aSubject.QueryInterface(Ci.nsIAutoCompleteInput);
- if (!input || input.id != "urlbar" || input.inPrivateContext ||
- input.popup.selectedIndex == -1) {
- return;
- }
- let controller = input.popup.view.QueryInterface(Ci.nsIAutoCompleteController);
- let idx = input.popup.selectedIndex;
- let value = controller.getValueAt(idx);
- let action = input._parseActionUrl(value);
- let actionType;
- if (action) {
- actionType = action.type == "searchengine" && action.params.searchSuggestion ?
- "searchsuggestion" : action.type;
- }
- if (actionType == "searchengine") {
- this.incrementValue("searchByDefaultEngine");
- } else if (actionType == "searchSuggestion") {
- this.incrementValue("searchBySuggestion");
- }
- },
-
- /**
- * Registers the presence of an event.
- *
- * @param eventName The data is logged with this name.
- */
- incrementValue(key, optionalData={}) {
- // Since we only care about search volume, ignore the rare cases where a
- // search opens in a new tab (currently onlye CTRL + Go button) and always
- // set where to "current".
- function recordOneOff(engine, source="unknown", type="unknown") {
- let engineId = engine ? engine.identifier ? engine.identifier
- : "other-" + engine.name
- : "other";
- BrowserUITelemetry.countOneoffSearchEvent(`${engineId}.${source}`, type,
- "current");
- }
-
- // Increment "search" / "urlbar" countable.
- // Since we only care about search volume, don't report details, like the
- // suggestion index clicked, to countSearchEvent.
- function recordSearch(engine=Services.search.currentEngine, source) {
- BrowserUITelemetry.countSearchEvent(source, null);
- let engineId = engine ? engine.identifier ? engine.identifier
- : "other-" + engine.name
- : "other";
- let count = Services.telemetry.getKeyedHistogramById("SEARCH_COUNTS");
- count.add(`${engineId}.${source}`);
- }
-
- switch(key) {
- case "searchSettingsClicked":
- BrowserUITelemetry.countSearchSettingsEvent("urlbar");
- break;
- case "searchByReturnKeyOnOneOffButton":
- recordOneOff(optionalData.engine, "urlbar-oneoff", "key");
- recordSearch(optionalData.engine, "urlbar");
- break;
- case "searchByClickOnOneOffButton":
- recordOneOff(optionalData.engine, "urlbar-oneoff", "mouse");
- recordSearch(optionalData.engine, "urlbar");
- break;
- case "searchByDefaultEngine":
- // Already increments "search" / "urlbar" countable, it must also
- // increment oneoff countable for the current engine.
- // We don't have data about mouse or keyboard interaction here yet, so
- // just pass unknown for the type.
- recordOneOff(Services.search.currentEngine, "urlbar", "unknown");
- break;
- default:
- // This data is currently unused, but stored for reference.
- let val = gValues.get(key) || 0;
- gValues.set(key, val);
- }
- },
-
- /**
- * Registers a key-value.
- *
- * @param key The data is logged with this name.
- * @param value The value of the data.
- */
- setValue(key, value) {
- // This data is currently unused, but stored for reference.
- gValues.set(key, value);
- },
-
- QueryInterface: XPCOMUtils.generateQI([Ci.nsIObserver ]),
-});
-
-/**
- * Collects data about the user's urlbar search suggestions opt-in choice, or
- * if none has been made yet, adds a pref observer for it.
- */
-function addSearchSuggestionsOptInTelemetry() {
- let userMadeChoice = Preferences.get(SEARCH_SUGGESTIONS_OPT_IN_CHOICE_PREF);
- Telemetry.setValue("userMadeSuggestionsChoice", userMadeChoice);
- if (!userMadeChoice) {
- Preferences.observe(SEARCH_SUGGESTIONS_OPT_IN_CHOICE_PREF, () => {
- Preferences.ignore(SEARCH_SUGGESTIONS_OPT_IN_CHOICE_PREF);
- addSearchSuggestionsOptInTelemetry();
- });
- }
- let optedIn = Preferences.get("browser.urlbar.suggest.searches");
- Telemetry.setValue("suggestionsEnabled", optedIn);
-}
diff --git a/code/content/style.css b/code/content/style.css
deleted file mode 100755
index 4903b6c..0000000
--- a/code/content/style.css
+++ /dev/null
@@ -1,80 +0,0 @@
-.urlbar-header {
- font-size: 10px;
- font-weight: normal;
- margin: 0;
- padding: 3px 6px;
- color: #666;
-}
-
-.urlbar-header > label {
- margin-top: 2px !important;
- margin-bottom: 2px !important;
-}
-
-.urlbar-current-input > label {
- margin: 2px 0 !important;
-}
-
-.urlbar-input-value {
- color: black;
-}
-
-.urlbar-one-offs {
- margin: 0 0 !important;
-}
-
-.urlbar-engine-one-off-item {
- -moz-appearance: none;
- display: inline-block;
- border: none;
- min-width: 48px;
- height: 32px;
- margin: 0 0;
- padding: 0 0;
-}
-
-.urlbar-engine-one-off-item:not(.dummy) {
- background-image: url('');
- background-repeat: no-repeat;
- background-position: right center;
-}
-
-.urlbar-engine-one-off-item:not(.last-row) {
- box-sizing: content-box;
- border-bottom: 1px solid #ccc;
-}
-
-.urlbar-engine-one-off-item.last-of-row {
- background-image: none;
-}
-
-.urlbar-engine-one-off-item[selected] {
- background-color: Highlight;
- background-image: none;
-}
-
-.urlbar-engine-one-off-item > .button-box > .button-text {
- display: none;
-}
-
-.urlbar-engine-one-off-item > .button-box > .button-icon {
- -moz-margin-start: 0;
- width: 16px;
- height: 16px;
-}
-
-#urlbar-search-settings2 {
- list-style-image: url("chrome://unified-urlbar/content/gear.svg#gear");
- background-position: left center;
- border-bottom: none;
-}
-
-#urlbar-search-settings2:hover {
- list-style-image: url("chrome://unified-urlbar/content/gear.svg#gear-inverted");
- background-color: Highlight;
-}
-
-#urlbar-search-footer2 {
- border-top: 1px solid hsla(210, 4%, 10%, 0.14);
- background-color: hsla(210, 4%, 10%, 0.07);
-}
diff --git a/code/install.rdf b/code/install.rdf
deleted file mode 100755
index 72ce290..0000000
--- a/code/install.rdf
+++ /dev/null
@@ -1,26 +0,0 @@
-
-
-
- unified-urlbar@experiments.mozilla.org
- 1.0.0
- 128
- true
- false
-
-
-
-
- {ec8030f7-c20a-464f-9b0e-13a3a9e97384}
- 43.0
- 47.0
-
-
-
- Unified Urlbar Experiment
-
- An experimental add-on testing the unified urlbar experience.
-
- https://bugzilla.mozilla.org/show_bug.cgi?id=1219505
-
-
diff --git a/docs/metrics.md b/docs/metrics.md
new file mode 100644
index 0000000..940ffb2
--- /dev/null
+++ b/docs/metrics.md
@@ -0,0 +1,128 @@
+```JSON
+{
+ "payload": {
+ "ver": 4,
+ "simpleMeasurements": {
+ "UITelemetry": {
+ "toolbars": {
+ "defaultKept": [
+ "edit-controls",
+ "zoom-controls",
+ "save-page-button",
+ "print-button",
+ "history-panelmenu",
+ "fullscreen-button",
+ "preferences-button",
+ "add-ons-button",
+ "privatebrowsing-button",
+ "urlbar-container",
+ "search-container",
+ "bookmarks-menu-button",
+ "downloads-button",
+ "menubar-items",
+ "tabbrowser-tabs",
+ "new-tab-button",
+ "alltabs-button",
+ "personal-bookmarks"
+ ],
+ "defaultMoved": [
+ "developer-button"
+ ],
+ "nondefaultAdded": [
+ "feed-button"
+ ],
+ "defaultRemoved": [
+ "sync-button",
+ "find-button",
+ "social-share-button",
+ "pocket-button",
+ "new-window-button",
+ "home-button"
+ ],
+ "currentSearchEngine": "engine_name",
+ "countableEvents": {
+ "__DEFAULT__": {
+ "click-builtin-item": {
+ "urlbar": {
+ "search-settings": 1
+ },
+ "searchbar": {
+ "search-settings": 1
+ }
+ },
+ "search": {
+ "urlbar": 3,
+ "searchbar": 3,
+ "newtab": 3,
+ "abouthome": 3,
+ "content-bing": 3,
+ "content-google": 3,
+ "content-yahoo": 3,
+ "content-result-bing": 3,
+ "content-result-google": 3,
+ "content-result-yahoo": 3,
+ "autocomplete-other": 3,
+ "autocomplete-default": 3,
+ },
+ "search-oneoff": {
+ "engine_name.ui_source": {
+ "mouse": {
+ "current": 1
+ },
+ "key": {
+ "current": 1
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ },
+ "histograms": {
+ "FX_URLBAR_SELECTED_RESULT_TYPE": {
+ "range": [
+ 1,
+ 14
+ ],
+ "bucket_count": 15,
+ "histogram_type": 1,
+ "values": {
+ "0": 4,
+ "1": 2,
+ "2": 8,
+ "3": 6,
+ "8": 2,
+ "9": 0
+ },
+ "sum": 52
+ }
+ },
+ "keyedHistograms": {
+ "SEARCH_COUNTS": {
+ "engine_name.ui_source": {
+ "range": [
+ 1,
+ 2
+ ],
+ "bucket_count": 3,
+ "histogram_type": 4,
+ "values": {
+ "0": 1,
+ "1": 0
+ },
+ "sum": 1
+ }
+ }
+ }
+ },
+ "environment": {
+ "settings": {
+ "userPrefs": {
+ "browser.urlbar.suggest.searches": true,
+ "browser.urlbar.userMadeSearchSuggestionsChoice": true,
+ }
+ }
+ }
+}
+```
diff --git a/docs/telemetry.md b/docs/telemetry.md
new file mode 100644
index 0000000..4f0f7b7
--- /dev/null
+++ b/docs/telemetry.md
@@ -0,0 +1,58 @@
+environment/settings/userPrefs/browser.urlbar.suggest.searches
+* tracks the search suggestions enabled pref
+
+environment/settings/userPrefs/browser.urlbar.userMadeSearchSuggestionsChoice
+* tracks the pref stating whether the user made a choice for search suggestions opt-in
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/searchbar
+* total number of searches started from the search bar
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/abouthome
+* total number of searches started from about:home
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/newtab
+* total number of searches started from about:newtab
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/urlbar
+* total number of searches started from the urlbar
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/content-bing
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/content-google
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/content-yahoo
+* In-content direct accesses to "google", "bing", "yahoo" root domains
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/content-result-bing
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/content-result-google
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/content-result-yahoo
+* In-content accesses to a "google", "bing", "yahoo" first result page (heuristic)
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/autocomplete-other
+* number of searches to a non-current engine started by revisiting a previous search result in the urlbar
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search/autocomplete-default
+* number of searches to the current engine started by revisiting a previous search result in the urlbar
+
+payload/simpleMeasurements/UITelemetry/toolbars/defaultKept
+payload/simpleMeasurements/UITelemetry/toolbars/defaultMoved
+payload/simpleMeasurements/UITelemetry/toolbars/defaultRemMoved
+* widget positions on toolbar, look for "search-container"
+
+payload/histograms/FX_URLBAR_SELECTED_RESULT_TYPE
+* type of result selected from the urlbar dropdown, see nsBroweserGlue.js::_handleURLBarTelemetry for enums
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/search-oneoff/
+* engine.unknown
+ * Serches done by just typing something and pressing Enter
+* engine.oneoff-searchbar
+ * searches from searchbar one-off buttons
+* engine.oneoff-urlbar
+ * searches from urlbar one-off buttons
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/click-builtin-item/searchbar/search-settings
+* number of times the searchbar search settings button is used
+
+payload/simpleMeasurements/UITelemetry/toolbars/countableEvents/__DEFAULT__/click-builtin-item/urlbar/search-settings
+* number of times the urlbar search settings button is used
+
+payload/keyedHistograms/SEARCH_COUNTS
+* should ideally be the sum of all the searches from urlbar, searchbar, abouthome, newtab
diff --git a/docs/testing.md b/docs/testing.md
new file mode 100644
index 0000000..454114e
--- /dev/null
+++ b/docs/testing.md
@@ -0,0 +1,15 @@
+Since UITelemetry is only collected after a restart, it's required to use the --profile and --no-copy options to ensure the profile persists across runs.
+It's also necessary to use an unbranded build (from https://wiki.mozilla.org/Add-ons/Extension_Signing) to test this on Beta 50 using the Shield cli.
+
+For example, after cloning the git report and entering the add-on folder:
+```Bash
+npm install -g shield-study-cli jpm
+npm install --save-dev jpm
+shield run . unified -- --profile path_to_profile --no-copy
+```
+After the first run, just close the browser and shield run again, UITelemetry should be now collected properly.
+
+unified is one of the three variations:
+ * unified: add oneoffs, remove searchbar
+ * oneoff: add oneoff
+ * control: no changes
diff --git a/filter.js b/filter.js
deleted file mode 100755
index 4606e32..0000000
--- a/filter.js
+++ /dev/null
@@ -1,3 +0,0 @@
-function filter(c) {
- return c.telemetryEnvironment.settings.telemetryEnabled;
-}
diff --git a/lib/feature.js b/lib/feature.js
new file mode 100644
index 0000000..247b4aa
--- /dev/null
+++ b/lib/feature.js
@@ -0,0 +1,73 @@
+var {Cc, Ci, Cu} = require("chrome");
+Cu.import("resource://gre/modules/XPCOMUtils.jsm");
+XPCOMUtils.defineLazyModuleGetter(this, "UnifiedUrlbar",
+ "chrome://unified-urlbar/content/UnifiedUrlbar.jsm");
+const prefSvc = require('sdk/preferences/service');
+
+const feature = {
+ isEligible() {
+ return UnifiedUrlbar.isUserEligible();
+ },
+
+ checkHasConflicts() {
+ // Exclude users having the Activity Stream add-on.
+ const { AddonManager } = Cu.import("resource://gre/modules/AddonManager.jsm");
+ return new Promise(resolve => {
+ AddonManager.getAddonsByIDs(["@activity-streams", "universal-search@mozilla.com"],
+ results => resolve(results.some(r => !!r)));
+ });
+ },
+
+ install(variation) {
+ // Enable extended telemetry if needed, but note this requires a restart
+ // before we can start collecting data.
+ if (!prefSvc.isSet("browser.urlbar.experiment.unified-urlbar.telemetry.enabled.mirror")) {
+ let telemetryEnabled = prefSvc.get("toolkit.telemetry.enabled", false);
+ if (!telemetryEnabled) {
+ prefSvc.set("browser.urlbar.experiment.unified-urlbar.telemetry.enabled.mirror", telemetryEnabled);
+ prefSvc.set("toolkit.telemetry.enabled", true);
+ }
+ }
+
+ switch (variation) {
+ case "unified":
+ UnifiedUrlbar.init({
+ forceSearchSuggestions: true,
+ addOneOffAndOnboarding: true,
+ removeSearchbar: true,
+ });
+ break;
+
+ case "oneoff":
+ UnifiedUrlbar.init({
+ forceSearchSuggestions: true,
+ addOneOffAndOnboarding: true,
+ removeSearchbar: false,
+ });
+ break;
+
+ default:
+ UnifiedUrlbar.init({
+ forceSearchSuggestions: false,
+ addOneOffAndOnboarding: false,
+ removeSearchbar: false,
+ });
+ break;
+ }
+ },
+
+ cleanup() {
+ UnifiedUrlbar.destroy();
+ },
+
+ uninstall() {
+ if (prefSvc.isSet("browser.urlbar.experiment.unified-urlbar.telemetry.enabled.mirror")) {
+ let telemetryEnabled = prefSvc.get("browser.urlbar.experiment.unified-urlbar.telemetry.enabled.mirror", false);
+ prefSvc.reset("browser.urlbar.experiment.unified-urlbar.telemetry.enabled.mirror");
+ prefSvc.set("toolkit.telemetry.enabled", telemetryEnabled);
+ }
+ prefSvc.reset("browser.urlbar.experiment.unified-urlbar.tipShownCount");
+ }
+};
+
+exports.feature = feature;
diff --git a/lib/index.js b/lib/index.js
new file mode 100644
index 0000000..f760e0e
--- /dev/null
+++ b/lib/index.js
@@ -0,0 +1,8 @@
+const self = require("sdk/self");
+const { study, checkHasConflicts } = require("./study");
+
+checkHasConflicts().then(hasConflicts => {
+ if (!hasConflicts) {
+ study.startup(self.loadReason);
+ }
+});
diff --git a/lib/study.js b/lib/study.js
new file mode 100644
index 0000000..58b959d
--- /dev/null
+++ b/lib/study.js
@@ -0,0 +1,66 @@
+const self = require("sdk/self");
+const shield = require("shield-studies-addon-utils");
+const { when: unload } = require("sdk/system/unload");
+const { feature } = require('./feature');
+const prefSvc = require('sdk/preferences/service');
+
+const studyConfig = {
+ name: self.addonId,
+ duration: 14,
+ variations: {
+ "unified": () => feature.install("unified"),
+ "oneoff": () => feature.install("oneoff"),
+ "control": () => feature.install("control")
+ },
+ surveyUrls: {
+ 'end-of-study': "https://qsurvey.mozilla.com/s3/search-study-1",
+ 'user-ended-study': "https://qsurvey.mozilla.com/s3/search-study-1",
+ 'ineligible': null
+ },
+}
+
+class OurStudy extends shield.Study {
+ constructor(config) {
+ super(config);
+ }
+
+ isEligible() {
+ return super.isEligible() && feature.isEligible();
+ }
+
+ whenIneligible () {
+ // Additional actions when the user isn't eligible.
+ super.whenIneligible();
+ }
+
+ whenInstalled() {
+ // Additional actions when the study gets installed.
+ super.whenInstalled();
+ }
+
+ cleanup() {
+ super.cleanup();
+ feature.cleanup();
+ }
+
+ whenComplete () {
+ // Additional actions when the study is naturally completed.
+ super.whenComplete(); // calls survey, uninstalls
+ }
+
+ whenUninstalled () {
+ // Additional actions when the user uninstalls the study.
+ super.whenUninstalled();
+ feature.uninstall();
+ }
+
+ decideVariation () {
+ return super.decideVariation(); // chooses at random
+ }
+}
+
+const thisStudy = new OurStudy(studyConfig);
+unload((reason) => thisStudy.shutdown(reason))
+
+exports.study = thisStudy;
+exports.checkHasConflicts = feature.checkHasConflicts;
diff --git a/manifest.json b/manifest.json
deleted file mode 100755
index 64dc6e7..0000000
--- a/manifest.json
+++ /dev/null
@@ -1,19 +0,0 @@
-{
- "publish" : true,
- "priority" : 2,
- "name" : "Unified Search in Urlbar",
- "description" : "Adds search features to the urlbar and moves the search bar to the customization panel",
- "info" : "Related bug
",
- "manifest" : {
- "id" : "unified-urlbar@experiments.mozilla.org",
- "startTime" : 1452470400,
- "endTime" : 1456272000,
- "disabled" : true,
- "maxActiveSeconds" : 1209600,
- "appName" : ["Firefox"],
- "channel" : ["beta"],
- "minVersion" : "44.0",
- "maxVersion" : "45.*",
- "sample" : 0.1
- }
-}
diff --git a/node_modules/shield-studies-addon-utils/LICENSE b/node_modules/shield-studies-addon-utils/LICENSE
new file mode 100644
index 0000000..a612ad9
--- /dev/null
+++ b/node_modules/shield-studies-addon-utils/LICENSE
@@ -0,0 +1,373 @@
+Mozilla Public License Version 2.0
+==================================
+
+1. Definitions
+--------------
+
+1.1. "Contributor"
+ means each individual or legal entity that creates, contributes to
+ the creation of, or owns Covered Software.
+
+1.2. "Contributor Version"
+ means the combination of the Contributions of others (if any) used
+ by a Contributor and that particular Contributor's Contribution.
+
+1.3. "Contribution"
+ means Covered Software of a particular Contributor.
+
+1.4. "Covered Software"
+ means Source Code Form to which the initial Contributor has attached
+ the notice in Exhibit A, the Executable Form of such Source Code
+ Form, and Modifications of such Source Code Form, in each case
+ including portions thereof.
+
+1.5. "Incompatible With Secondary Licenses"
+ means
+
+ (a) that the initial Contributor has attached the notice described
+ in Exhibit B to the Covered Software; or
+
+ (b) that the Covered Software was made available under the terms of
+ version 1.1 or earlier of the License, but not also under the
+ terms of a Secondary License.
+
+1.6. "Executable Form"
+ means any form of the work other than Source Code Form.
+
+1.7. "Larger Work"
+ means a work that combines Covered Software with other material, in
+ a separate file or files, that is not Covered Software.
+
+1.8. "License"
+ means this document.
+
+1.9. "Licensable"
+ means having the right to grant, to the maximum extent possible,
+ whether at the time of the initial grant or subsequently, any and
+ all of the rights conveyed by this License.
+
+1.10. "Modifications"
+ means any of the following:
+
+ (a) any file in Source Code Form that results from an addition to,
+ deletion from, or modification of the contents of Covered
+ Software; or
+
+ (b) any new file in Source Code Form that contains any Covered
+ Software.
+
+1.11. "Patent Claims" of a Contributor
+ means any patent claim(s), including without limitation, method,
+ process, and apparatus claims, in any patent Licensable by such
+ Contributor that would be infringed, but for the grant of the
+ License, by the making, using, selling, offering for sale, having
+ made, import, or transfer of either its Contributions or its
+ Contributor Version.
+
+1.12. "Secondary License"
+ means either the GNU General Public License, Version 2.0, the GNU
+ Lesser General Public License, Version 2.1, the GNU Affero General
+ Public License, Version 3.0, or any later versions of those
+ licenses.
+
+1.13. "Source Code Form"
+ means the form of the work preferred for making modifications.
+
+1.14. "You" (or "Your")
+ means an individual or a legal entity exercising rights under this
+ License. For legal entities, "You" includes any entity that
+ controls, is controlled by, or is under common control with You. For
+ purposes of this definition, "control" means (a) the power, direct
+ or indirect, to cause the direction or management of such entity,
+ whether by contract or otherwise, or (b) ownership of more than
+ fifty percent (50%) of the outstanding shares or beneficial
+ ownership of such entity.
+
+2. License Grants and Conditions
+--------------------------------
+
+2.1. Grants
+
+Each Contributor hereby grants You a world-wide, royalty-free,
+non-exclusive license:
+
+(a) under intellectual property rights (other than patent or trademark)
+ Licensable by such Contributor to use, reproduce, make available,
+ modify, display, perform, distribute, and otherwise exploit its
+ Contributions, either on an unmodified basis, with Modifications, or
+ as part of a Larger Work; and
+
+(b) under Patent Claims of such Contributor to make, use, sell, offer
+ for sale, have made, import, and otherwise transfer either its
+ Contributions or its Contributor Version.
+
+2.2. Effective Date
+
+The licenses granted in Section 2.1 with respect to any Contribution
+become effective for each Contribution on the date the Contributor first
+distributes such Contribution.
+
+2.3. Limitations on Grant Scope
+
+The licenses granted in this Section 2 are the only rights granted under
+this License. No additional rights or licenses will be implied from the
+distribution or licensing of Covered Software under this License.
+Notwithstanding Section 2.1(b) above, no patent license is granted by a
+Contributor:
+
+(a) for any code that a Contributor has removed from Covered Software;
+ or
+
+(b) for infringements caused by: (i) Your and any other third party's
+ modifications of Covered Software, or (ii) the combination of its
+ Contributions with other software (except as part of its Contributor
+ Version); or
+
+(c) under Patent Claims infringed by Covered Software in the absence of
+ its Contributions.
+
+This License does not grant any rights in the trademarks, service marks,
+or logos of any Contributor (except as may be necessary to comply with
+the notice requirements in Section 3.4).
+
+2.4. Subsequent Licenses
+
+No Contributor makes additional grants as a result of Your choice to
+distribute the Covered Software under a subsequent version of this
+License (see Section 10.2) or under the terms of a Secondary License (if
+permitted under the terms of Section 3.3).
+
+2.5. Representation
+
+Each Contributor represents that the Contributor believes its
+Contributions are its original creation(s) or it has sufficient rights
+to grant the rights to its Contributions conveyed by this License.
+
+2.6. Fair Use
+
+This License is not intended to limit any rights You have under
+applicable copyright doctrines of fair use, fair dealing, or other
+equivalents.
+
+2.7. Conditions
+
+Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted
+in Section 2.1.
+
+3. Responsibilities
+-------------------
+
+3.1. Distribution of Source Form
+
+All distribution of Covered Software in Source Code Form, including any
+Modifications that You create or to which You contribute, must be under
+the terms of this License. You must inform recipients that the Source
+Code Form of the Covered Software is governed by the terms of this
+License, and how they can obtain a copy of this License. You may not
+attempt to alter or restrict the recipients' rights in the Source Code
+Form.
+
+3.2. Distribution of Executable Form
+
+If You distribute Covered Software in Executable Form then:
+
+(a) such Covered Software must also be made available in Source Code
+ Form, as described in Section 3.1, and You must inform recipients of
+ the Executable Form how they can obtain a copy of such Source Code
+ Form by reasonable means in a timely manner, at a charge no more
+ than the cost of distribution to the recipient; and
+
+(b) You may distribute such Executable Form under the terms of this
+ License, or sublicense it under different terms, provided that the
+ license for the Executable Form does not attempt to limit or alter
+ the recipients' rights in the Source Code Form under this License.
+
+3.3. Distribution of a Larger Work
+
+You may create and distribute a Larger Work under terms of Your choice,
+provided that You also comply with the requirements of this License for
+the Covered Software. If the Larger Work is a combination of Covered
+Software with a work governed by one or more Secondary Licenses, and the
+Covered Software is not Incompatible With Secondary Licenses, this
+License permits You to additionally distribute such Covered Software
+under the terms of such Secondary License(s), so that the recipient of
+the Larger Work may, at their option, further distribute the Covered
+Software under the terms of either this License or such Secondary
+License(s).
+
+3.4. Notices
+
+You may not remove or alter the substance of any license notices
+(including copyright notices, patent notices, disclaimers of warranty,
+or limitations of liability) contained within the Source Code Form of
+the Covered Software, except that You may alter any license notices to
+the extent required to remedy known factual inaccuracies.
+
+3.5. Application of Additional Terms
+
+You may choose to offer, and to charge a fee for, warranty, support,
+indemnity or liability obligations to one or more recipients of Covered
+Software. However, You may do so only on Your own behalf, and not on
+behalf of any Contributor. You must make it absolutely clear that any
+such warranty, support, indemnity, or liability obligation is offered by
+You alone, and You hereby agree to indemnify every Contributor for any
+liability incurred by such Contributor as a result of warranty, support,
+indemnity or liability terms You offer. You may include additional
+disclaimers of warranty and limitations of liability specific to any
+jurisdiction.
+
+4. Inability to Comply Due to Statute or Regulation
+---------------------------------------------------
+
+If it is impossible for You to comply with any of the terms of this
+License with respect to some or all of the Covered Software due to
+statute, judicial order, or regulation then You must: (a) comply with
+the terms of this License to the maximum extent possible; and (b)
+describe the limitations and the code they affect. Such description must
+be placed in a text file included with all distributions of the Covered
+Software under this License. Except to the extent prohibited by statute
+or regulation, such description must be sufficiently detailed for a
+recipient of ordinary skill to be able to understand it.
+
+5. Termination
+--------------
+
+5.1. The rights granted under this License will terminate automatically
+if You fail to comply with any of its terms. However, if You become
+compliant, then the rights granted under this License from a particular
+Contributor are reinstated (a) provisionally, unless and until such
+Contributor explicitly and finally terminates Your grants, and (b) on an
+ongoing basis, if such Contributor fails to notify You of the
+non-compliance by some reasonable means prior to 60 days after You have
+come back into compliance. Moreover, Your grants from a particular
+Contributor are reinstated on an ongoing basis if such Contributor
+notifies You of the non-compliance by some reasonable means, this is the
+first time You have received notice of non-compliance with this License
+from such Contributor, and You become compliant prior to 30 days after
+Your receipt of the notice.
+
+5.2. If You initiate litigation against any entity by asserting a patent
+infringement claim (excluding declaratory judgment actions,
+counter-claims, and cross-claims) alleging that a Contributor Version
+directly or indirectly infringes any patent, then the rights granted to
+You by any and all Contributors for the Covered Software under Section
+2.1 of this License shall terminate.
+
+5.3. In the event of termination under Sections 5.1 or 5.2 above, all
+end user license agreements (excluding distributors and resellers) which
+have been validly granted by You or Your distributors under this License
+prior to termination shall survive termination.
+
+************************************************************************
+* *
+* 6. Disclaimer of Warranty *
+* ------------------------- *
+* *
+* Covered Software is provided under this License on an "as is" *
+* basis, without warranty of any kind, either expressed, implied, or *
+* statutory, including, without limitation, warranties that the *
+* Covered Software is free of defects, merchantable, fit for a *
+* particular purpose or non-infringing. The entire risk as to the *
+* quality and performance of the Covered Software is with You. *
+* Should any Covered Software prove defective in any respect, You *
+* (not any Contributor) assume the cost of any necessary servicing, *
+* repair, or correction. This disclaimer of warranty constitutes an *
+* essential part of this License. No use of any Covered Software is *
+* authorized under this License except under this disclaimer. *
+* *
+************************************************************************
+
+************************************************************************
+* *
+* 7. Limitation of Liability *
+* -------------------------- *
+* *
+* Under no circumstances and under no legal theory, whether tort *
+* (including negligence), contract, or otherwise, shall any *
+* Contributor, or anyone who distributes Covered Software as *
+* permitted above, be liable to You for any direct, indirect, *
+* special, incidental, or consequential damages of any character *
+* including, without limitation, damages for lost profits, loss of *
+* goodwill, work stoppage, computer failure or malfunction, or any *
+* and all other commercial damages or losses, even if such party *
+* shall have been informed of the possibility of such damages. This *
+* limitation of liability shall not apply to liability for death or *
+* personal injury resulting from such party's negligence to the *
+* extent applicable law prohibits such limitation. Some *
+* jurisdictions do not allow the exclusion or limitation of *
+* incidental or consequential damages, so this exclusion and *
+* limitation may not apply to You. *
+* *
+************************************************************************
+
+8. Litigation
+-------------
+
+Any litigation relating to this License may be brought only in the
+courts of a jurisdiction where the defendant maintains its principal
+place of business and such litigation shall be governed by laws of that
+jurisdiction, without reference to its conflict-of-law provisions.
+Nothing in this Section shall prevent a party's ability to bring
+cross-claims or counter-claims.
+
+9. Miscellaneous
+----------------
+
+This License represents the complete agreement concerning the subject
+matter hereof. If any provision of this License is held to be
+unenforceable, such provision shall be reformed only to the extent
+necessary to make it enforceable. Any law or regulation which provides
+that the language of a contract shall be construed against the drafter
+shall not be used to construe this License against a Contributor.
+
+10. Versions of the License
+---------------------------
+
+10.1. New Versions
+
+Mozilla Foundation is the license steward. Except as provided in Section
+10.3, no one other than the license steward has the right to modify or
+publish new versions of this License. Each version will be given a
+distinguishing version number.
+
+10.2. Effect of New Versions
+
+You may distribute the Covered Software under the terms of the version
+of the License under which You originally received the Covered Software,
+or under the terms of any subsequent version published by the license
+steward.
+
+10.3. Modified Versions
+
+If you create software not governed by this License, and you want to
+create a new license for such software, you may create and use a
+modified version of this License if you rename the license and remove
+any references to the name of the license steward (except to note that
+such modified license differs from this License).
+
+10.4. Distributing Source Code Form that is Incompatible With Secondary
+Licenses
+
+If You choose to distribute Source Code Form that is Incompatible With
+Secondary Licenses under the terms of this version of the License, the
+notice described in Exhibit B of this License must be attached.
+
+Exhibit A - Source Code Form License Notice
+-------------------------------------------
+
+ 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/.
+
+If it is not possible or desirable to put the notice in a particular
+file, then You may include the notice in a location (such as a LICENSE
+file in a relevant directory) where a recipient would be likely to look
+for such a notice.
+
+You may add additional accurate notices of copyright ownership.
+
+Exhibit B - "Incompatible With Secondary Licenses" Notice
+---------------------------------------------------------
+
+ This Source Code Form is "Incompatible With Secondary Licenses", as
+ defined by the Mozilla Public License, v. 2.0.
diff --git a/node_modules/shield-studies-addon-utils/README.md b/node_modules/shield-studies-addon-utils/README.md
new file mode 100644
index 0000000..1fa2557
--- /dev/null
+++ b/node_modules/shield-studies-addon-utils/README.md
@@ -0,0 +1,61 @@
+# Shield Studies Addon Utils [](https://travis-ci.org/tombell/travis-ci-status)
+
+- Instrument your Firefox Addon!
+- Build Shield Study (Normandy) compatible addons without having to think very much.
+
+## install
+
+```
+npm install --save-dev shield-studies-addon-utils
+```
+
+## Tutorial and Full Usage
+
+[fully worked tutorial - How To Shield Study](./howToShieldStudy.md)
+
+## Examples
+
+See `examples` directory.
+
+## Summary
+
+### Design Case
+
+Your Study is:
+
+- side-by-side variations (1 or more)
+- 'one-phase'. No warm up periods. If you want that stuff, handle it yourself, or file bugs
+
+### Benefits
+
+Using this, you get this analysis FOR FREE (and it's fast!)
+
+- Branch x channel x VARIATION x experiment-id x PHASE (install, reject, alive etc) using UNIFIED TELEMETRY
+
+- In particular, this library has 100% test coverage for lots of **startup** and **shutdown** cases, so that your addon does the Right Thing across restarts.
+
+ - maintains same variation across restarts
+ - testable, tested code
+ - doesn't care much about your variations, so long as they are 'multi-callable' safely.
+
+
+## Development
+
+- open an issue
+- hack and file a PR
+
+
+## Gotchas, Opinions, Side Effects, and Misfeatures
+
+1. This assumes `jetpack` (`jpm`) style addons, with
+
+ - `require`
+ - `jpm` startup and shutdown semantics
+
+2. Some prefs will be set and running during an experiment in the "addon-specific" pref tree.
+
+3. Disable also uninstalls (and cleans up)
+
+4. Undoubtedly, there are others. It scratches my itch. I have built a lot of things in the past.
+
+
diff --git a/node_modules/shield-studies-addon-utils/lib/event-target.js b/node_modules/shield-studies-addon-utils/lib/event-target.js
new file mode 100644
index 0000000..4335d8c
--- /dev/null
+++ b/node_modules/shield-studies-addon-utils/lib/event-target.js
@@ -0,0 +1,55 @@
+/**
+ * Drop-in replacement for {@link external:sdk/event/target.EventTarget} for use
+ * with es6 classes.
+ * @module event-target
+ * @author Martin Giger
+ * @license MPL-2.0
+ */
+ /**
+ * An SDK class that add event reqistration methods
+ * @external sdk/event/target
+ * @requires sdk/event/target
+ */
+/**
+ * @class EventTarget
+ * @memberof external:sdk/event/target
+ * @see {@link https://developer.mozilla.org/en-US/Add-ons/SDK/Low-Level_APIs/event_target#EventTarget}
+ */
+
+// slightly modified from: https://raw.githubusercontent.com/freaktechnik/justintv-stream-notifications/master/lib/event-target.js
+
+"use strict";
+
+const { on, once, off, setListeners } = require("sdk/event/core");
+
+/* istanbul ignore next */
+/**
+ * @class
+ */
+class EventTarget {
+ constructor(options) {
+ setListeners(this, options);
+ }
+
+ on(...args) {
+ on(this, ...args);
+ return this;
+ }
+
+ once(...args) {
+ once(this, ...args);
+ return this;
+ }
+
+ off(...args) {
+ off(this, ...args);
+ return this;
+ }
+
+ removeListener(...args) {
+ off(this, ...args);
+ return this;
+ }
+}
+
+exports.EventTarget = EventTarget;
diff --git a/node_modules/shield-studies-addon-utils/lib/index.js b/node_modules/shield-studies-addon-utils/lib/index.js
new file mode 100644
index 0000000..9d4ec95
--- /dev/null
+++ b/node_modules/shield-studies-addon-utils/lib/index.js
@@ -0,0 +1,428 @@
+"use strict";
+
+// Chrome privileged
+const {Cu} = require("chrome");
+const { Services } = Cu.import("resource://gre/modules/Services.jsm");
+const { TelemetryController } = Cu.import("resource://gre/modules/TelemetryController.jsm");
+const CID = Cu.import("resource://gre/modules/ClientID.jsm");
+
+// sdk
+const { merge } = require("sdk/util/object");
+const querystring = require("sdk/querystring");
+const { prefs } = require("sdk/simple-prefs");
+const prefSvc = require("sdk/preferences/service");
+const { setInterval } = require("sdk/timers");
+const tabs = require("sdk/tabs");
+const { URL } = require("sdk/url");
+
+const { EventTarget } = require("./event-target");
+const { emit } = require("sdk/event/core");
+const self = require("sdk/self");
+
+const DAY = 86400*1000;
+
+// ongoing within-addon fuses / timers
+let lastDailyPing = Date.now();
+
+/* Functional, self-contained utils */
+
+// equal probability choices from a list "choices"
+function chooseVariation(choices,rng=Math.random()) {
+ let l = choices.length;
+ return choices[Math.floor(l*Math.random())];
+}
+
+function dateToUTC(date) {
+ return Date.UTC(date.getUTCFullYear(), date.getUTCMonth(), date.getUTCDate(), date.getUTCHours(), date.getUTCMinutes(), date.getUTCSeconds());
+}
+
+function generateTelemetryIdIfNeeded() {
+ let id = TelemetryController.clientID;
+ /* istanbul ignore next */
+ if (id == undefined) {
+ return CID.ClientIDImpl._doLoadClientID()
+ } else {
+ return Promise.resolve(id)
+ }
+}
+
+function userId () {
+ return prefSvc.get("toolkit.telemetry.cachedClientID","unknown");
+}
+
+var Reporter = new EventTarget().on("report",
+ (d) => prefSvc.get('shield.debug') && console.log("report",d)
+);
+
+function report(data, src="addon", bucket="shield-study") {
+ data = merge({}, data , {
+ study_version: self.version,
+ about: {
+ _src: src,
+ _v: 2
+ }
+ });
+ if (prefSvc.get('shield.testing')) data.testing = true
+
+ emit(Reporter, "report", data);
+ let telOptions = {addClientId: true, addEnvironment: true}
+ return TelemetryController.submitExternalPing(bucket, data, telOptions);
+}
+
+function survey (url, queryArgs={}) {
+ if (! url) return
+
+ let U = new URL(url);
+ let q = U.search;
+ if (q) {
+ url = U.href.split(q)[0];
+ q = querystring.parse(querystring.unescape(q.slice(1)));
+ } else {
+ q = {};
+ }
+ // get user info.
+ let newArgs = merge({},
+ q,
+ queryArgs
+ );
+ let searchstring = querystring.stringify(newArgs);
+ url = url + "?" + searchstring;
+ return url;
+}
+
+
+function setOrGetFirstrun () {
+ let firstrun = prefs["shield.firstrun"];
+ if (firstrun === undefined) {
+ firstrun = prefs["shield.firstrun"] = String(dateToUTC(new Date())) // in utc, user set
+ }
+ return Number(firstrun)
+}
+
+function reuseVariation (choices) {
+ return prefs["shield.variation"];
+}
+
+function setVariation (choice) {
+ prefs["shield.variation"] = choice
+ return choice
+}
+
+function die (addonId=self.id) {
+ /* istanbul ignore else */
+ if (prefSvc.get("shield.fakedie")) return;
+ /* istanbul ignore next */
+ require("sdk/addon/installer").uninstall(addonId);
+}
+
+// TODO: GRL vulnerable to clock time issues #1
+function expired (xconfig, now = Date.now() ) {
+ return ((now - Number(xconfig.firstrun))/ DAY) > xconfig.days;
+}
+
+function resetShieldPrefs () {
+ delete prefs['shield.firstrun'];
+ delete prefs['shield.variation'];
+}
+
+function cleanup () {
+ prefSvc.keys(`extensions.${self.preferencesBranch}`).forEach (
+ (p) => {
+ delete prefs[p];
+ })
+}
+
+function telemetrySubset (xconfig) {
+ return {
+ study_name: xconfig.name,
+ branch: xconfig.variation,
+ }
+}
+
+class Study extends EventTarget {
+ constructor (config) {
+ super();
+ this.config = merge({
+ name: self.addonId,
+ variations: {'observe-only': () => {}},
+ surveyUrls: {},
+ days: 7
+ },config);
+
+ this.config.firstrun = setOrGetFirstrun();
+
+ let variation = reuseVariation();
+ if (variation === undefined) {
+ variation = this.decideVariation();
+ if (!(variation in this.config.variations)) {
+ // chaijs doesn't think this is an instanceof Error
+ // freaktechnik and gregglind debugged for a while.
+ // sdk errors might not be 'Errors' or chai is wack, who knows.
+ // https://dxr.mozilla.org/mozilla-central/search?q=regexp%3AError%5Cs%3F(%3A%7C%3D)+path%3Aaddon-sdk%2Fsource%2F&redirect=false would list
+ throw new Error("Study Error: chosen variation must be in config.variations")
+ }
+ setVariation(variation);
+ }
+ this.config.variation = variation;
+
+ this.flags = {
+ ineligibleDie: undefined
+ };
+ this.states = [];
+ // all these work, but could be cleaner. I hate the `bind` stuff.
+ this.on(
+ "change", (function (newstate) {
+ prefSvc.get('shield.debug') && console.log(newstate, this.states);
+ this.states.push(newstate);
+ emit(this, newstate); // could have checks here.
+ }).bind(this)
+ )
+ this.on(
+ "starting", (function () {
+ this.changeState("modifying");
+ }).bind(this)
+ )
+ this.on(
+ "maybe-installing", (function () {
+ if (!this.isEligible()) {
+ this.changeState("ineligible-die");
+ } else {
+ this.changeState("installed")
+ }
+ }).bind(this)
+ )
+ this.on(
+ "ineligible-die", (function () {
+ try {this.whenIneligible()} catch (err) {/*ok*/} finally { /*ok*/ }
+ this.flags.ineligibleDie = true;
+ this.report(merge({}, telemetrySubset(this.config), {study_state: "ineligible"}), "shield");
+ this.final();
+ die();
+ }).bind(this)
+ )
+ this.on(
+ "installed", (function () {
+ try {this.whenInstalled()} catch (err) {/*ok*/} finally { /*ok*/ }
+ this.report(merge({}, telemetrySubset(this.config), {study_state: "install"}), "shield");
+ this.changeState("modifying");
+ }).bind(this)
+ )
+ this.on(
+ "modifying", (function () {
+ var mybranchname = this.variation;
+ this.config.variations[mybranchname](); // do the effect
+ this.changeState("running");
+ }).bind(this)
+ )
+ this.on( // the one 'many'
+ "running", (function () {
+ // report success
+ this.report(merge({}, telemetrySubset(this.config), {study_state: "running"}), "shield");
+ this.final();
+ }).bind(this)
+ )
+ this.on(
+ "normal-shutdown", (function () {
+ this.flags.dying = true;
+ this.report(merge({}, telemetrySubset(this.config), {study_state: "shutdown"}), "shield");
+ this.final();
+ }).bind(this)
+ )
+ this.on(
+ "end-of-study", (function () {
+ if (this.flags.expired) { // safe to call multiple times
+ this.final();
+ return;
+ } else {
+ // first time seen.
+ this.flags.expired = true;
+ try {this.whenComplete()} catch (err) { /*ok*/ } finally { /*ok*/ }
+ this.report(merge({}, telemetrySubset(this.config) ,{study_state: "end-of-study"}), "shield");
+ // survey for end of study
+ let that = this;
+ generateTelemetryIdIfNeeded().then(()=>that.showSurvey("end-of-study"));
+ try {this.cleanup()} catch (err) {/*ok*/} finally { /*ok*/ }
+ this.final();
+ die();
+ }
+ }).bind(this)
+ )
+ this.on(
+ "user-uninstall-disable", (function () {
+ if (this.flags.dying) {
+ this.final();
+ return;
+ }
+ this.flags.dying = true;
+ this.report(merge({}, telemetrySubset(this.config), {study_state: "user-ended-study"}), "shield");
+ let that = this;
+ generateTelemetryIdIfNeeded().then(()=>that.showSurvey("user-ended-study"));
+ try {this.cleanup()} catch (err) {/*ok*/} finally { /*ok*/ }
+ this.final();
+ die();
+ }).bind(this)
+ )
+ }
+
+ get state () {
+ let n = this.states.length;
+ return n ? this.states[n-1] : undefined
+ }
+
+ get variation () {
+ return this.config.variation;
+ }
+
+ get firstrun () {
+ return this.config.firstrun;
+ }
+
+ dieIfExpired () {
+ let xconfig = this.config;
+ if (expired(xconfig)) {
+ emit(this, "change", "end-of-study");
+ return true
+ } else {
+ return false
+ }
+ }
+
+ alivenessPulse (last=lastDailyPing) {
+ // check for new day, phone home if true.
+ let t = Date.now();
+ if ((t - last) >= DAY) {
+ lastDailyPing = t;
+ // phone home
+ emit(this,"change","running");
+ }
+ // check expiration, and die with report if needed
+ return this.dieIfExpired();
+ }
+
+ changeState (newstate) {
+ emit(this,'change', newstate);
+ }
+
+ final () {
+ emit(this,'final', {});
+ }
+
+ startup (reason) {
+ // https://developer.mozilla.org/en-US/Add-ons/SDK/Tutorials/Listening_for_load_and_unload
+
+ // check expiry first, before anything, quit and die if so
+
+ // check once, right away, short circuit both install and startup
+ // to prevent modifications from happening.
+ if (this.dieIfExpired()) return this
+
+ switch (reason) {
+ case "install":
+ emit(this, "change", "maybe-installing");
+ break;
+
+ case "enable":
+ case "startup":
+ case "upgrade":
+ case "downgrade":
+ emit(this, "change", "starting");
+ }
+
+ if (! this._pulseTimer) this._pulseTimer = setInterval(this.alivenessPulse.bind(this), 5*60*1000 /*5 minutes */)
+ return this;
+ }
+
+ shutdown (reason) {
+ // https://developer.mozilla.org/en-US/Add-ons/SDK/Tutorials/Listening_for_load_and_unload
+ if (this.flags.ineligibleDie ||
+ this.flags.expired ||
+ this.flags.dying
+ ) { return this } // special cases.
+
+ switch (reason) {
+ case "uninstall":
+ case "disable":
+ emit(this, "change", "user-uninstall-disable");
+ break;
+
+ // 5. usual end of session.
+ case "shutdown":
+ case "upgrade":
+ case "downgrade":
+ emit(this, "change", "normal-shutdown")
+ break;
+ }
+ return this;
+ }
+
+ cleanup () {
+ // do the simple prefs and simplestorage cleanup
+ // extend by extension
+ resetShieldPrefs();
+ cleanup();
+ }
+
+ isEligible () {
+ return true;
+ }
+
+ whenIneligible () {
+ // empty function unless overrided
+ }
+
+ whenInstalled () {
+ // empty unless overrided
+ }
+
+ whenComplete () {
+ // when the study expires
+ }
+
+ /**
+ * equal choice from varations, by default. override to get unequal
+ */
+ decideVariation (rng=Math.random()) {
+ return chooseVariation(Object.keys(this.config.variations), rng);
+ }
+
+ get surveyQueryArgs () {
+ return {
+ variation: this.variation,
+ xname: this.config.name,
+ who: userId(),
+ updateChannel: Services.appinfo.defaultUpdateChannel,
+ fxVersion: Services.appinfo.version,
+ }
+ }
+
+ showSurvey(reason) {
+ let partial = this.config.surveyUrls[reason];
+
+ let queryArgs = this.surveyQueryArgs;
+ queryArgs.reason = reason;
+ if (partial) {
+ let url = survey(partial, queryArgs);
+ tabs.open(url);
+ return url
+ } else {
+ return
+ }
+ }
+
+ report () { // convenience only
+ return report.apply(null, arguments);
+ }
+}
+
+module.exports = {
+ chooseVariation: chooseVariation,
+ die: die,
+ expired: expired,
+ generateTelemetryIdIfNeeded: generateTelemetryIdIfNeeded,
+ report: report,
+ Reporter: Reporter,
+ resetShieldPrefs: resetShieldPrefs,
+ Study: Study,
+ cleanup: cleanup,
+ survey: survey
+}
diff --git a/node_modules/shield-studies-addon-utils/package.json b/node_modules/shield-studies-addon-utils/package.json
new file mode 100644
index 0000000..34eaa81
--- /dev/null
+++ b/node_modules/shield-studies-addon-utils/package.json
@@ -0,0 +1,70 @@
+{
+ "name": "shield-studies-addon-utils",
+ "version": "2.0.0",
+ "description": "Utilities for building Shield-Study Mozilla Firefox addons.",
+ "main": "lib/index.js",
+ "scripts": {
+ "test": "grunt test && istanbul check-coverage --statements 100 --functions 100 --branches 100 --lines 100 coverage/reports/coverage.json"
+ },
+ "author": {
+ "name": "Gregg Lind",
+ "email": "glind@mozilla.com"
+ },
+ "license": "MPL-2.0",
+ "dependencies": {},
+ "devDependencies": {
+ "chai": "3.5.0",
+ "grunt": "1.0.1",
+ "grunt-babel": "6.0.0",
+ "grunt-cli": "1.2.0",
+ "grunt-eslint": "18.1.0",
+ "grunt-explainjs": "0.0.4",
+ "grunt-istanbul": "0.7.0",
+ "grunt-shell": "1.3.0",
+ "istanbul-jpm": "0.1.0",
+ "jpm": "1.1.4"
+ },
+ "bugs": {
+ "url": "https://github.com/gregglind/shield-studies-addon-utils/issues"
+ },
+ "files": [
+ "lib/"
+ ],
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/mozilla/shield-studies-addon-utils.git"
+ },
+ "keywords": [
+ "mozilla",
+ "addon",
+ "shield",
+ "shield-study"
+ ],
+ "homepage": "https://github.com/mozilla/shield-studies-addon-utils#readme",
+ "gitHead": "e12b0dee92afe4bfc26b3aa0c1fac22311d4caaf",
+ "_id": "shield-studies-addon-utils@2.0.0",
+ "_shasum": "cde2579ca6b8444fe5bd69f64443464cdcdcbcf6",
+ "_from": "shield-studies-addon-utils@latest",
+ "_npmVersion": "3.8.3",
+ "_nodeVersion": "5.10.1",
+ "_npmUser": {
+ "name": "gregglind",
+ "email": "gregg.lind@gmail.com"
+ },
+ "maintainers": [
+ {
+ "name": "gregglind",
+ "email": "gregg.lind@gmail.com"
+ }
+ ],
+ "dist": {
+ "shasum": "cde2579ca6b8444fe5bd69f64443464cdcdcbcf6",
+ "tarball": "https://registry.npmjs.org/shield-studies-addon-utils/-/shield-studies-addon-utils-2.0.0.tgz"
+ },
+ "_npmOperationalInternal": {
+ "host": "packages-12-west.internal.npmjs.com",
+ "tmp": "tmp/shield-studies-addon-utils-2.0.0.tgz_1473277641626_0.6047592030372471"
+ },
+ "directories": {},
+ "_resolved": "https://registry.npmjs.org/shield-studies-addon-utils/-/shield-studies-addon-utils-2.0.0.tgz"
+}
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..27ef0a2
--- /dev/null
+++ b/package.json
@@ -0,0 +1,23 @@
+{
+ "name": "unified-urlbar-shield-study",
+ "description": "A Shield Study add-on testing the search experience.",
+ "version": "2.1.1",
+ "author": "Marco Bonardo ",
+ "devDependencies": {
+ "jpm": "^1.0.7",
+ "shield-studies-addon-utils": "^2.0.0"
+ },
+ "engines": {
+ "firefox": ">=50.0a1"
+ },
+ "keywords": [
+ "jetpack",
+ "shield-study"
+ ],
+ "permissions": {
+ "multiprocess": true
+ },
+ "license": "MPL-2.0",
+ "main": "lib/index.js",
+ "title": "Search Shield Study"
+}