Skip to content

Commit

Permalink
Bug 1867098: FxA PXI Toolbar experiment r=teshaq,desktop-theme-review…
Browse files Browse the repository at this point in the history
…ers,flod,dao

Differential Revision: https://phabricator.services.mozilla.com/D194916
  • Loading branch information
skhamis committed Dec 11, 2023
1 parent fbf7297 commit dc64003
Show file tree
Hide file tree
Showing 10 changed files with 383 additions and 12 deletions.
1 change: 1 addition & 0 deletions browser/app/profile/firefox.js
Original file line number Diff line number Diff line change
Expand Up @@ -2438,6 +2438,7 @@ pref("browser.toolbars.bookmarks.showInPrivateBrowsing", false);
pref("identity.fxaccounts.toolbar.enabled", true);
pref("identity.fxaccounts.toolbar.accessed", false);
pref("identity.fxaccounts.toolbar.defaultVisible", false);
pref("identity.fxaccounts.toolbar.pxiToolbarEnabled", false);

// Check bundled omni JARs for corruption.
pref("corroborator.enabled", true);
Expand Down
52 changes: 49 additions & 3 deletions browser/base/content/appmenu-viewcache.inc.xhtml
Original file line number Diff line number Diff line change
Expand Up @@ -559,6 +559,7 @@
data-l10n-id="appmenuitem-fxa-sign-in"/>
<label id="fxa-menu-header-description"
crop="end"
hidden="true"
data-l10n-id="fxa-menu-turn-on-sync"/>
</vbox>
</toolbarbutton>
Expand All @@ -579,6 +580,7 @@
crop="end"/>
</hbox>
</toolbarbutton>
<vbox id="PanelUI-signedin-panel" hidden="true">
<toolbarbutton id="PanelUI-fxa-menu-setup-sync-button"
class="subviewbutton"
data-l10n-id="appmenu-fxa-setup-sync"
Expand All @@ -600,15 +602,59 @@
data-l10n-id="fxa-menu-sync-settings"
hidden="true"
oncommand="gSync.openPrefsFromFxaMenu('sync_settings', this);"/>
<toolbarseparator id="PanelUI-sign-out-separator"/>
<toolbarseparator id="PanelUI-sign-out-separator" />
<toolbarbutton id="PanelUI-fxa-menu-account-signout-button"
class="subviewbutton"
data-l10n-id="fxa-menu-sign-out"
oncommand="gSync.disconnect();"/>
<toolbarseparator id="PanelUI-remote-tabs-separator"/>
oncommand="gSync.disconnect();"
hidden="true"/>
</vbox>
<!-- updateCTAPanel will control if we show this panel -->
<vbox id="PanelUI-fxa-pxi-cta-menu">
<toolbarbutton id="PanelUI-fxa-menu-sync-button" class="subviewbutton subviewbutton-iconic"
oncommand="gSync.openPrefsFromFxaMenu('sync_cta', this);">
<vbox flex="1">
<label id="fxa-menu-header-title" crop="end" data-l10n-id="sync-menu-title" />
<label id="cta-menu-header-description" crop="end" data-l10n-id="sync-menu-description" />
</vbox>
<toolbarseparator id="PanelUI-remote-tabs-separator" />
</toolbarbutton>
<toolbarseparator id="PanelUI-remote-tabs-separator" />
<toolbarbutton id="PanelUI-fxa-menu-monitor-button" class="subviewbutton subviewbutton-iconic"
oncommand="gSync.openMonitorLink(this)">
<vbox flex="1">
<hbox>
<image class="PanelUI-fxa-menu-monitor-button ctaMenuLogo" role="presentation" />
<label id="fxa-menu-header-title" crop="end" data-l10n-id="pxi-menu-monitor-title" />
</hbox>
<label id="cta-menu-header-description" crop="end" data-l10n-id="pxi-menu-monitor-description" />
</vbox>
</toolbarbutton>
<toolbarbutton id="PanelUI-fxa-menu-relay-button" class="subviewbutton subviewbutton-iconic"
oncommand="gSync.openRelayLink(this)">
<vbox flex="1">
<hbox>
<image class="PanelUI-fxa-menu-relay-button ctaMenuLogo" role="presentation" />
<label id="fxa-menu-header-title" crop="end" data-l10n-id="pxi-menu-relay-title" />
</hbox>
<label id="cta-menu-header-description" crop="end" data-l10n-id="pxi-menu-relay-description" />
</vbox>
</toolbarbutton>
<toolbarbutton id="PanelUI-fxa-menu-vpn-button" class="subviewbutton subviewbutton-iconic"
oncommand="gSync.openVPNLink(this)">
<vbox flex="1">
<hbox>
<image class="PanelUI-fxa-menu-vpn-button ctaMenuLogo" role="presentation" />
<label id="fxa-menu-header-title" crop="end" data-l10n-id="pxi-menu-vpn-title" />
</hbox>
<label id="cta-menu-header-description" crop="end" data-l10n-id="pxi-menu-vpn-description" />
</vbox>
</toolbarbutton>
</vbox>
<deck id="PanelUI-fxa-remotetabs-deck">
<!-- Sync is ready to Sync and the "tabs" engine is enabled -->
<vbox id="PanelUI-fxa-remotetabs-tabspane">
<toolbarseparator id="PanelUI-remote-tabs-separator" />
<vbox id="PanelUI-fxa-remotetabs-tabslist"
notabsforclientlabel="appmenu-remote-tabs-notabs"
/>
Expand Down
168 changes: 160 additions & 8 deletions browser/base/content/browser-sync.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,14 @@
// This file is loaded into the browser window scope.
/* eslint-env mozilla/browser-window */

