diff options
Diffstat (limited to 'browser/components/aboutwelcome/actors')
-rw-r--r-- | browser/components/aboutwelcome/actors/AboutWelcomeChild.sys.mjs | 898 | ||||
-rw-r--r-- | browser/components/aboutwelcome/actors/AboutWelcomeParent.sys.mjs | 299 |
2 files changed, 1197 insertions, 0 deletions
diff --git a/browser/components/aboutwelcome/actors/AboutWelcomeChild.sys.mjs b/browser/components/aboutwelcome/actors/AboutWelcomeChild.sys.mjs new file mode 100644 index 0000000000..24bf73c80a --- /dev/null +++ b/browser/components/aboutwelcome/actors/AboutWelcomeChild.sys.mjs @@ -0,0 +1,898 @@ +/* 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/. */ + +import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs"; + +const lazy = {}; + +ChromeUtils.defineESModuleGetters(lazy, { + AboutWelcomeDefaults: + "resource:///modules/aboutwelcome/AboutWelcomeDefaults.sys.mjs", + ExperimentAPI: "resource://nimbus/ExperimentAPI.sys.mjs", + NimbusFeatures: "resource://nimbus/ExperimentAPI.sys.mjs", +}); + +ChromeUtils.defineLazyGetter(lazy, "log", () => { + const { Logger } = ChromeUtils.importESModule( + "resource://messaging-system/lib/Logger.sys.mjs" + ); + return new Logger("AboutWelcomeChild"); +}); + +export class AboutWelcomeChild extends JSWindowActorChild { + // Can be used to avoid accesses to the document/contentWindow after it's + // destroyed, which may throw unhandled exceptions. + _destroyed = false; + + didDestroy() { + this._destroyed = true; + } + + actorCreated() { + this.exportFunctions(); + } + + /** + * Send event that can be handled by the page + * + * @param {{type: string, data?: any}} action + */ + sendToPage(action) { + lazy.log.debug(`Sending to page: ${action.type}`); + const win = this.document.defaultView; + const event = new win.CustomEvent("AboutWelcomeChromeToContent", { + detail: Cu.cloneInto(action, win), + }); + win.dispatchEvent(event); + } + + /** + * Export functions that can be called by page js + */ + exportFunctions() { + let window = this.contentWindow; + + Cu.exportFunction(this.AWAddScreenImpression.bind(this), window, { + defineAs: "AWAddScreenImpression", + }); + + Cu.exportFunction(this.AWGetFeatureConfig.bind(this), window, { + defineAs: "AWGetFeatureConfig", + }); + + Cu.exportFunction(this.AWGetFxAMetricsFlowURI.bind(this), window, { + defineAs: "AWGetFxAMetricsFlowURI", + }); + + Cu.exportFunction(this.AWGetSelectedTheme.bind(this), window, { + defineAs: "AWGetSelectedTheme", + }); + + Cu.exportFunction(this.AWSelectTheme.bind(this), window, { + defineAs: "AWSelectTheme", + }); + + Cu.exportFunction(this.AWEvaluateScreenTargeting.bind(this), window, { + defineAs: "AWEvaluateScreenTargeting", + }); + + Cu.exportFunction(this.AWSendEventTelemetry.bind(this), window, { + defineAs: "AWSendEventTelemetry", + }); + + Cu.exportFunction(this.AWSendToParent.bind(this), window, { + defineAs: "AWSendToParent", + }); + + Cu.exportFunction(this.AWWaitForMigrationClose.bind(this), window, { + defineAs: "AWWaitForMigrationClose", + }); + + Cu.exportFunction(this.AWFinish.bind(this), window, { + defineAs: "AWFinish", + }); + + Cu.exportFunction(this.AWEnsureAddonInstalled.bind(this), window, { + defineAs: "AWEnsureAddonInstalled", + }); + + Cu.exportFunction(this.AWEnsureLangPackInstalled.bind(this), window, { + defineAs: "AWEnsureLangPackInstalled", + }); + + Cu.exportFunction( + this.AWNegotiateLangPackForLanguageMismatch.bind(this), + window, + { + defineAs: "AWNegotiateLangPackForLanguageMismatch", + } + ); + + Cu.exportFunction(this.AWSetRequestedLocales.bind(this), window, { + defineAs: "AWSetRequestedLocales", + }); + + Cu.exportFunction(this.AWSendToDeviceEmailsSupported.bind(this), window, { + defineAs: "AWSendToDeviceEmailsSupported", + }); + + Cu.exportFunction(this.AWNewScreen.bind(this), window, { + defineAs: "AWNewScreen", + }); + } + + /** + * Wrap a promise so content can use Promise methods. + */ + wrapPromise(promise) { + return new this.contentWindow.Promise((resolve, reject) => + promise.then(resolve, reject) + ); + } + + /** + * Clones the result of the query into the content window. + */ + sendQueryAndCloneForContent(...sendQueryArgs) { + return this.wrapPromise( + (async () => { + return Cu.cloneInto( + await this.sendQuery(...sendQueryArgs), + this.contentWindow + ); + })() + ); + } + + AWSelectTheme(data) { + return this.wrapPromise( + this.sendQuery("AWPage:SELECT_THEME", data.toUpperCase()) + ); + } + + AWEvaluateScreenTargeting(data) { + return this.sendQueryAndCloneForContent( + "AWPage:EVALUATE_SCREEN_TARGETING", + data + ); + } + + AWAddScreenImpression(screen) { + return this.wrapPromise( + this.sendQuery("AWPage:ADD_SCREEN_IMPRESSION", screen) + ); + } + + /** + * Send initial data to page including experiment information + */ + async getAWContent() { + let attributionData = await this.sendQuery("AWPage:GET_ATTRIBUTION_DATA"); + + // Return to AMO gets returned early. + if (attributionData?.template) { + lazy.log.debug("Loading about:welcome with RTAMO attribution data"); + return Cu.cloneInto(attributionData, this.contentWindow); + } else if (attributionData?.ua) { + lazy.log.debug("Loading about:welcome with UA attribution"); + } + + let experimentMetadata = + lazy.ExperimentAPI.getExperimentMetaData({ + featureId: "aboutwelcome", + }) || {}; + + lazy.log.debug( + `Loading about:welcome with ${ + experimentMetadata?.slug ?? "no" + } experiment` + ); + + let featureConfig = lazy.NimbusFeatures.aboutwelcome.getAllVariables(); + featureConfig.needDefault = await this.sendQuery("AWPage:NEED_DEFAULT"); + featureConfig.needPin = await this.sendQuery("AWPage:DOES_APP_NEED_PIN"); + if (featureConfig.languageMismatchEnabled) { + featureConfig.appAndSystemLocaleInfo = await this.sendQuery( + "AWPage:GET_APP_AND_SYSTEM_LOCALE_INFO" + ); + } + + // FeatureConfig (from experiments) has higher precendence + // to defaults. But the `screens` property isn't defined we shouldn't + // override the default with `null` + let defaults = lazy.AboutWelcomeDefaults.getDefaults(); + + const content = await lazy.AboutWelcomeDefaults.prepareContentForReact({ + ...attributionData, + ...experimentMetadata, + ...defaults, + ...featureConfig, + screens: featureConfig.screens ?? defaults.screens, + backdrop: featureConfig.backdrop ?? defaults.backdrop, + }); + + return Cu.cloneInto(content, this.contentWindow); + } + + AWGetFeatureConfig() { + return this.wrapPromise(this.getAWContent()); + } + + AWGetFxAMetricsFlowURI() { + return this.wrapPromise(this.sendQuery("AWPage:FXA_METRICS_FLOW_URI")); + } + + AWGetSelectedTheme() { + return this.wrapPromise(this.sendQuery("AWPage:GET_SELECTED_THEME")); + } + + /** + * Send Event Telemetry + * + * @param {object} eventData + */ + AWSendEventTelemetry(eventData) { + this.AWSendToParent("TELEMETRY_EVENT", { + ...eventData, + event_context: { + ...eventData.event_context, + }, + }); + } + + /** + * Send message that can be handled by AboutWelcomeParent.jsm + * + * @param {string} type + * @param {any=} data + * @returns {Promise<unknown>} + */ + AWSendToParent(type, data) { + return this.sendQueryAndCloneForContent(`AWPage:${type}`, data); + } + + AWWaitForMigrationClose() { + return this.wrapPromise(this.sendQuery("AWPage:WAIT_FOR_MIGRATION_CLOSE")); + } + + AWFinish() { + const shouldFocusNewtabUrlBar = + lazy.NimbusFeatures.aboutwelcome.getVariable("newtabUrlBarFocus"); + + this.contentWindow.location.href = "about:home"; + if (shouldFocusNewtabUrlBar) { + this.AWSendToParent("SPECIAL_ACTION", { + type: "FOCUS_URLBAR", + }); + } + } + + AWEnsureAddonInstalled(addonId) { + return this.wrapPromise( + this.sendQuery("AWPage:ENSURE_ADDON_INSTALLED", addonId) + ); + } + + AWEnsureLangPackInstalled(negotiated, screenContent) { + const content = Cu.cloneInto(screenContent, {}); + return this.wrapPromise( + this.sendQuery( + "AWPage:ENSURE_LANG_PACK_INSTALLED", + negotiated.langPack + ).then(() => { + const formatting = []; + const l10n = new Localization( + ["branding/brand.ftl", "browser/newtab/onboarding.ftl"], + false, + undefined, + // Use the system-ish then app then default locale. + [...negotiated.requestSystemLocales, "en-US"] + ); + + // Add the negotiated language name as args. + function addMessageArgsAndUseLangPack(obj) { + for (const value of Object.values(obj)) { + if (value?.string_id) { + value.args = { + ...value.args, + negotiatedLanguage: negotiated.langPackDisplayName, + }; + + // Expose fluent strings wanting lang pack as raw. + if (value.useLangPack) { + formatting.push( + l10n.formatValue(value.string_id, value.args).then(raw => { + delete value.string_id; + value.raw = raw; + }) + ); + } + } + } + } + addMessageArgsAndUseLangPack(content.languageSwitcher); + addMessageArgsAndUseLangPack(content); + return Promise.all(formatting).then(() => + Cu.cloneInto(content, this.contentWindow) + ); + }) + ); + } + + AWSetRequestedLocales(requestSystemLocales) { + return this.sendQueryAndCloneForContent( + "AWPage:SET_REQUESTED_LOCALES", + requestSystemLocales + ); + } + + AWNegotiateLangPackForLanguageMismatch(appAndSystemLocaleInfo) { + return this.sendQueryAndCloneForContent( + "AWPage:NEGOTIATE_LANGPACK", + appAndSystemLocaleInfo + ); + } + + AWSendToDeviceEmailsSupported() { + return this.wrapPromise( + this.sendQuery("AWPage:SEND_TO_DEVICE_EMAILS_SUPPORTED") + ); + } + + AWNewScreen(screenId) { + return this.wrapPromise(this.sendQuery("AWPage:NEW_SCREEN", screenId)); + } + + /** + * @param {{type: string, detail?: any}} event + * @override + */ + handleEvent(event) { + lazy.log.debug(`Received page event ${event.type}`); + } +} + +const OPTIN_DEFAULT = { + id: "FAKESPOT_OPTIN_DEFAULT", + template: "multistage", + backdrop: "transparent", + aria_role: "alert", + UTMTerm: "opt-in", + screens: [ + { + id: "FS_OPT_IN", + content: { + position: "split", + title: { string_id: "shopping-onboarding-headline" }, + // We set the dynamic subtitle ID below at the same time as the args; + // to prevent intermittents caused by the args loading too late. + subtitle: { string_id: "" }, + above_button_content: [ + { + type: "text", + text: { + string_id: "shopping-onboarding-body", + }, + link_keys: ["learn_more"], + }, + { + type: "image", + url: "chrome://browser/content/shopping/assets/optInLight.avif", + darkModeImageURL: + "chrome://browser/content/shopping/assets/optInDark.avif", + marginInline: "24px", + }, + { + type: "text", + text: { + string_id: + "shopping-onboarding-opt-in-privacy-policy-and-terms-of-use3", + }, + link_keys: ["privacy_policy", "terms_of_use"], + font_styles: "legal", + }, + ], + learn_more: { + action: { + type: "OPEN_URL", + data: { + args: "https://support.mozilla.org/1/firefox/%VERSION%/%OS%/%LOCALE%/review-checker-review-quality?utm_source=review-checker&utm_campaign=learn-more&utm_medium=in-product", + where: "tab", + }, + }, + }, + privacy_policy: { + action: { + type: "OPEN_URL", + data: { + args: "https://www.mozilla.org/privacy/firefox?utm_source=review-checker&utm_campaign=privacy-policy&utm_medium=in-product&utm_term=opt-in-screen", + where: "tab", + }, + }, + }, + terms_of_use: { + action: { + type: "OPEN_URL", + data: { + args: "https://www.fakespot.com/terms?utm_source=review-checker&utm_campaign=terms-of-use&utm_medium=in-product", + where: "tab", + }, + }, + }, + primary_button: { + should_focus_button: true, + label: { string_id: "shopping-onboarding-opt-in-button" }, + action: { + type: "SET_PREF", + data: { + pref: { + name: "browser.shopping.experience2023.optedIn", + value: 1, + }, + }, + }, + }, + additional_button: { + label: { + string_id: "shopping-onboarding-not-now-button", + }, + style: "link", + flow: "column", + action: { + type: "SET_PREF", + data: { + pref: { + name: "browser.shopping.experience2023.active", + value: false, + }, + }, + }, + }, + }, + }, + ], +}; + +const SHOPPING_MICROSURVEY = { + id: "SHOPPING_MICROSURVEY", + template: "multistage", + backdrop: "transparent", + transitions: true, + UTMTerm: "survey", + screens: [ + { + id: "SHOPPING_MICROSURVEY_SCREEN_1", + above_button_steps_indicator: true, + content: { + position: "split", + layout: "survey", + steps_indicator: { + string_id: "shopping-onboarding-welcome-steps-indicator-label", + }, + title: { + string_id: "shopping-survey-headline", + }, + primary_button: { + label: { + string_id: "shopping-survey-next-button-label", + paddingBlock: "5px", + marginBlock: "0 12px", + }, + action: { + type: "MULTI_ACTION", + collectSelect: true, + data: { + actions: [], + }, + navigate: true, + }, + disabled: "hasActiveMultiSelect", + }, + additional_button: { + label: { + string_id: "shopping-survey-terms-link", + }, + style: "link", + flow: "column", + action: { + type: "OPEN_URL", + data: { + args: "https://www.mozilla.org/about/legal/terms/mozilla/?utm_source=review-checker&utm_campaign=terms-of-use-screen-1&utm_medium=in-product", + where: "tab", + }, + }, + }, + dismiss_button: { + action: { + dismiss: true, + }, + label: { + string_id: "shopping-onboarding-dialog-close-button", + }, + }, + tiles: { + type: "multiselect", + style: { + flexDirection: "column", + alignItems: "flex-start", + }, + label: { + string_id: "shopping-survey-question-one", + }, + data: [ + { + id: "radio-1", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q1-radio-1-label" }, + }, + { + id: "radio-2", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q1-radio-2-label" }, + }, + { + id: "radio-3", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q1-radio-3-label" }, + }, + { + id: "radio-4", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q1-radio-4-label" }, + }, + { + id: "radio-5", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q1-radio-5-label" }, + }, + ], + }, + }, + }, + { + id: "SHOPPING_MICROSURVEY_SCREEN_2", + above_button_steps_indicator: true, + content: { + position: "split", + layout: "survey", + steps_indicator: { + string_id: "shopping-onboarding-welcome-steps-indicator-label", + }, + title: { + string_id: "shopping-survey-headline", + }, + primary_button: { + label: { + string_id: "shopping-survey-submit-button-label", + paddingBlock: "5px", + marginBlock: "0 12px", + }, + action: { + type: "MULTI_ACTION", + collectSelect: true, + data: { + actions: [], + }, + navigate: true, + }, + disabled: "hasActiveMultiSelect", + }, + additional_button: { + label: { + string_id: "shopping-survey-terms-link", + }, + style: "link", + flow: "column", + action: { + type: "OPEN_URL", + data: { + args: "https://www.mozilla.org/about/legal/terms/mozilla/?utm_source=review-checker&utm_campaign=terms-of-use-screen-2&utm_medium=in-product", + where: "tab", + }, + }, + }, + dismiss_button: { + action: { + dismiss: true, + }, + label: { + string_id: "shopping-onboarding-dialog-close-button", + }, + }, + tiles: { + type: "multiselect", + style: { + flexDirection: "column", + alignItems: "flex-start", + }, + label: { + string_id: "shopping-survey-question-two", + }, + data: [ + { + id: "radio-1", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q2-radio-1-label" }, + }, + { + id: "radio-2", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q2-radio-2-label" }, + }, + { + id: "radio-3", + type: "radio", + group: "radios", + defaultValue: false, + label: { string_id: "shopping-survey-q2-radio-3-label" }, + }, + ], + }, + }, + }, + ], +}; + +const OPTED_IN_TIME_PREF = "browser.shopping.experience2023.survey.optedInTime"; + +XPCOMUtils.defineLazyPreferenceGetter( + lazy, + "isSurveySeen", + "browser.shopping.experience2023.survey.hasSeen", + false +); + +XPCOMUtils.defineLazyPreferenceGetter( + lazy, + "pdpVisits", + "browser.shopping.experience2023.survey.pdpVisits", + 0 +); + +XPCOMUtils.defineLazyPreferenceGetter( + lazy, + "optedInTime", + OPTED_IN_TIME_PREF, + 0 +); + +let optInDynamicContent; +// Limit pref increase to 5 as we don't need to count any higher than that +const MIN_VISITS_TO_SHOW_SURVEY = 5; +// Wait 24 hours after opt in to show survey +const MIN_TIME_AFTER_OPT_IN = 24 * 60 * 60; + +export class AboutWelcomeShoppingChild extends AboutWelcomeChild { + // Static state used to track session in which user opted-in + static optedInSession = false; + + // Static used to track PDP visits per session for showing survey + static eligiblePDPvisits = []; + + constructor() { + super(); + this.surveyEnabled = + lazy.NimbusFeatures.shopping2023.getVariable("surveyEnabled"); + + // Used by tests + this.resetChildStates = () => { + AboutWelcomeShoppingChild.eligiblePDPvisits.length = 0; + AboutWelcomeShoppingChild.optedInSession = false; + }; + } + + computeEligiblePDPCount(data) { + // Increment our pref if this isn't a page we've already seen this session + if (lazy.pdpVisits < MIN_VISITS_TO_SHOW_SURVEY) { + this.AWSendToParent("SPECIAL_ACTION", { + type: "SET_PREF", + data: { + pref: { + name: "browser.shopping.experience2023.survey.pdpVisits", + value: lazy.pdpVisits + 1, + }, + }, + }); + } + + // Add this product to our list of unique eligible PDPs visited + // to prevent errors caused by multiple events being fired simultaneously + AboutWelcomeShoppingChild.eligiblePDPvisits.push(data?.product_id); + } + + evaluateAndShowSurvey() { + // Re-evaluate if we should show the survey + // Render survey if user is opted-in and has met survey seen conditions + const now = Date.now() / 1000; + const hasBeen24HrsSinceOptin = + lazy.optedInTime && now - lazy.optedInTime >= MIN_TIME_AFTER_OPT_IN; + + this.showMicroSurvey = + this.surveyEnabled && + !lazy.isSurveySeen && + !AboutWelcomeShoppingChild.optedInSession && + lazy.pdpVisits >= MIN_VISITS_TO_SHOW_SURVEY && + hasBeen24HrsSinceOptin; + + if (this.showMicroSurvey) { + this.renderMessage(); + } + } + + setOptInTime() { + const now = Date.now() / 1000; + this.AWSendToParent("SPECIAL_ACTION", { + type: "SET_PREF", + data: { + pref: { + name: OPTED_IN_TIME_PREF, + value: now, + }, + }, + }); + } + + handleEvent(event) { + // Decide when to show/hide onboarding and survey message + const { productUrl, showOnboarding, data } = event.detail; + + // Display onboarding if a user hasn't opted-in + const optInReady = showOnboarding && productUrl; + if (optInReady) { + // Render opt-in message + AboutWelcomeShoppingChild.optedInSession = true; + this.AWSetProductURL(new URL(productUrl).hostname); + this.renderMessage(); + return; + } + + //Store timestamp if user opts in + if ( + Object.hasOwn(event.detail, "showOnboarding") && + !event.detail.showOnboarding && + !lazy.optedInTime + ) { + this.setOptInTime(); + } + // Hide the container until the user is eligible to see the survey + // or user has just completed opt-in + if (!lazy.isSurveySeen || AboutWelcomeShoppingChild.optedInSession) { + this.document.getElementById("multi-stage-message-root").hidden = true; + } + + // Early exit if user has seen survey, if we have no data, encountered + // an error, or if pdp is ineligible or not unique + if ( + lazy.isSurveySeen || + !data || + data.error || + !productUrl || + (data.needs_analysis && + (!data.product_id || !data.grade || !data.adjusted_rating)) || + AboutWelcomeShoppingChild.eligiblePDPvisits.includes(data.product_id) + ) { + return; + } + + this.computeEligiblePDPCount(data, productUrl); + this.evaluateAndShowSurvey(); + } + + renderMessage() { + this.document.getElementById("multi-stage-message-root").hidden = false; + this.document.dispatchEvent( + new this.contentWindow.CustomEvent("RenderWelcome", { + bubbles: true, + }) + ); + } + + // TODO - Move messages into an ASRouter message provider. See bug 1848251. + AWGetFeatureConfig() { + let messageContent = optInDynamicContent; + if (this.showMicroSurvey) { + messageContent = SHOPPING_MICROSURVEY; + this.setShoppingSurveySeen(); + } + return Cu.cloneInto(messageContent, this.contentWindow); + } + + setShoppingSurveySeen() { + this.AWSendToParent("SPECIAL_ACTION", { + type: "SET_PREF", + data: { + pref: { + name: "browser.shopping.experience2023.survey.hasSeen", + value: true, + }, + }, + }); + } + + // TODO - Add dismiss: true to the primary CTA so it cleans up the React + // content, which will stop being rendered on opt-in. See bug 1848429. + AWFinish() { + if (this._destroyed) { + return; + } + const root = this.document.getElementById("multi-stage-message-root"); + if (root) { + root.innerHTML = ""; + root + .appendChild(this.document.createElement("shopping-message-bar")) + .setAttribute("type", "thank-you-for-feedback"); + this.contentWindow.setTimeout(() => { + root.hidden = true; + }, 5000); + } + } + + AWSetProductURL(productUrl) { + let content = JSON.parse(JSON.stringify(OPTIN_DEFAULT)); + const [optInScreen] = content.screens; + + if (productUrl) { + optInScreen.content.subtitle.string_id = + "shopping-onboarding-dynamic-subtitle-1"; + + switch ( + productUrl // Insert the productUrl into content + ) { + case "www.amazon.fr": + case "www.amazon.de": + optInScreen.content.subtitle.string_id = + "shopping-onboarding-single-subtitle"; + optInScreen.content.subtitle.args = { + currentSite: "Amazon", + }; + break; + case "www.amazon.com": + optInScreen.content.subtitle.args = { + currentSite: "Amazon", + secondSite: "Walmart", + thirdSite: "Best Buy", + }; + break; + case "www.walmart.com": + optInScreen.content.subtitle.args = { + currentSite: "Walmart", + secondSite: "Amazon", + thirdSite: "Best Buy", + }; + break; + case "www.bestbuy.com": + optInScreen.content.subtitle.args = { + currentSite: "Best Buy", + secondSite: "Amazon", + thirdSite: "Walmart", + }; + break; + default: + optInScreen.content.subtitle.args = { + currentSite: "Amazon", + secondSite: "Walmart", + thirdSite: "Best Buy", + }; + } + } + + optInDynamicContent = content; + } + + AWEnsureLangPackInstalled() {} +} diff --git a/browser/components/aboutwelcome/actors/AboutWelcomeParent.sys.mjs b/browser/components/aboutwelcome/actors/AboutWelcomeParent.sys.mjs new file mode 100644 index 0000000000..1eb77da0d8 --- /dev/null +++ b/browser/components/aboutwelcome/actors/AboutWelcomeParent.sys.mjs @@ -0,0 +1,299 @@ +/* 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/. */ + +const lazy = {}; + +ChromeUtils.defineESModuleGetters(lazy, { + AboutWelcomeDefaults: + "resource:///modules/aboutwelcome/AboutWelcomeDefaults.sys.mjs", + AboutWelcomeTelemetry: + "resource:///modules/aboutwelcome/AboutWelcomeTelemetry.sys.mjs", + AddonManager: "resource://gre/modules/AddonManager.sys.mjs", + AWScreenUtils: "resource:///modules/aboutwelcome/AWScreenUtils.sys.mjs", + BrowserUtils: "resource://gre/modules/BrowserUtils.sys.mjs", + BuiltInThemes: "resource:///modules/BuiltInThemes.sys.mjs", + FxAccounts: "resource://gre/modules/FxAccounts.sys.mjs", + LangPackMatcher: "resource://gre/modules/LangPackMatcher.sys.mjs", + ShellService: "resource:///modules/ShellService.sys.mjs", + SpecialMessageActions: + "resource://messaging-system/lib/SpecialMessageActions.sys.mjs", +}); + +ChromeUtils.defineLazyGetter(lazy, "log", () => { + const { Logger } = ChromeUtils.importESModule( + "resource://messaging-system/lib/Logger.sys.mjs" + ); + return new Logger("AboutWelcomeParent"); +}); + +ChromeUtils.defineLazyGetter( + lazy, + "Telemetry", + () => new lazy.AboutWelcomeTelemetry() +); + +const DID_SEE_ABOUT_WELCOME_PREF = "trailhead.firstrun.didSeeAboutWelcome"; +const AWTerminate = { + WINDOW_CLOSED: "welcome-window-closed", + TAB_CLOSED: "welcome-tab-closed", + APP_SHUT_DOWN: "app-shut-down", + ADDRESS_BAR_NAVIGATED: "address-bar-navigated", +}; +const LIGHT_WEIGHT_THEMES = { + AUTOMATIC: "default-theme@mozilla.org", + DARK: "firefox-compact-dark@mozilla.org", + LIGHT: "firefox-compact-light@mozilla.org", + ALPENGLOW: "firefox-alpenglow@mozilla.org", +}; + +class AboutWelcomeObserver { + constructor() { + Services.obs.addObserver(this, "quit-application"); + + this.win = Services.focus.activeWindow; + if (!this.win) { + return; + } + + this.terminateReason = AWTerminate.ADDRESS_BAR_NAVIGATED; + + this.onWindowClose = () => { + this.terminateReason = AWTerminate.WINDOW_CLOSED; + }; + + this.onTabClose = () => { + this.terminateReason = AWTerminate.TAB_CLOSED; + }; + + this.win.addEventListener("TabClose", this.onTabClose, { once: true }); + this.win.addEventListener("unload", this.onWindowClose, { once: true }); + } + + observe(aSubject, aTopic, aData) { + switch (aTopic) { + case "quit-application": + this.terminateReason = AWTerminate.APP_SHUT_DOWN; + break; + } + } + + // Added for testing + get AWTerminate() { + return AWTerminate; + } + + stop() { + lazy.log.debug(`Terminate reason is ${this.terminateReason}`); + Services.obs.removeObserver(this, "quit-application"); + if (!this.win) { + return; + } + this.win.removeEventListener("TabClose", this.onTabClose); + this.win.removeEventListener("unload", this.onWindowClose); + this.win = null; + } +} + +export class AboutWelcomeParent extends JSWindowActorParent { + constructor() { + super(); + this.startAboutWelcomeObserver(); + } + + startAboutWelcomeObserver() { + this.AboutWelcomeObserver = new AboutWelcomeObserver(); + } + + // Static methods that calls into ShellService to check + // if Firefox is pinned or already default + static doesAppNeedPin() { + return lazy.ShellService.doesAppNeedPin(); + } + + static isDefaultBrowser() { + return lazy.ShellService.isDefaultBrowser(); + } + + didDestroy() { + if (this.AboutWelcomeObserver) { + this.AboutWelcomeObserver.stop(); + } + this.RegionHomeObserver?.stop(); + + lazy.Telemetry.sendTelemetry({ + event: "SESSION_END", + event_context: { + reason: this.AboutWelcomeObserver.terminateReason, + page: "about:welcome", + }, + message_id: this.AWMessageId, + }); + } + + /** + * Handle messages from AboutWelcomeChild.jsm + * + * @param {string} type + * @param {any=} data + * @param {Browser} the xul:browser rendering the page + */ + async onContentMessage(type, data, browser) { + lazy.log.debug(`Received content event: ${type}`); + switch (type) { + case "AWPage:SET_WELCOME_MESSAGE_SEEN": + this.AWMessageId = data; + try { + Services.prefs.setBoolPref(DID_SEE_ABOUT_WELCOME_PREF, true); + } catch (e) { + lazy.log.debug(`Fails to set ${DID_SEE_ABOUT_WELCOME_PREF}.`); + } + break; + case "AWPage:SPECIAL_ACTION": + return lazy.SpecialMessageActions.handleAction(data, browser); + case "AWPage:FXA_METRICS_FLOW_URI": + return lazy.FxAccounts.config.promiseMetricsFlowURI("aboutwelcome"); + case "AWPage:TELEMETRY_EVENT": + lazy.Telemetry.sendTelemetry(data); + break; + case "AWPage:GET_ATTRIBUTION_DATA": + let attributionData = + await lazy.AboutWelcomeDefaults.getAttributionContent(); + return attributionData; + case "AWPage:ENSURE_ADDON_INSTALLED": + return new Promise(resolve => { + let listener = { + onInstallEnded(install, addon) { + if (addon.id === data) { + lazy.AddonManager.removeInstallListener(listener); + resolve("complete"); + } + }, + onInstallCancelled() { + lazy.AddonManager.removeInstallListener(listener); + resolve("install cancelled"); + }, + onInstallFailed() { + lazy.AddonManager.removeInstallListener(listener); + resolve("install failed"); + }, + }; + lazy.AddonManager.addInstallListener(listener); + }); + case "AWPage:GET_ADDON_DETAILS": + let addonDetails = + await lazy.AboutWelcomeDefaults.getAddonFromRepository(data); + + return { + label: addonDetails.name, + icon: addonDetails.iconURL, + type: addonDetails.type, + screenshots: addonDetails.screenshots, + url: addonDetails.url, + }; + case "AWPage:SELECT_THEME": + await lazy.BuiltInThemes.ensureBuiltInThemes(); + return lazy.AddonManager.getAddonByID(LIGHT_WEIGHT_THEMES[data]).then( + addon => addon.enable() + ); + case "AWPage:GET_SELECTED_THEME": + let themes = await lazy.AddonManager.getAddonsByTypes(["theme"]); + let activeTheme = themes.find(addon => addon.isActive); + // Store the current theme ID so user can restore their previous theme. + if (activeTheme?.id) { + LIGHT_WEIGHT_THEMES.AUTOMATIC = activeTheme.id; + } + // convert this to the short form name that the front end code + // expects + let themeShortName = Object.keys(LIGHT_WEIGHT_THEMES).find( + key => LIGHT_WEIGHT_THEMES[key] === activeTheme?.id + ); + return themeShortName?.toLowerCase(); + case "AWPage:DOES_APP_NEED_PIN": + return AboutWelcomeParent.doesAppNeedPin(); + case "AWPage:NEED_DEFAULT": + // Only need to set default if we're supposed to check and not default. + return ( + Services.prefs.getBoolPref("browser.shell.checkDefaultBrowser") && + !AboutWelcomeParent.isDefaultBrowser() + ); + case "AWPage:WAIT_FOR_MIGRATION_CLOSE": + // Support multiples types of migration: 1) content modal 2) old + // migration modal 3) standalone content modal + return new Promise(resolve => { + const topics = [ + "MigrationWizard:Closed", + "MigrationWizard:Destroyed", + ]; + const observer = () => { + topics.forEach(t => Services.obs.removeObserver(observer, t)); + resolve(); + }; + topics.forEach(t => Services.obs.addObserver(observer, t)); + }); + case "AWPage:GET_APP_AND_SYSTEM_LOCALE_INFO": + return lazy.LangPackMatcher.getAppAndSystemLocaleInfo(); + case "AWPage:EVALUATE_SCREEN_TARGETING": + return lazy.AWScreenUtils.evaluateTargetingAndRemoveScreens(data); + case "AWPage:ADD_SCREEN_IMPRESSION": + return lazy.AWScreenUtils.addScreenImpression(data); + case "AWPage:NEGOTIATE_LANGPACK": + return lazy.LangPackMatcher.negotiateLangPackForLanguageMismatch(data); + case "AWPage:ENSURE_LANG_PACK_INSTALLED": + return lazy.LangPackMatcher.ensureLangPackInstalled(data); + case "AWPage:SET_REQUESTED_LOCALES": + return lazy.LangPackMatcher.setRequestedAppLocales(data); + case "AWPage:SEND_TO_DEVICE_EMAILS_SUPPORTED": { + return lazy.BrowserUtils.sendToDeviceEmailsSupported(); + } + default: + lazy.log.debug(`Unexpected event ${type} was not handled.`); + } + + return undefined; + } + + /** + * @param {{name: string, data?: any}} message + * @override + */ + receiveMessage(message) { + const { name, data } = message; + let browser; + + if (this.manager.rootFrameLoader) { + browser = this.manager.rootFrameLoader.ownerElement; + return this.onContentMessage(name, data, browser); + } + + lazy.log.warn(`Not handling ${name} because the browser doesn't exist.`); + return null; + } +} + +export class AboutWelcomeShoppingParent extends AboutWelcomeParent { + /** + * Handle messages from AboutWelcomeChild.jsm + * + * @param {string} type + * @param {any=} data + * @param {Browser} the xul:browser rendering the page + */ + onContentMessage(type, data, browser) { + // Only handle the messages that are relevant to the shopping page. + switch (type) { + case "AWPage:SPECIAL_ACTION": + case "AWPage:TELEMETRY_EVENT": + case "AWPage:EVALUATE_SCREEN_TARGETING": + case "AWPage:ADD_SCREEN_IMPRESSION": + return super.onContentMessage(type, data, browser); + } + + return undefined; + } + + // Override unnecessary methods + startAboutWelcomeObserver() {} + + didDestroy() {} +} |