[fenix] Closes https://github.com/mozilla-mobile/fenix/issues/18522: Re-add call to action in tabs tray
parent
856e133f4d
commit
425c9d857b
@ -0,0 +1,124 @@
|
||||
/* 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.tabstray
|
||||
|
||||
import android.content.Context
|
||||
import android.view.View.VISIBLE
|
||||
import android.view.ViewGroup
|
||||
import androidx.annotation.VisibleForTesting
|
||||
import kotlin.math.max
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
||||
import kotlinx.coroutines.cancel
|
||||
import kotlinx.coroutines.flow.collect
|
||||
import kotlinx.coroutines.flow.map
|
||||
import mozilla.components.browser.state.selector.normalTabs
|
||||
import mozilla.components.browser.state.selector.privateTabs
|
||||
import mozilla.components.browser.state.store.BrowserStore
|
||||
import mozilla.components.lib.state.ext.flowScoped
|
||||
import mozilla.components.support.base.feature.LifecycleAwareFeature
|
||||
import mozilla.components.support.ktx.kotlinx.coroutines.flow.ifChanged
|
||||
import org.mozilla.fenix.R
|
||||
import org.mozilla.fenix.browser.infobanner.InfoBanner
|
||||
import org.mozilla.fenix.components.metrics.Event
|
||||
import org.mozilla.fenix.components.metrics.MetricController
|
||||
import org.mozilla.fenix.utils.Settings
|
||||
|
||||
class TabsTrayInfoBannerBinding(
|
||||
private val context: Context,
|
||||
private val store: BrowserStore,
|
||||
private val infoBannerView: ViewGroup,
|
||||
private val settings: Settings,
|
||||
private val navigationInteractor: NavigationInteractor,
|
||||
private val metrics: MetricController?
|
||||
) : LifecycleAwareFeature {
|
||||
private var scope: CoroutineScope? = null
|
||||
|
||||
@VisibleForTesting
|
||||
internal var banner: InfoBanner? = null
|
||||
|
||||
@ExperimentalCoroutinesApi
|
||||
override fun start() {
|
||||
scope = store.flowScoped { flow ->
|
||||
flow.map { state -> max(state.normalTabs.size, state.privateTabs.size) }
|
||||
.ifChanged()
|
||||
.collect { tabCount ->
|
||||
if (tabCount >= TAB_COUNT_SHOW_CFR) {
|
||||
displayInfoBannerIfNeeded(settings)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override fun stop() {
|
||||
scope?.cancel()
|
||||
}
|
||||
|
||||
private fun displayInfoBannerIfNeeded(settings: Settings) {
|
||||
banner = displayGridViewBannerIfNeeded(settings)
|
||||
?: displayAutoCloseTabsBannerIfNeeded(settings)
|
||||
|
||||
banner?.apply {
|
||||
infoBannerView.visibility = VISIBLE
|
||||
showBanner()
|
||||
}
|
||||
}
|
||||
|
||||
private fun displayGridViewBannerIfNeeded(settings: Settings): InfoBanner? {
|
||||
return if (
|
||||
settings.shouldShowGridViewBanner &&
|
||||
settings.canShowCfr &&
|
||||
settings.listTabView
|
||||
) {
|
||||
InfoBanner(
|
||||
context = context,
|
||||
message = context.getString(R.string.tab_tray_grid_view_banner_message),
|
||||
dismissText = context.getString(R.string.tab_tray_grid_view_banner_negative_button_text),
|
||||
actionText = context.getString(R.string.tab_tray_grid_view_banner_positive_button_text),
|
||||
container = infoBannerView,
|
||||
dismissByHiding = true,
|
||||
dismissAction = {
|
||||
metrics?.track(Event.TabsTrayCfrDismissed)
|
||||
settings.shouldShowGridViewBanner = false
|
||||
}
|
||||
) {
|
||||
navigationInteractor.onTabSettingsClicked()
|
||||
settings.shouldShowGridViewBanner = false
|
||||
}
|
||||
} else {
|
||||
null
|
||||
}
|
||||
}
|
||||
|
||||
private fun displayAutoCloseTabsBannerIfNeeded(settings: Settings): InfoBanner? {
|
||||
return if (
|
||||
settings.shouldShowAutoCloseTabsBanner &&
|
||||
settings.canShowCfr
|
||||
) {
|
||||
InfoBanner(
|
||||
context = context,
|
||||
message = context.getString(R.string.tab_tray_close_tabs_banner_message),
|
||||
dismissText = context.getString(R.string.tab_tray_close_tabs_banner_negative_button_text),
|
||||
actionText = context.getString(R.string.tab_tray_close_tabs_banner_positive_button_text),
|
||||
container = infoBannerView,
|
||||
dismissByHiding = true,
|
||||
dismissAction = {
|
||||
metrics?.track(Event.TabsTrayCfrDismissed)
|
||||
settings.shouldShowAutoCloseTabsBanner = false
|
||||
}
|
||||
) {
|
||||
navigationInteractor.onTabSettingsClicked()
|
||||
settings.shouldShowAutoCloseTabsBanner = false
|
||||
}
|
||||
} else {
|
||||
null
|
||||
}
|
||||
}
|
||||
|
||||
companion object {
|
||||
@VisibleForTesting
|
||||
internal const val TAB_COUNT_SHOW_CFR = 6
|
||||
}
|
||||
}
|
@ -0,0 +1,221 @@
|
||||
/* 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.tabstray
|
||||
|
||||
import android.view.View.GONE
|
||||
import android.view.View.VISIBLE
|
||||
import android.view.ViewGroup
|
||||
import androidx.coordinatorlayout.widget.CoordinatorLayout
|
||||
import io.mockk.mockk
|
||||
import io.mockk.verify
|
||||
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
||||
import kotlinx.coroutines.test.TestCoroutineDispatcher
|
||||
import mozilla.components.browser.state.action.TabListAction
|
||||
import mozilla.components.browser.state.state.createTab
|
||||
import mozilla.components.browser.state.store.BrowserStore
|
||||
import mozilla.components.support.test.libstate.ext.waitUntilIdle
|
||||
import mozilla.components.support.test.robolectric.testContext
|
||||
import mozilla.components.support.test.rule.MainCoroutineRule
|
||||
import org.junit.Before
|
||||
import org.junit.Rule
|
||||
import org.junit.Test
|
||||
import org.junit.runner.RunWith
|
||||
import org.mozilla.fenix.components.metrics.MetricController
|
||||
import org.mozilla.fenix.helpers.FenixRobolectricTestRunner
|
||||
import org.mozilla.fenix.tabstray.TabsTrayInfoBannerBinding.Companion.TAB_COUNT_SHOW_CFR
|
||||
import org.mozilla.fenix.utils.Settings
|
||||
|
||||
@RunWith(FenixRobolectricTestRunner::class)
|
||||
@OptIn(ExperimentalCoroutinesApi::class)
|
||||
class TabsTrayInfoBannerTest {
|
||||
|
||||
private lateinit var store: BrowserStore
|
||||
private lateinit var view: ViewGroup
|
||||
private lateinit var interactor: NavigationInteractor
|
||||
private lateinit var metrics: MetricController
|
||||
private lateinit var settings: Settings
|
||||
|
||||
@get:Rule
|
||||
val coroutinesTestRule = MainCoroutineRule(TestCoroutineDispatcher())
|
||||
|
||||
@Before
|
||||
fun setUp() {
|
||||
store = BrowserStore()
|
||||
view = CoordinatorLayout(testContext)
|
||||
interactor = mockk(relaxed = true)
|
||||
metrics = mockk(relaxed = true)
|
||||
settings = Settings(testContext)
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `WHEN tab number reaches CFR count THEN banner is shown`() {
|
||||
view.visibility = GONE
|
||||
|
||||
val binding =
|
||||
TabsTrayInfoBannerBinding(
|
||||
context = testContext,
|
||||
store = store,
|
||||
infoBannerView = view,
|
||||
settings = settings,
|
||||
navigationInteractor = interactor,
|
||||
metrics = metrics
|
||||
)
|
||||
|
||||
binding.start()
|
||||
for (i in 1 until TAB_COUNT_SHOW_CFR) {
|
||||
store.dispatch(TabListAction.AddTabAction(createTab("https://mozilla.org")))
|
||||
store.waitUntilIdle()
|
||||
|
||||
assert(view.visibility == GONE)
|
||||
}
|
||||
|
||||
store.dispatch(TabListAction.AddTabAction(createTab("https://mozilla.org")))
|
||||
store.waitUntilIdle()
|
||||
assert(view.visibility == VISIBLE)
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `WHEN in list view THEN grid view banner info banner should be shown`() {
|
||||
view.visibility = GONE
|
||||
settings.listTabView = true
|
||||
|
||||
val binding =
|
||||
TabsTrayInfoBannerBinding(
|
||||
context = testContext,
|
||||
store = store,
|
||||
infoBannerView = view,
|
||||
settings = settings,
|
||||
navigationInteractor = interactor,
|
||||
metrics = metrics
|
||||
)
|
||||
|
||||
binding.start()
|
||||
for (i in 1..TAB_COUNT_SHOW_CFR) {
|
||||
store.dispatch(TabListAction.AddTabAction(createTab("https://mozilla.org")))
|
||||
store.waitUntilIdle()
|
||||
}
|
||||
|
||||
assert(view.visibility == VISIBLE)
|
||||
binding.banner?.actionToPerform?.invoke()
|
||||
|
||||
verify { interactor.onTabSettingsClicked() }
|
||||
assert(!settings.shouldShowGridViewBanner)
|
||||
assert(settings.shouldShowAutoCloseTabsBanner)
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `WHEN grid view banner already shown THEN auto close tabs info banner should be shown`() {
|
||||
view.visibility = GONE
|
||||
settings.listTabView = true
|
||||
settings.shouldShowGridViewBanner = false
|
||||
|
||||
val binding =
|
||||
TabsTrayInfoBannerBinding(
|
||||
context = testContext,
|
||||
store = store,
|
||||
infoBannerView = view,
|
||||
settings = settings,
|
||||
navigationInteractor = interactor,
|
||||
metrics = metrics
|
||||
)
|
||||
|
||||
binding.start()
|
||||
for (i in 1..TAB_COUNT_SHOW_CFR) {
|
||||
store.dispatch(TabListAction.AddTabAction(createTab("https://mozilla.org")))
|
||||
store.waitUntilIdle()
|
||||
}
|
||||
|
||||
assert(view.visibility == VISIBLE)
|
||||
binding.banner?.actionToPerform?.invoke()
|
||||
|
||||
verify { interactor.onTabSettingsClicked() }
|
||||
assert(!settings.shouldShowGridViewBanner)
|
||||
assert(!settings.shouldShowAutoCloseTabsBanner)
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `WHEN dismiss THEN grid view info banner will not open tab settings`() {
|
||||
view.visibility = GONE
|
||||
settings.listTabView = true
|
||||
|
||||
val binding =
|
||||
TabsTrayInfoBannerBinding(
|
||||
context = testContext,
|
||||
store = store,
|
||||
infoBannerView = view,
|
||||
settings = settings,
|
||||
navigationInteractor = interactor,
|
||||
metrics = metrics
|
||||
)
|
||||
|
||||
binding.start()
|
||||
for (i in 1..TAB_COUNT_SHOW_CFR) {
|
||||
store.dispatch(TabListAction.AddTabAction(createTab("https://mozilla.org")))
|
||||
store.waitUntilIdle()
|
||||
}
|
||||
|
||||
assert(view.visibility == VISIBLE)
|
||||
binding.banner?.dismissAction?.invoke()
|
||||
|
||||
verify(exactly = 0) { interactor.onTabSettingsClicked() }
|
||||
assert(!settings.shouldShowGridViewBanner)
|
||||
assert(settings.shouldShowAutoCloseTabsBanner)
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `WHEN dismiss THEN auto close tabs info banner will not open tab settings`() {
|
||||
view.visibility = GONE
|
||||
settings.listTabView = false
|
||||
|
||||
val binding =
|
||||
TabsTrayInfoBannerBinding(
|
||||
context = testContext,
|
||||
store = store,
|
||||
infoBannerView = view,
|
||||
settings = settings,
|
||||
navigationInteractor = interactor,
|
||||
metrics = metrics
|
||||
)
|
||||
|
||||
binding.start()
|
||||
for (i in 1..TAB_COUNT_SHOW_CFR) {
|
||||
store.dispatch(TabListAction.AddTabAction(createTab("https://mozilla.org")))
|
||||
store.waitUntilIdle()
|
||||
}
|
||||
|
||||
assert(view.visibility == VISIBLE)
|
||||
binding.banner?.dismissAction?.invoke()
|
||||
|
||||
verify(exactly = 0) { interactor.onTabSettingsClicked() }
|
||||
assert(settings.shouldShowGridViewBanner)
|
||||
assert(!settings.shouldShowAutoCloseTabsBanner)
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `WHEN both banners were already shown THEN no info banner should be shown`() {
|
||||
view.visibility = GONE
|
||||
settings.listTabView = true
|
||||
settings.shouldShowGridViewBanner = false
|
||||
settings.shouldShowAutoCloseTabsBanner = false
|
||||
|
||||
val binding =
|
||||
TabsTrayInfoBannerBinding(
|
||||
context = testContext,
|
||||
store = store,
|
||||
infoBannerView = view,
|
||||
settings = settings,
|
||||
navigationInteractor = interactor,
|
||||
metrics = metrics
|
||||
)
|
||||
|
||||
binding.start()
|
||||
for (i in 1..TAB_COUNT_SHOW_CFR) {
|
||||
store.dispatch(TabListAction.AddTabAction(createTab("https://mozilla.org")))
|
||||
store.waitUntilIdle()
|
||||
}
|
||||
|
||||
assert(view.visibility == GONE)
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue