/* 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/. */ package org.mozilla.fenix.components import android.app.Application import android.content.Context import android.content.Intent import android.os.StrictMode import androidx.compose.runtime.Composable import androidx.compose.ui.platform.LocalContext import androidx.core.net.toUri import com.google.android.play.core.review.ReviewManagerFactory import mozilla.components.feature.addons.AddonManager import mozilla.components.feature.addons.migration.DefaultSupportedAddonsChecker import mozilla.components.feature.addons.update.DefaultAddonUpdater import mozilla.components.feature.autofill.AutofillConfiguration import mozilla.components.lib.publicsuffixlist.PublicSuffixList import mozilla.components.support.base.worker.Frequency import mozilla.components.support.locale.LocaleManager import io.github.forkmaintainers.iceraven.components.PagedAddonCollectionProvider import org.mozilla.fenix.BuildConfig import org.mozilla.fenix.Config import org.mozilla.fenix.HomeActivity import org.mozilla.fenix.R import org.mozilla.fenix.autofill.AutofillConfirmActivity import org.mozilla.fenix.autofill.AutofillSearchActivity import org.mozilla.fenix.autofill.AutofillUnlockActivity import org.mozilla.fenix.components.appstate.AppState import org.mozilla.fenix.datastore.pocketStoriesSelectedCategoriesDataStore import org.mozilla.fenix.ext.asRecentTabs import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.filterState import org.mozilla.fenix.ext.settings import org.mozilla.fenix.gleanplumb.state.MessagingMiddleware import org.mozilla.fenix.ext.sort import org.mozilla.fenix.home.PocketUpdatesMiddleware import org.mozilla.fenix.home.blocklist.BlocklistHandler import org.mozilla.fenix.home.blocklist.BlocklistMiddleware import org.mozilla.fenix.perf.AppStartReasonProvider import org.mozilla.fenix.perf.StartupActivityLog import org.mozilla.fenix.perf.StartupStateProvider import org.mozilla.fenix.perf.StrictModeManager import org.mozilla.fenix.perf.lazyMonitored import org.mozilla.fenix.utils.ClipboardHandler import org.mozilla.fenix.utils.Settings import org.mozilla.fenix.wallpapers.WallpaperDownloader import org.mozilla.fenix.wallpapers.WallpaperFileManager import org.mozilla.fenix.wallpapers.WallpaperManager import org.mozilla.fenix.wifi.WifiConnectionMonitor import java.util.concurrent.TimeUnit private const val AMO_COLLECTION_MAX_CACHE_AGE = 2 * 24 * 60L // Two days in minutes /** * Provides access to all components. This class is an implementation of the Service Locator * pattern, which helps us manage the dependencies in our app. * * Note: these aren't just "components" from "android-components": they're any "component" that * can be considered a building block of our app. */ class Components(private val context: Context) { val backgroundServices by lazyMonitored { BackgroundServices( context, push, analytics.crashReporter, core.lazyHistoryStorage, core.lazyBookmarksStorage, core.lazyPasswordsStorage, core.lazyRemoteTabsStorage, core.lazyAutofillStorage, strictMode ) } val services by lazyMonitored { Services(context, backgroundServices.accountManager) } val core by lazyMonitored { Core(context, analytics.crashReporter, strictMode) } @Suppress("Deprecation") val useCases by lazyMonitored { UseCases( context, core.engine, core.store, core.webAppShortcutManager, core.topSitesStorage, core.bookmarksStorage, core.historyStorage ) } val intentProcessors by lazyMonitored { IntentProcessors( context, core.store, useCases.sessionUseCases, useCases.tabsUseCases, useCases.customTabsUseCases, useCases.searchUseCases, core.webAppManifestStorage, core.engine, ) } val addonCollectionProvider by lazyMonitored { // Check if we have a customized (overridden) AMO collection (only supported in Nightly) if (Config.channel.isNightlyOrDebug && context.settings().amoCollectionOverrideConfigured()) { PagedAddonCollectionProvider( context, core.client, collectionAccount = context.settings().overrideAmoUser, collectionName = context.settings().overrideAmoCollection ) } // Use Iceraven settings config otherwise else { PagedAddonCollectionProvider( context, core.client, serverURL = BuildConfig.AMO_SERVER_URL, collectionAccount = context.settings().customAddonsAccount, collectionName = context.settings().customAddonsCollection, maxCacheAgeInMinutes = AMO_COLLECTION_MAX_CACHE_AGE ) } } @Suppress("MagicNumber") val addonUpdater by lazyMonitored { DefaultAddonUpdater(context, Frequency(12, TimeUnit.HOURS)) } @Suppress("MagicNumber") val supportedAddonsChecker by lazyMonitored { DefaultSupportedAddonsChecker( context, Frequency(12, TimeUnit.HOURS), onNotificationClickIntent = Intent(context, HomeActivity::class.java).apply { action = Intent.ACTION_VIEW flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK data = "${BuildConfig.DEEP_LINK_SCHEME}://settings_addon_manager".toUri() } ) } val addonManager by lazyMonitored { AddonManager(core.store, core.engine, addonCollectionProvider, addonUpdater) } fun updateAddonManager() { addonCollectionProvider.deleteCacheFile(context) val addonsAccount = context.settings().customAddonsAccount val addonsCollection = context.settings().customAddonsCollection addonCollectionProvider.setCollectionAccount(addonsAccount) addonCollectionProvider.setCollectionName(addonsCollection) } val analytics by lazyMonitored { Analytics(context) } val publicSuffixList by lazyMonitored { PublicSuffixList(context) } val clipboardHandler by lazyMonitored { ClipboardHandler(context) } val performance by lazyMonitored { PerformanceComponent() } val push by lazyMonitored { Push(context, analytics.crashReporter) } val wifiConnectionMonitor by lazyMonitored { WifiConnectionMonitor(context as Application) } val strictMode by lazyMonitored { StrictModeManager(Config, this) } val wallpaperManager by lazyMonitored { val currentLocale = strictMode.resetAfter(StrictMode.allowThreadDiskReads()) { LocaleManager.getCurrentLocale(context)?.toLanguageTag() ?: LocaleManager.getSystemDefault().toLanguageTag() } strictMode.resetAfter(StrictMode.allowThreadDiskReads()) { WallpaperManager( settings, WallpaperDownloader(context, core.client), WallpaperFileManager(context.filesDir), currentLocale ) } } val settings by lazyMonitored { Settings(context) } val reviewPromptController by lazyMonitored { ReviewPromptController( manager = ReviewManagerFactory.create(context), reviewSettings = FenixReviewSettings(settings) ) } val autofillConfiguration by lazyMonitored { AutofillConfiguration( storage = core.passwordsStorage, publicSuffixList = publicSuffixList, unlockActivity = AutofillUnlockActivity::class.java, confirmActivity = AutofillConfirmActivity::class.java, searchActivity = AutofillSearchActivity::class.java, applicationName = context.getString(R.string.app_name), httpClient = core.client ) } val appStartReasonProvider by lazyMonitored { AppStartReasonProvider() } val startupActivityLog by lazyMonitored { StartupActivityLog() } val startupStateProvider by lazyMonitored { StartupStateProvider(startupActivityLog, appStartReasonProvider) } val appStore by lazyMonitored { val blocklistHandler = BlocklistHandler(settings) AppStore( initialState = AppState( collections = core.tabCollectionStorage.cachedTabCollections, expandedCollections = emptySet(), topSites = core.topSitesStorage.cachedTopSites.sort(), recentBookmarks = emptyList(), showCollectionPlaceholder = settings.showCollectionsPlaceholderOnHome, // Provide an initial state for recent tabs to prevent re-rendering on the home screen. // This will otherwise cause a visual jump as the section gets rendered from no state // to some state. recentTabs = if (settings.showRecentTabsFeature) { core.store.state.asRecentTabs() } else { emptyList() }, recentHistory = emptyList() ).run { filterState(blocklistHandler) }, middlewares = listOf( BlocklistMiddleware(blocklistHandler), PocketUpdatesMiddleware( core.pocketStoriesService, context.pocketStoriesSelectedCategoriesDataStore ), MessagingMiddleware(messagingStorage = analytics.messagingStorage) ) ) } } /** * Returns the [Components] object from within a [Composable]. */ val components: Components @Composable get() = LocalContext.current.components