mirror of
https://github.com/fork-maintainers/iceraven-browser
synced 2024-11-11 13:11:01 +00:00
238 lines
9.1 KiB
Kotlin
238 lines
9.1 KiB
Kotlin
/* 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
|
|
|
|
import android.content.Context
|
|
import android.content.Intent
|
|
import android.os.Build
|
|
import android.os.Bundle
|
|
import android.preference.PreferenceManager
|
|
import android.util.AttributeSet
|
|
import android.view.View
|
|
import androidx.appcompat.app.AppCompatActivity
|
|
import androidx.appcompat.widget.Toolbar
|
|
import androidx.navigation.fragment.NavHostFragment
|
|
import androidx.navigation.ui.AppBarConfiguration
|
|
import androidx.navigation.ui.NavigationUI
|
|
import mozilla.components.browser.search.SearchEngine
|
|
import mozilla.components.browser.session.Session
|
|
import mozilla.components.browser.session.SessionManager
|
|
import mozilla.components.concept.engine.EngineView
|
|
import mozilla.components.feature.intent.IntentProcessor
|
|
import mozilla.components.lib.crash.Crash
|
|
import mozilla.components.support.base.feature.BackHandler
|
|
import mozilla.components.support.ktx.kotlin.isUrl
|
|
import mozilla.components.support.ktx.kotlin.toNormalizedUrl
|
|
import mozilla.components.support.utils.SafeIntent
|
|
import org.mozilla.fenix.components.metrics.Event
|
|
import org.mozilla.fenix.ext.components
|
|
import org.mozilla.fenix.home.HomeFragmentDirections
|
|
import org.mozilla.fenix.library.bookmarks.BookmarkFragmentDirections
|
|
import org.mozilla.fenix.library.bookmarks.selectfolder.SelectBookmarkFolderFragmentDirections
|
|
import org.mozilla.fenix.library.history.HistoryFragmentDirections
|
|
import org.mozilla.fenix.search.SearchFragmentDirections
|
|
import org.mozilla.fenix.settings.SettingsFragmentDirections
|
|
|
|
@SuppressWarnings("TooManyFunctions")
|
|
open class HomeActivity : AppCompatActivity() {
|
|
open val isCustomTab = false
|
|
private var sessionObserver: SessionManager.Observer? = null
|
|
var allSessionsRemoved = false
|
|
|
|
val themeManager = DefaultThemeManager().also {
|
|
it.onThemeChange = { theme ->
|
|
setTheme(theme)
|
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P) {
|
|
// Older versions of android do not support window animation style, so we use a transition instead
|
|
finish()
|
|
overridePendingTransition(android.R.anim.fade_in, android.R.anim.fade_out)
|
|
startActivity(intent)
|
|
} else {
|
|
recreate()
|
|
}
|
|
}
|
|
}
|
|
|
|
private val navHost by lazy {
|
|
supportFragmentManager.findFragmentById(R.id.container) as NavHostFragment
|
|
}
|
|
|
|
lateinit var browsingModeManager: DefaultBrowsingModeManager
|
|
|
|
override fun onCreate(savedInstanceState: Bundle?) {
|
|
super.onCreate(savedInstanceState)
|
|
sessionObserver = subscribeToSessions()
|
|
|
|
themeManager.temporaryThemeManagerStorage =
|
|
when (PreferenceManager.getDefaultSharedPreferences(this)
|
|
.getBoolean(this.getString(R.string.pref_key_private_mode), false)) {
|
|
true -> ThemeManager.Theme.Private
|
|
false -> ThemeManager.Theme.Normal
|
|
}
|
|
setTheme(themeManager.currentTheme)
|
|
DefaultThemeManager.applyStatusBarTheme(window, themeManager, this)
|
|
browsingModeManager = DefaultBrowsingModeManager(this)
|
|
|
|
components.core.setEnginePreferredColorScheme()
|
|
setContentView(R.layout.activity_home)
|
|
|
|
val appBarConfiguration = AppBarConfiguration.Builder(setOf(R.id.libraryFragment)).build()
|
|
val navigationToolbar = findViewById<Toolbar>(R.id.navigationToolbar)
|
|
setSupportActionBar(navigationToolbar)
|
|
NavigationUI.setupWithNavController(navigationToolbar, navHost.navController, appBarConfiguration)
|
|
|
|
intent
|
|
?.let { SafeIntent(it) }
|
|
?.let {
|
|
when {
|
|
isCustomTab -> Event.OpenedApp.Source.CUSTOM_TAB
|
|
it.isLauncherIntent -> Event.OpenedApp.Source.APP_ICON
|
|
it.action == Intent.ACTION_VIEW -> Event.OpenedApp.Source.LINK
|
|
else -> null
|
|
}
|
|
}
|
|
?.also { components.analytics.metrics.track(Event.OpenedApp(it)) }
|
|
|
|
handleOpenedFromExternalSourceIfNecessary(intent)
|
|
}
|
|
|
|
override fun onResume() {
|
|
super.onResume()
|
|
// All sessions have been removed; we should try to pop inclusive to browser if not in private mode
|
|
if (allSessionsRemoved && !browsingModeManager.isPrivate) {
|
|
navHost.navController.popBackStack(R.id.browserFragment, true)
|
|
allSessionsRemoved = false
|
|
}
|
|
}
|
|
|
|
override fun onDestroy() {
|
|
super.onDestroy()
|
|
sessionObserver?.let {
|
|
components.core.sessionManager.unregister(it)
|
|
}
|
|
}
|
|
|
|
override fun onNewIntent(intent: Intent?) {
|
|
super.onNewIntent(intent)
|
|
handleCrashIfNecessary(intent)
|
|
handleOpenedFromExternalSourceIfNecessary(intent)
|
|
}
|
|
|
|
override fun onCreateView(
|
|
parent: View?,
|
|
name: String,
|
|
context: Context,
|
|
attrs: AttributeSet
|
|
): View? =
|
|
when (name) {
|
|
EngineView::class.java.name -> components.core.engine.createView(context, attrs).asView()
|
|
else -> super.onCreateView(parent, name, context, attrs)
|
|
}
|
|
|
|
override fun onBackPressed() {
|
|
supportFragmentManager.primaryNavigationFragment?.childFragmentManager?.fragments?.forEach {
|
|
if (it is BackHandler && it.onBackPressed()) {
|
|
return
|
|
}
|
|
}
|
|
super.onBackPressed()
|
|
}
|
|
|
|
private fun handleCrashIfNecessary(intent: Intent?) {
|
|
if (intent == null) { return }
|
|
if (!Crash.isCrashIntent(intent)) { return }
|
|
|
|
openToCrashReporter(intent)
|
|
}
|
|
|
|
private fun openToCrashReporter(intent: Intent) {
|
|
val directions = NavGraphDirections.actionGlobalCrashReporter(intent)
|
|
navHost.navController.navigate(directions)
|
|
}
|
|
|
|
private fun handleOpenedFromExternalSourceIfNecessary(intent: Intent?) {
|
|
if (intent?.extras?.getBoolean(OPEN_TO_BROWSER) == true) {
|
|
handleOpenedFromExternalSource()
|
|
}
|
|
}
|
|
|
|
private fun handleOpenedFromExternalSource() {
|
|
intent?.putExtra(OPEN_TO_BROWSER, false)
|
|
openToBrowser(SafeIntent(intent).getStringExtra(IntentProcessor.ACTIVE_SESSION_ID), BrowserDirection.FromGlobal)
|
|
}
|
|
|
|
fun openToBrowserAndLoad(
|
|
text: String,
|
|
sessionId: String? = null,
|
|
engine: SearchEngine? = null,
|
|
from: BrowserDirection
|
|
) {
|
|
openToBrowser(sessionId, from)
|
|
load(text, sessionId, engine)
|
|
}
|
|
|
|
fun openToBrowser(sessionId: String?, from: BrowserDirection) {
|
|
val directions = when (from) {
|
|
BrowserDirection.FromGlobal -> NavGraphDirections.actionGlobalBrowser(sessionId)
|
|
BrowserDirection.FromHome -> HomeFragmentDirections.actionHomeFragmentToBrowserFragment(sessionId)
|
|
BrowserDirection.FromSearch -> SearchFragmentDirections.actionSearchFragmentToBrowserFragment(sessionId)
|
|
BrowserDirection.FromSettings ->
|
|
SettingsFragmentDirections.actionSettingsFragmentToBrowserFragment(sessionId)
|
|
BrowserDirection.FromBookmarks ->
|
|
BookmarkFragmentDirections.actionBookmarkFragmentToBrowserFragment(sessionId)
|
|
BrowserDirection.FromBookmarksFolderSelect ->
|
|
SelectBookmarkFolderFragmentDirections.actionBookmarkSelectFolderFragmentToBrowserFragment(sessionId)
|
|
BrowserDirection.FromHistory ->
|
|
HistoryFragmentDirections.actionHistoryFragmentToBrowserFragment(sessionId)
|
|
}
|
|
|
|
navHost.navController.navigate(directions)
|
|
}
|
|
|
|
private fun load(text: String, sessionId: String?, engine: SearchEngine?) {
|
|
val isPrivate = this.browsingModeManager.isPrivate
|
|
|
|
val loadUrlUseCase = if (sessionId == null) {
|
|
if (isPrivate) {
|
|
components.useCases.tabsUseCases.addPrivateTab
|
|
} else {
|
|
components.useCases.tabsUseCases.addTab
|
|
}
|
|
} else components.useCases.sessionUseCases.loadUrl
|
|
|
|
val searchUseCase: (String) -> Unit = { searchTerms ->
|
|
if (sessionId == null) {
|
|
components.useCases.searchUseCases.newTabSearch
|
|
.invoke(searchTerms, Session.Source.USER_ENTERED, true, isPrivate, searchEngine = engine)
|
|
} else components.useCases.searchUseCases.defaultSearch.invoke(searchTerms, engine)
|
|
}
|
|
|
|
if (text.isUrl()) {
|
|
loadUrlUseCase.invoke(text.toNormalizedUrl())
|
|
} else {
|
|
searchUseCase.invoke(text)
|
|
}
|
|
}
|
|
|
|
private fun subscribeToSessions(): SessionManager.Observer {
|
|
val observer = object : SessionManager.Observer {
|
|
override fun onAllSessionsRemoved() {
|
|
super.onAllSessionsRemoved()
|
|
allSessionsRemoved = true
|
|
}
|
|
}
|
|
components.core.sessionManager.register(observer)
|
|
return observer
|
|
}
|
|
|
|
companion object {
|
|
const val OPEN_TO_BROWSER = "open_to_browser"
|
|
}
|
|
}
|
|
|
|
enum class BrowserDirection {
|
|
FromGlobal, FromHome, FromSearch, FromSettings, FromBookmarks, FromBookmarksFolderSelect, FromHistory
|
|
}
|