const {
FX_MONITOR_OAUTH_CLIENT_ID,
FX_RELAY_OAUTH_CLIENT_ID,
VPN_OAUTH_CLIENT_ID,
} = ChromeUtils.importESModule(
"resource://gre/modules/FxAccountsCommon.sys.mjs"
);

const { UIState } = ChromeUtils.importESModule(
"resource://services-sync/UIState.sys.mjs"
);
Expand Down Expand Up @@ -348,6 +356,8 @@ var gSync = {
"browser/appmenu.ftl",
"browser/sync.ftl",
"toolkit/branding/accounts.ftl",
// untranslated FTL
"preview/appmenu.ftl",
],
true
));
Expand Down Expand Up @@ -406,6 +416,11 @@ var gSync = {
"FXA_ENABLED",
"identity.fxaccounts.enabled"
);
XPCOMUtils.defineLazyPreferenceGetter(
this,
"PXI_TOOLBAR_ENABLED",
"identity.fxaccounts.toolbar.pxiToolbarEnabled"
);
},

maybeUpdateUIState() {
Expand Down Expand Up @@ -492,6 +507,13 @@ var gSync = {
fxaPanelView.addEventListener("ViewShowing", this);
fxaPanelView.addEventListener("ViewHiding", this);

// If the experiment is enabled, we'll need to update the panels
// to show some different text to the user
if (this.PXI_TOOLBAR_ENABLED) {
this.updateFxAPanel(UIState.get());
this.updateCTAPanel();
}

this._initialized = true;
},

Expand Down Expand Up @@ -536,6 +558,14 @@ var gSync = {
);
syncPrefsButtonEl.hidden = !UIState.get().syncEnabled;

// We should ensure that we do not show the sign out button
// if the user is not signed in
const signOutButtonEl = PanelMultiView.getViewNode(
document,
"PanelUI-fxa-menu-account-signout-button"
);
signOutButtonEl.hidden = !this.isSignedIn;

panelview.syncedTabsPanelList = new SyncedTabsPanelList(
panelview,
PanelMultiView.getViewNode(document, "PanelUI-fxa-remotetabs-deck"),
Expand Down Expand Up @@ -580,6 +610,7 @@ var gSync = {
this.updateSyncButtonsTooltip(state);
this.updateSyncStatus(state);
this.updateFxAPanel(state);
this.updateCTAPanel(state);
// Ensure we have something in the device list in the background.
this.ensureFxaDevices();
},
Expand Down Expand Up @@ -786,15 +817,29 @@ var gSync = {
entrypoint_variation: fxaButtonVisibilityExperiment.branch.slug,
};
}

let panel =
anchor.id == "appMenu-fxa-label2"
? PanelMultiView.getViewNode(document, "PanelUI-fxa")
: undefined;
this.openFxAEmailFirstPageFromFxaMenu(panel, extraParams);
PanelUI.hide();
// If we're signed out but have the PXI pref enabled
// we should show the PXI panel instead of taking the user
// straight to FxA sign-in
if (this.PXI_TOOLBAR_ENABLED) {
this.updateCTAPanel();
PanelUI.showSubView("PanelUI-fxa", anchor, aEvent);
} else {
let panel =
anchor.id == "appMenu-fxa-label2"
? PanelMultiView.getViewNode(document, "PanelUI-fxa")
: undefined;
this.openFxAEmailFirstPageFromFxaMenu(panel, extraParams);
PanelUI.hide();
}
return;
}
// If the user is signed in and we have the PXI pref enabled then add
// the pxi panel to the existing toolbar
if (this.PXI_TOOLBAR_ENABLED) {
this.updateCTAPanel();
} else {
PanelUI.showSubView("PanelUI-fxa", anchor, aEvent);
}

