/* 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.home.sessioncontrol import android.os.Build import android.view.View import androidx.lifecycle.ProcessLifecycleOwner import androidx.recyclerview.widget.ItemTouchHelper import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.RecyclerView import kotlinx.android.extensions.LayoutContainer import kotlinx.coroutines.ExperimentalCoroutinesApi import mozilla.components.feature.tab.collections.TabCollection import mozilla.components.feature.top.sites.TopSite import mozilla.components.lib.state.ext.consumeFrom import org.mozilla.fenix.R import org.mozilla.fenix.home.HomeFragmentState import org.mozilla.fenix.home.HomeFragmentStore import org.mozilla.fenix.home.Mode import org.mozilla.fenix.home.OnboardingState import org.mozilla.fenix.home.Tab val noTabMessage = AdapterItem.NoContentMessage( R.drawable.ic_tabs, R.string.no_open_tabs_header_2, R.string.no_open_tabs_description ) val noCollectionMessage = AdapterItem.NoContentMessage( R.drawable.ic_tab_collection, R.string.no_collections_header, R.string.collections_description ) private fun normalModeAdapterItems( tabs: List, topSites: List, collections: List, expandedCollections: Set ): List { val items = mutableListOf() if (topSites.isNotEmpty()) { items.add(AdapterItem.TopSiteList(topSites)) } items.add(AdapterItem.TabHeader(false, tabs.isNotEmpty())) if (tabs.isNotEmpty()) { items.addAll(tabs.reversed().map(AdapterItem::TabItem)) items.add(AdapterItem.SaveTabGroup) } else { items.add(noTabMessage) } items.add(AdapterItem.CollectionHeader) if (collections.isNotEmpty()) { // If the collection is expanded, we want to add all of its tabs beneath it in the adapter collections.map { AdapterItem.CollectionItem(it, expandedCollections.contains(it.id), tabs.isNotEmpty()) }.forEach { items.add(it) if (it.expanded) { items.addAll(collectionTabItems(it.collection)) } } } else { items.add(noCollectionMessage) } return items } private fun privateModeAdapterItems(tabs: List): List { val items = mutableListOf() items.add(AdapterItem.TabHeader(true, tabs.isNotEmpty())) if (tabs.isNotEmpty()) { items.addAll(tabs.reversed().map(AdapterItem::TabItem)) } else { items.add(AdapterItem.PrivateBrowsingDescription) } return items } private fun onboardingAdapterItems(onboardingState: OnboardingState): List { val items: MutableList = mutableListOf(AdapterItem.OnboardingHeader) // Customize FxA items based on where we are with the account state: items.addAll(when (onboardingState) { OnboardingState.SignedOutNoAutoSignIn -> { listOf( AdapterItem.OnboardingManualSignIn ) } is OnboardingState.SignedOutCanAutoSignIn -> { listOf( AdapterItem.OnboardingAutomaticSignIn(onboardingState) ) } OnboardingState.SignedIn -> listOf() }) items.addAll(listOf( AdapterItem.OnboardingSectionHeader { val appName = it.getString(R.string.app_name) it.getString(R.string.onboarding_feature_section_header, appName) }, AdapterItem.OnboardingWhatsNew, AdapterItem.OnboardingTrackingProtection, AdapterItem.OnboardingThemePicker, AdapterItem.OnboardingPrivateBrowsing, AdapterItem.OnboardingToolbarPositionPicker, AdapterItem.OnboardingPrivacyNotice, AdapterItem.OnboardingFinish )) return items } private fun HomeFragmentState.toAdapterList(): List = when (mode) { is Mode.Normal -> normalModeAdapterItems(tabs, topSites, collections, expandedCollections) is Mode.Private -> privateModeAdapterItems(tabs) is Mode.Onboarding -> onboardingAdapterItems(mode.state) } private fun collectionTabItems(collection: TabCollection) = collection.tabs.mapIndexed { index, tab -> AdapterItem.TabInCollectionItem(collection, tab, index == collection.tabs.lastIndex) } @ExperimentalCoroutinesApi class SessionControlView( private val homeFragmentStore: HomeFragmentStore, override val containerView: View?, interactor: SessionControlInteractor ) : LayoutContainer { val view: RecyclerView = containerView as RecyclerView private val sessionControlAdapter = SessionControlAdapter(interactor) init { view.apply { adapter = sessionControlAdapter layoutManager = LinearLayoutManager(containerView!!.context) val itemTouchHelper = ItemTouchHelper( SwipeToDeleteCallback( interactor ) ) itemTouchHelper.attachToRecyclerView(this) view.consumeFrom(homeFragmentStore, ProcessLifecycleOwner.get()) { update(it) } } } fun update(state: HomeFragmentState) { // Workaround for list not updating until scroll on Android 5 + 6 if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) { sessionControlAdapter.submitList(null) } sessionControlAdapter.submitList(state.toAdapterList()) } }