/* 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.collections import android.app.Dialog import android.os.Bundle import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import androidx.fragment.app.DialogFragment import androidx.lifecycle.ViewModelProviders import androidx.lifecycle.lifecycleScope import kotlinx.android.synthetic.main.fragment_create_collection.view.* import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import org.mozilla.fenix.FenixViewModelProvider import org.mozilla.fenix.R import org.mozilla.fenix.components.metrics.Event import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.requireComponents import org.mozilla.fenix.home.sessioncontrol.Tab import org.mozilla.fenix.home.sessioncontrol.toSessionBundle import org.mozilla.fenix.mvi.ActionBusFactory import org.mozilla.fenix.mvi.getAutoDisposeObservable import org.mozilla.fenix.mvi.getManagedEmitter class CreateCollectionFragment : DialogFragment() { private lateinit var collectionCreationComponent: CollectionCreationComponent private lateinit var viewModel: CreateCollectionViewModel override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) isCancelable = false setStyle(STYLE_NO_TITLE, R.style.CreateCollectionDialogStyle) } override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { val view = inflater.inflate(R.layout.fragment_create_collection, container, false) viewModel = activity!!.run { ViewModelProviders.of(this).get(CreateCollectionViewModel::class.java) } collectionCreationComponent = CollectionCreationComponent( view.createCollectionWrapper, ActionBusFactory.get(this), FenixViewModelProvider.create( this, CollectionCreationViewModel::class.java ) { CollectionCreationViewModel( CollectionCreationState( viewModel.tabs, viewModel.selectedTabs, viewModel.saveCollectionStep, viewModel.tabCollections, viewModel.selectedTabCollection ) ) } ) return view } override fun onCreateDialog(savedInstanceState: Bundle?): Dialog { val dialog = super.onCreateDialog(savedInstanceState) dialog.setOnKeyListener { _, keyCode, event -> (collectionCreationComponent.uiView as CollectionCreationUIView).onKey(keyCode, event) } return dialog } override fun onResume() { super.onResume() (collectionCreationComponent.uiView as CollectionCreationUIView).onResumed() subscribeToActions() } @Suppress("ComplexMethod") private fun subscribeToActions() { getAutoDisposeObservable().subscribe { when (it) { is CollectionCreationAction.Close -> dismiss() is CollectionCreationAction.SaveTabsToCollection -> { getManagedEmitter() .onNext( CollectionCreationChange.StepChanged( viewModel.tabCollections.getStepForCollectionsSize() ) ) } is CollectionCreationAction.AddTabToSelection -> { getManagedEmitter() .onNext(CollectionCreationChange.TabAdded(it.tab)) } is CollectionCreationAction.RemoveTabFromSelection -> { getManagedEmitter() .onNext(CollectionCreationChange.TabRemoved(it.tab)) } is CollectionCreationAction.SelectAllTapped -> { getManagedEmitter() .onNext(CollectionCreationChange.AddAllTabs) } is CollectionCreationAction.DeselectAllTapped -> { getManagedEmitter() .onNext(CollectionCreationChange.RemoveAllTabs) } is CollectionCreationAction.AddNewCollection -> getManagedEmitter().onNext( CollectionCreationChange.StepChanged(SaveCollectionStep.NameCollection) ) is CollectionCreationAction.BackPressed -> handleBackPress(backPressFrom = it.backPressFrom) is CollectionCreationAction.SaveCollectionName -> { dismiss() context?.let { context -> val sessionBundle = it.tabs.toList().toSessionBundle(context) viewLifecycleOwner.lifecycleScope.launch(Dispatchers.IO) { context.components.core.tabCollectionStorage.createCollection(it.name, sessionBundle) } context.components.analytics.metrics.track( Event.CollectionSaved(context.components.core.sessionManager.size, sessionBundle.size) ) closeTabsIfNecessary(it.tabs) } } is CollectionCreationAction.SelectCollection -> { dismiss() context?.let { context -> val sessionBundle = it.tabs.toList().toSessionBundle(context) viewLifecycleOwner.lifecycleScope.launch(Dispatchers.IO) { context.components.core.tabCollectionStorage .addTabsToCollection(it.collection, sessionBundle) } context.components.analytics.metrics.track( Event.CollectionTabsAdded(context.components.core.sessionManager.size, sessionBundle.size) ) closeTabsIfNecessary(it.tabs) } } is CollectionCreationAction.RenameCollection -> { dismiss() viewLifecycleOwner.lifecycleScope.launch(Dispatchers.IO) { context?.components?.core?.tabCollectionStorage?.renameCollection(it.collection, it.name) context?.components?.analytics?.metrics?.track(Event.CollectionRenamed) } } } } } private fun handleBackPress(backPressFrom: SaveCollectionStep) { when (backPressFrom) { SaveCollectionStep.SelectTabs -> dismiss() SaveCollectionStep.SelectCollection -> { if (viewModel.tabs.size <= 1) dismiss() else { getManagedEmitter().onNext( CollectionCreationChange.StepChanged(SaveCollectionStep.SelectTabs) ) } } SaveCollectionStep.NameCollection -> { if (viewModel.tabCollections.isEmpty() && viewModel.tabs.size == 1) { dismiss() } else { getManagedEmitter() .onNext( CollectionCreationChange.StepChanged( viewModel.tabCollections.getBackStepForCollectionsSize() ) ) } } SaveCollectionStep.RenameCollection -> { dismiss() } } } private fun closeTabsIfNecessary(tabs: List) { // Only close the tabs if the user is not on the BrowserFragment if (viewModel.previousFragmentId == R.id.browserFragment) { return } tabs.forEach { requireComponents.core.sessionManager.findSessionById(it.sessionId)?.let { session -> requireComponents.useCases.tabsUseCases.removeTab.invoke(session) } } } }