if (!gFxaToolbarAccessed) {
Services.prefs.setBoolPref("identity.fxaccounts.toolbar.accessed", true);
Expand Down Expand Up @@ -856,19 +901,33 @@ var gSync = {
"fxa-manage-account-button"
);

const signedInContainer = PanelMultiView.getViewNode(
document,
"PanelUI-signedin-panel"
);

cadButtonEl.setAttribute("disabled", true);
syncNowButtonEl.hidden = true;
signedInContainer.hidden = true;
fxaMenuAccountButtonEl.classList.remove("subviewbutton-nav");
fxaMenuAccountButtonEl.removeAttribute("closemenu");
syncSetupButtonEl.removeAttribute("hidden");

let headerTitleL10nId = "appmenuitem-fxa-sign-in";
let headerTitleL10nId = this.PXI_TOOLBAR_ENABLED
? "appmenuitem-moz-accounts-sign-in"
: "appmenuitem-fxa-sign-in";
let headerDescription;
if (state.status === UIState.STATUS_NOT_CONFIGURED) {
mainWindowEl.style.removeProperty("--avatar-image-url");
headerDescription = this.fluentStrings.formatValueSync(
"appmenu-fxa-signed-in-label"
);
// Signed out, expeirment enabled is the only state we want to hide the
// header description, so we make it empty and check for that when setting
// the value
if (this.PXI_TOOLBAR_ENABLED) {
headerDescription = "";
}
} else if (state.status === UIState.STATUS_LOGIN_FAILED) {
stateValue = "login-failed";
headerTitleL10nId = "account-disconnected2";
Expand All @@ -895,6 +954,8 @@ var gSync = {
mainWindowEl.style.removeProperty("--avatar-image-url");
};
img.src = state.avatarURL;
signedInContainer.hidden = false;
menuHeaderDescriptionEl.hidden = false;
} else {
mainWindowEl.style.removeProperty("--avatar-image-url");
}
Expand All @@ -917,6 +978,8 @@ var gSync = {

menuHeaderTitleEl.value =
this.fluentStrings.formatValueSync(headerTitleL10nId);
// If we description is empty, we hide it
menuHeaderDescriptionEl.hidden = !headerDescription;
menuHeaderDescriptionEl.value = headerDescription;
// We remove the data-l10n-id attribute here to prevent the node's value
// attribute from being overwritten by Fluent when the panel is moved
Expand Down Expand Up @@ -1959,6 +2022,95 @@ var gSync = {
}
},

// This should only be shown if we have enabled the pxiPanel via
// an experiment or explicitly through prefs
updateCTAPanel() {
const mainPanelEl = PanelMultiView.getViewNode(
document,
"PanelUI-fxa-pxi-cta-menu"
);

const syncCtaEl = PanelMultiView.getViewNode(
document,
"PanelUI-fxa-menu-sync-button"
);
// If we're not in the experiment then we do not enable this at all
if (!this.PXI_TOOLBAR_ENABLED) {
// If we've previously shown this but got disabled
// we should ensure we hide the panel
mainPanelEl.hidden = true;
return;
}

// If we're already signed in an syncing, we shouldn't show the sync CTA
syncCtaEl.hidden = this.isSignedIn;

// ensure the container is visible
mainPanelEl.hidden = false;
},
async openMonitorLink(panel) {
this.emitFxaToolbarTelemetry("monitor_cta", panel);
await this.openPXILink(
FX_MONITOR_OAUTH_CLIENT_ID,
new URL("https://monitor.firefox.com"),
new URL("https://monitor.firefox.com/user/breaches")
);
},

async openRelayLink(panel) {
this.emitFxaToolbarTelemetry("relay_cta", panel);
await this.openPXILink(
FX_RELAY_OAUTH_CLIENT_ID,
new URL("https://relay.firefox.com"),
new URL("https://relay.firefox.com/accounts/profile")
);
},

async openVPNLink(panel) {
this.emitFxaToolbarTelemetry("vpn_cta", panel);
await this.openPXILink(
VPN_OAUTH_CLIENT_ID,
new URL("https://www.mozilla.org/en-US/products/vpn/"),
new URL("https://www.mozilla.org/en-US/products/vpn/")
);
},

// A generic opening based on
async openPXILink(clientId, defaultUrl, signedInUrl) {
const params = {
utm_medium: "firefox-desktop",
utm_source: "toolbar",
utm_campaign: "discovery",
};
const pxiSearchParams = new URLSearchParams(params);

if (!this.isSignedIn) {
// Add the base params + not signed in
defaultUrl.search = pxiSearchParams.toString();
defaultUrl.searchParams.append("utm_content", "notsignedin");
this.openLink(defaultUrl);
PanelUI.hide();
return;
}

// Note: This is a network call
let attachedClients = await fxAccounts.listAttachedOAuthClients();
// If we have at least one client based on clientId passed in
let hasPXIClient = attachedClients.some(c => !!c.id && c.id === clientId);

const url = hasPXIClient ? signedInUrl : defaultUrl;
// Add base params + signed in
url.search = pxiSearchParams.toString();
url.searchParams.append("utm_content", "signedIn");

this.openLink(url);
PanelUI.hide();
},

openLink(url) {
switchToTabHavingURI(url, true, { replaceQueryString: true });
},

QueryInterface: ChromeUtils.generateQI([
"nsIObserver",
"nsISupportsWeakReference",
Expand Down
1 change: 1 addition & 0 deletions browser/base/content/browser.xhtml
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,7 @@
<link rel="localization" href="preview/interventions.ftl" />
<link rel="localization" href="browser/shopping.ftl"/>
<link rel="localization" href="preview/shopping.ftl"/>
<link rel="localization" href="preview/appmenu.ftl" />

<title data-l10n-id="browser-main-window-title"></title>

Expand Down
Loading

0 comments on commit dc64003

Please sign in to comment.