From e6b9e695cbc35b2afc47f8ab9eb0c792655eeaf6 Mon Sep 17 00:00:00 2001 From: Sawyer Blatz Date: Thu, 14 Mar 2019 14:06:14 -0700 Subject: [PATCH] Closes #832 & For #222: Adds basic custom tab integration --- .../mozilla/fenix/browser/BrowserFragment.kt | 19 ++ .../fenix/components/toolbar/ToolbarMenu.kt | 199 +++++++----------- .../fenix/customtabs/CustomTabsIntegration.kt | 162 ++++++++++++++ buildSrc/src/main/java/Dependencies.kt | 3 +- 4 files changed, 258 insertions(+), 125 deletions(-) create mode 100644 app/src/main/java/org/mozilla/fenix/customtabs/CustomTabsIntegration.kt diff --git a/app/src/main/java/org/mozilla/fenix/browser/BrowserFragment.kt b/app/src/main/java/org/mozilla/fenix/browser/BrowserFragment.kt index 9c099a0a4206..eb4a4073beaa 100644 --- a/app/src/main/java/org/mozilla/fenix/browser/BrowserFragment.kt +++ b/app/src/main/java/org/mozilla/fenix/browser/BrowserFragment.kt @@ -49,6 +49,7 @@ import org.mozilla.fenix.components.toolbar.ToolbarComponent import org.mozilla.fenix.components.toolbar.ToolbarIntegration import org.mozilla.fenix.components.toolbar.ToolbarMenu import org.mozilla.fenix.components.toolbar.ToolbarUIView +import org.mozilla.fenix.customtabs.CustomTabsIntegration import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.requireComponents import org.mozilla.fenix.ext.share @@ -68,6 +69,8 @@ class BrowserFragment : Fragment(), BackHandler, ComponentCallbacks2 { private val sitePermissionsFeature = ViewBoundFeatureWrapper() private val fullScreenFeature = ViewBoundFeatureWrapper() private val thumbnailsFeature = ViewBoundFeatureWrapper() + private val customTabsIntegration = ViewBoundFeatureWrapper() + var sessionId: String? = null override fun onCreateView( @@ -216,6 +219,22 @@ class BrowserFragment : Fragment(), BackHandler, ComponentCallbacks2 { owner = this, view = view ) + + val actionEmitter = ActionBusFactory.get(this).getManagedEmitter(SearchAction::class.java) + sessionId?.let { id -> + customTabsIntegration.set( + feature = CustomTabsIntegration( + requireContext(), + requireComponents.core.sessionManager, + toolbar, + id, + requireActivity(), + onItemTapped = { actionEmitter.onNext(SearchAction.ToolbarMenuItemTapped(it)) } + ), + owner = this, + view = view + ) + } } override fun onResume() { diff --git a/app/src/main/java/org/mozilla/fenix/components/toolbar/ToolbarMenu.kt b/app/src/main/java/org/mozilla/fenix/components/toolbar/ToolbarMenu.kt index 37b2e33d905b..eee267e8779b 100644 --- a/app/src/main/java/org/mozilla/fenix/components/toolbar/ToolbarMenu.kt +++ b/app/src/main/java/org/mozilla/fenix/components/toolbar/ToolbarMenu.kt @@ -10,8 +10,6 @@ import mozilla.components.browser.menu.item.BrowserMenuDivider import mozilla.components.browser.menu.item.BrowserMenuImageText import mozilla.components.browser.menu.item.BrowserMenuItemToolbar import mozilla.components.browser.menu.item.BrowserMenuSwitch -import mozilla.components.browser.menu.item.SimpleBrowserMenuItem -import mozilla.components.browser.session.runWithSession import org.mozilla.fenix.DefaultThemeManager import org.mozilla.fenix.R import org.mozilla.fenix.ext.components @@ -109,128 +107,83 @@ class ToolbarMenu( BrowserMenuItemToolbar(listOf(back, forward, refresh)) } - private val isCustomTab by lazy { - context.components.core.sessionManager.runWithSession(sessionId) { - it.isCustomTabSession() - } - } - private val menuItems by lazy { - if (isCustomTab) { - listOf( - SimpleBrowserMenuItem( - context.getString(R.string.browser_menu_powered_by), - CAPTION_TEXT_SIZE, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ), - BrowserMenuDivider(), - SimpleBrowserMenuItem( - context.getString(R.string.browser_menu_open_in_fenix), - textColorResource = DefaultThemeManager.resolveAttribute( - R.attr.browserToolbarMenuIcons, - context - ) - ) { - onItemTapped.invoke(Item.OpenInFenix) - }, - BrowserMenuImageText( - context.getString(R.string.browser_menu_find_in_page), - R.drawable.mozac_ic_search, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.FindInPage) - }, - BrowserMenuSwitch(context.getString(R.string.browser_menu_desktop_site), - requestDesktopStateProvider, { checked -> - onItemTapped.invoke(Item.RequestDesktop(checked)) - }), - BrowserMenuImageText( - context.getString(R.string.browser_menu_share), - R.drawable.mozac_ic_share, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.Share) - }, - menuToolbar - ) - } else { - listOf( - BrowserMenuImageText( - context.getString(R.string.browser_menu_help), - R.drawable.ic_help, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.Help) - }, - - BrowserMenuImageText( - context.getString(R.string.browser_menu_settings), - R.drawable.ic_settings, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.Settings) - }, - - BrowserMenuImageText( - context.getString(R.string.browser_menu_library), - R.drawable.ic_library, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.Library) - }, - - BrowserMenuDivider(), - - BrowserMenuSwitch(context.getString(R.string.browser_menu_desktop_site), - requestDesktopStateProvider, { checked -> - onItemTapped.invoke(Item.RequestDesktop(checked)) - }), - - BrowserMenuImageText( - context.getString(R.string.browser_menu_find_in_page), - R.drawable.mozac_ic_search, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.FindInPage) - }, - - BrowserMenuImageText( - context.getString(R.string.browser_menu_private_tab), - R.drawable.ic_private_browsing, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.NewPrivateTab) - }, - - BrowserMenuImageText( - context.getString(R.string.browser_menu_new_tab), - R.drawable.ic_new, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.NewTab) - }, - - BrowserMenuImageText( - context.getString(R.string.browser_menu_share), - R.drawable.mozac_ic_share, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.Share) - }, - - BrowserMenuImageText( - context.getString(R.string.browser_menu_report_issue), - R.drawable.ic_report_issues, - DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) - ) { - onItemTapped.invoke(Item.ReportIssue) - }, - - BrowserMenuDivider(), - - menuToolbar - ) - } + listOf( + BrowserMenuImageText( + context.getString(R.string.browser_menu_help), + R.drawable.ic_help, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.Help) + }, + + BrowserMenuImageText( + context.getString(R.string.browser_menu_settings), + R.drawable.ic_settings, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.Settings) + }, + + BrowserMenuImageText( + context.getString(R.string.browser_menu_library), + R.drawable.ic_library, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.Library) + }, + + BrowserMenuDivider(), + + BrowserMenuSwitch(context.getString(R.string.browser_menu_desktop_site), + requestDesktopStateProvider, { checked -> + onItemTapped.invoke(Item.RequestDesktop(checked)) + }), + + BrowserMenuImageText( + context.getString(R.string.browser_menu_find_in_page), + R.drawable.mozac_ic_search, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.FindInPage) + }, + + BrowserMenuImageText( + context.getString(R.string.browser_menu_private_tab), + R.drawable.ic_private_browsing, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.NewPrivateTab) + }, + + BrowserMenuImageText( + context.getString(R.string.browser_menu_new_tab), + R.drawable.ic_new, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.NewTab) + }, + + BrowserMenuImageText( + context.getString(R.string.browser_menu_share), + R.drawable.mozac_ic_share, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.Share) + }, + + BrowserMenuImageText( + context.getString(R.string.browser_menu_report_issue), + R.drawable.ic_report_issues, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(Item.ReportIssue) + }, + + BrowserMenuDivider(), + + menuToolbar + ) } companion object { diff --git a/app/src/main/java/org/mozilla/fenix/customtabs/CustomTabsIntegration.kt b/app/src/main/java/org/mozilla/fenix/customtabs/CustomTabsIntegration.kt new file mode 100644 index 000000000000..de96cc1b2a4f --- /dev/null +++ b/app/src/main/java/org/mozilla/fenix/customtabs/CustomTabsIntegration.kt @@ -0,0 +1,162 @@ +/* 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.customtabs + +import android.app.Activity +import android.content.Context +import mozilla.components.browser.menu.BrowserMenuBuilder +import mozilla.components.browser.menu.item.BrowserMenuDivider +import mozilla.components.browser.menu.item.BrowserMenuImageText +import mozilla.components.browser.menu.item.BrowserMenuItemToolbar +import mozilla.components.browser.menu.item.BrowserMenuSwitch +import mozilla.components.browser.menu.item.SimpleBrowserMenuItem +import mozilla.components.browser.session.SessionManager +import mozilla.components.browser.toolbar.BrowserToolbar +import mozilla.components.feature.customtabs.CustomTabsToolbarFeature +import mozilla.components.support.base.feature.BackHandler +import mozilla.components.support.base.feature.LifecycleAwareFeature +import org.mozilla.fenix.DefaultThemeManager +import org.mozilla.fenix.R +import org.mozilla.fenix.components.toolbar.ToolbarMenu +import org.mozilla.fenix.ext.components + +class CustomTabsIntegration( + context: Context, + sessionManager: SessionManager, + toolbar: BrowserToolbar, + sessionId: String, + activity: Activity?, + onItemTapped: (ToolbarMenu.Item) -> Unit = {} + +) : LifecycleAwareFeature, BackHandler { + private val session = sessionManager.findSessionById(sessionId) + + val menuBuilder by lazy { BrowserMenuBuilder(menuItems) } + + val menuToolbar by lazy { + val back = BrowserMenuItemToolbar.TwoStateButton( + primaryImageResource = mozilla.components.ui.icons.R.drawable.mozac_ic_back, + primaryContentDescription = context.getString(R.string.browser_menu_back), + primaryImageTintResource = DefaultThemeManager.resolveAttribute( + R.attr.browserToolbarMenuIcons, + context + ), + isInPrimaryState = { + context.components.core.sessionManager.selectedSession?.canGoBack ?: true + }, + secondaryImageTintResource = DefaultThemeManager.resolveAttribute( + R.attr.disabledIconColor, + context + ), + disableInSecondaryState = true + ) { + onItemTapped.invoke(ToolbarMenu.Item.Back) + } + + val forward = BrowserMenuItemToolbar.TwoStateButton( + primaryImageResource = mozilla.components.ui.icons.R.drawable.mozac_ic_forward, + primaryContentDescription = context.getString(R.string.browser_menu_forward), + primaryImageTintResource = DefaultThemeManager.resolveAttribute( + R.attr.browserToolbarMenuIcons, + context + ), + isInPrimaryState = { + context.components.core.sessionManager.selectedSession?.canGoForward ?: true + }, + secondaryImageTintResource = DefaultThemeManager.resolveAttribute( + R.attr.disabledIconColor, + context + ), + disableInSecondaryState = true + ) { + onItemTapped.invoke(ToolbarMenu.Item.Forward) + } + + val refresh = BrowserMenuItemToolbar.TwoStateButton( + primaryImageResource = mozilla.components.ui.icons.R.drawable.mozac_ic_refresh, + primaryContentDescription = context.getString(R.string.browser_menu_refresh), + primaryImageTintResource = DefaultThemeManager.resolveAttribute( + R.attr.browserToolbarMenuIcons, + context + ), + isInPrimaryState = { + val loading = context.components.core.sessionManager.selectedSession?.loading + loading == false + }, + secondaryImageResource = mozilla.components.ui.icons.R.drawable.mozac_ic_stop, + secondaryContentDescription = context.getString(R.string.browser_menu_stop), + secondaryImageTintResource = DefaultThemeManager.resolveAttribute( + R.attr.browserToolbarMenuIcons, + context + ), + disableInSecondaryState = false + ) { + if (context.components.core.sessionManager.selectedSession?.loading == true) { + onItemTapped.invoke(ToolbarMenu.Item.Stop) + } else { + onItemTapped.invoke(ToolbarMenu.Item.Reload) + } + } + + BrowserMenuItemToolbar(listOf(back, forward, refresh)) + } + + private val menuItems by lazy { + listOf( + SimpleBrowserMenuItem( + context.getString(R.string.browser_menu_powered_by), + ToolbarMenu.CAPTION_TEXT_SIZE, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ), + BrowserMenuDivider(), + SimpleBrowserMenuItem( + context.getString(R.string.browser_menu_open_in_fenix), + textColorResource = DefaultThemeManager.resolveAttribute( + R.attr.browserToolbarMenuIcons, + context + ) + ) { + onItemTapped.invoke(ToolbarMenu.Item.OpenInFenix) + }, + BrowserMenuImageText( + context.getString(R.string.browser_menu_find_in_page), + R.drawable.mozac_ic_search, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(ToolbarMenu.Item.FindInPage) + }, + BrowserMenuSwitch(context.getString(R.string.browser_menu_desktop_site), + { session?.desktopMode ?: false }, { checked -> + onItemTapped.invoke(ToolbarMenu.Item.RequestDesktop(checked)) + }), + BrowserMenuImageText( + context.getString(R.string.browser_menu_share), + R.drawable.mozac_ic_share, + DefaultThemeManager.resolveAttribute(R.attr.browserToolbarMenuIcons, context) + ) { + onItemTapped.invoke(ToolbarMenu.Item.Share) + }, + menuToolbar + ) + } + private val feature = CustomTabsToolbarFeature( + sessionManager, + toolbar, + sessionId, + menuBuilder, + closeListener = { activity?.finish() }) + + override fun start() { + feature.start() + } + + override fun stop() { + feature.stop() + } + + override fun onBackPressed(): Boolean { + return feature.onBackPressed() + } +} diff --git a/buildSrc/src/main/java/Dependencies.kt b/buildSrc/src/main/java/Dependencies.kt index f7c4c2425ca7..2c0a6c5762d8 100644 --- a/buildSrc/src/main/java/Dependencies.kt +++ b/buildSrc/src/main/java/Dependencies.kt @@ -46,7 +46,7 @@ object Deps { const val tools_appservicesgradle = "org.mozilla.appservices:gradle-plugin:${Versions.appservices_gradle_plugin}" const val kotlin_stdlib = "org.jetbrains.kotlin:kotlin-stdlib-jdk7:${Versions.kotlin}" - const val allopen = "org.jetbrains.kotlin:kotlin-allopen:${Versions.kotlin}" + const val allopen = "org.jetbrains.kotlin:kotlin-allopen:${Versions.kotlin}" const val rxKotlin = "io.reactivex.rxjava2:rxkotlin:${Versions.rxKotlin}" const val rxAndroid = "io.reactivex.rxjava2:rxandroid:${Versions.rxAndroid}" @@ -147,4 +147,3 @@ object Deps { const val glide = "com.github.bumptech.glide:glide:${Versions.glide}" const val glideAnnotationProcessor = "com.github.bumptech.glide:compiler:${Versions.glide}" } -