123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586 |
- package eu.kanade.tachiyomi.ui.library
- import android.content.res.Configuration
- import android.os.Bundle
- import android.view.LayoutInflater
- import android.view.Menu
- import android.view.MenuInflater
- import android.view.MenuItem
- import android.view.View
- import androidx.appcompat.app.AppCompatActivity
- import androidx.appcompat.view.ActionMode
- import androidx.core.graphics.drawable.DrawableCompat
- import androidx.core.view.isVisible
- import com.bluelinelabs.conductor.ControllerChangeHandler
- import com.bluelinelabs.conductor.ControllerChangeType
- import com.google.android.material.tabs.TabLayout
- import com.jakewharton.rxrelay.BehaviorRelay
- import com.jakewharton.rxrelay.PublishRelay
- import com.tfcporciuncula.flow.Preference
- import dev.chrisbanes.insetter.applyInsetter
- import eu.kanade.tachiyomi.R
- import eu.kanade.tachiyomi.data.database.models.Category
- import eu.kanade.tachiyomi.data.database.models.Manga
- import eu.kanade.tachiyomi.data.library.LibraryUpdateService
- import eu.kanade.tachiyomi.data.preference.PreferencesHelper
- import eu.kanade.tachiyomi.data.preference.asImmediateFlow
- import eu.kanade.tachiyomi.databinding.LibraryControllerBinding
- import eu.kanade.tachiyomi.source.LocalSource
- import eu.kanade.tachiyomi.ui.base.controller.RootController
- import eu.kanade.tachiyomi.ui.base.controller.SearchableNucleusController
- import eu.kanade.tachiyomi.ui.base.controller.TabbedController
- import eu.kanade.tachiyomi.ui.base.controller.withFadeTransaction
- import eu.kanade.tachiyomi.ui.browse.source.globalsearch.GlobalSearchController
- import eu.kanade.tachiyomi.ui.main.MainActivity
- import eu.kanade.tachiyomi.ui.manga.MangaController
- import eu.kanade.tachiyomi.util.system.getResourceColor
- import eu.kanade.tachiyomi.util.system.toast
- import kotlinx.coroutines.flow.drop
- import kotlinx.coroutines.flow.launchIn
- import kotlinx.coroutines.flow.onEach
- import reactivecircus.flowbinding.android.view.clicks
- import reactivecircus.flowbinding.viewpager.pageSelections
- import rx.Subscription
- import uy.kohesive.injekt.Injekt
- import uy.kohesive.injekt.api.get
- class LibraryController(
- bundle: Bundle? = null,
- private val preferences: PreferencesHelper = Injekt.get()
- ) : SearchableNucleusController<LibraryControllerBinding, LibraryPresenter>(bundle),
- RootController,
- TabbedController,
- ActionMode.Callback,
- ChangeMangaCategoriesDialog.Listener,
- DeleteLibraryMangasDialog.Listener {
- /**
- * Position of the active category.
- */
- private var activeCategory: Int = preferences.lastUsedCategory().get()
- /**
- * Action mode for selections.
- */
- private var actionMode: ActionMode? = null
- /**
- * Currently selected mangas.
- */
- val selectedMangas = mutableSetOf<Manga>()
- /**
- * Relay to notify the UI of selection updates.
- */
- val selectionRelay: PublishRelay<LibrarySelectionEvent> = PublishRelay.create()
- /**
- * Relay to notify search query changes.
- */
- val searchRelay: BehaviorRelay<String> = BehaviorRelay.create()
- /**
- * Relay to notify the library's viewpager for updates.
- */
- val libraryMangaRelay: BehaviorRelay<LibraryMangaEvent> = BehaviorRelay.create()
- /**
- * Relay to notify the library's viewpager to select all manga
- */
- val selectAllRelay: PublishRelay<Int> = PublishRelay.create()
- /**
- * Relay to notify the library's viewpager to select the inverse
- */
- val selectInverseRelay: PublishRelay<Int> = PublishRelay.create()
- /**
- * Number of manga per row in grid mode.
- */
- var mangaPerRow = 0
- private set
- /**
- * Adapter of the view pager.
- */
- private var adapter: LibraryAdapter? = null
- /**
- * Sheet containing filter/sort/display items.
- */
- private var settingsSheet: LibrarySettingsSheet? = null
- private var tabsVisibilityRelay: BehaviorRelay<Boolean> = BehaviorRelay.create(false)
- private var mangaCountVisibilityRelay: BehaviorRelay<Boolean> = BehaviorRelay.create(false)
- private var tabsVisibilitySubscription: Subscription? = null
- private var mangaCountVisibilitySubscription: Subscription? = null
- init {
- setHasOptionsMenu(true)
- retainViewMode = RetainViewMode.RETAIN_DETACH
- }
- private var currentTitle: String? = null
- set(value) {
- if (field != value) {
- field = value
- setTitle()
- }
- }
- override fun getTitle(): String? {
- return currentTitle ?: resources?.getString(R.string.label_library)
- }
- private fun updateTitle() {
- val showCategoryTabs = preferences.categoryTabs().get()
- val currentCategory = adapter?.categories?.getOrNull(binding.libraryPager.currentItem)
- var title = if (showCategoryTabs) {
- resources?.getString(R.string.label_library)
- } else {
- currentCategory?.name
- }
- if (preferences.categoryNumberOfItems().get() && libraryMangaRelay.hasValue()) {
- libraryMangaRelay.value.mangas.let { mangaMap ->
- if (!showCategoryTabs) {
- title += " (${mangaMap[currentCategory?.id]?.size ?: 0})"
- } else if (adapter?.categories?.size == 1) {
- // Only "Default" category
- title += " (${mangaMap[0]?.size ?: 0})"
- }
- }
- }
- currentTitle = title
- }
- override fun createPresenter(): LibraryPresenter {
- return LibraryPresenter()
- }
- override fun createBinding(inflater: LayoutInflater) = LibraryControllerBinding.inflate(inflater)
- override fun onViewCreated(view: View) {
- super.onViewCreated(view)
- binding.actionToolbar.applyInsetter {
- type(navigationBars = true) {
- margin(bottom = true)
- }
- }
- adapter = LibraryAdapter(this)
- binding.libraryPager.adapter = adapter
- binding.libraryPager.pageSelections()
- .onEach {
- preferences.lastUsedCategory().set(it)
- activeCategory = it
- updateTitle()
- }
- .launchIn(viewScope)
- getColumnsPreferenceForCurrentOrientation().asImmediateFlow { mangaPerRow = it }
- .drop(1)
- // Set again the adapter to recalculate the covers height
- .onEach { reattachAdapter() }
- .launchIn(viewScope)
- if (selectedMangas.isNotEmpty()) {
- createActionModeIfNeeded()
- }
- settingsSheet = LibrarySettingsSheet(router) { group ->
- when (group) {
- is LibrarySettingsSheet.Filter.FilterGroup -> onFilterChanged()
- is LibrarySettingsSheet.Sort.SortGroup -> onSortChanged()
- is LibrarySettingsSheet.Display.DisplayGroup -> reattachAdapter()
- is LibrarySettingsSheet.Display.BadgeGroup -> onBadgeSettingChanged()
- is LibrarySettingsSheet.Display.TabsGroup -> onTabsSettingsChanged()
- }
- }
- binding.btnGlobalSearch.clicks()
- .onEach {
- router.pushController(
- GlobalSearchController(presenter.query).withFadeTransaction()
- )
- }
- .launchIn(viewScope)
- (activity as? MainActivity)?.fixViewToBottom(binding.actionToolbar)
- }
- override fun onChangeStarted(handler: ControllerChangeHandler, type: ControllerChangeType) {
- super.onChangeStarted(handler, type)
- if (type.isEnter) {
- (activity as? MainActivity)?.binding?.tabs?.setupWithViewPager(binding.libraryPager)
- presenter.subscribeLibrary()
- }
- }
- override fun onDestroyView(view: View) {
- destroyActionModeIfNeeded()
- (activity as? MainActivity)?.clearFixViewToBottom(binding.actionToolbar)
- binding.actionToolbar.destroy()
- adapter?.onDestroy()
- adapter = null
- settingsSheet = null
- tabsVisibilitySubscription?.unsubscribe()
- tabsVisibilitySubscription = null
- super.onDestroyView(view)
- }
- override fun configureTabs(tabs: TabLayout) {
- with(tabs) {
- tabGravity = TabLayout.GRAVITY_START
- tabMode = TabLayout.MODE_SCROLLABLE
- }
- tabsVisibilitySubscription?.unsubscribe()
- tabsVisibilitySubscription = tabsVisibilityRelay.subscribe { visible ->
- val tabAnimator = (activity as? MainActivity)?.tabAnimator
- if (visible) {
- tabAnimator?.expand()
- } else {
- tabAnimator?.collapse()
- }
- }
- mangaCountVisibilitySubscription?.unsubscribe()
- mangaCountVisibilitySubscription = mangaCountVisibilityRelay.subscribe {
- adapter?.notifyDataSetChanged()
- }
- }
- override fun cleanupTabs(tabs: TabLayout) {
- tabsVisibilitySubscription?.unsubscribe()
- tabsVisibilitySubscription = null
- }
- fun showSettingsSheet() {
- settingsSheet?.show()
- }
- fun onNextLibraryUpdate(categories: List<Category>, mangaMap: Map<Int, List<LibraryItem>>) {
- val view = view ?: return
- val adapter = adapter ?: return
- // Show empty view if needed
- if (mangaMap.isNotEmpty()) {
- binding.emptyView.hide()
- } else {
- binding.emptyView.show(R.string.information_empty_library)
- }
- // Get the current active category.
- val activeCat = if (adapter.categories.isNotEmpty()) {
- binding.libraryPager.currentItem
- } else {
- activeCategory
- }
- // Set the categories
- adapter.categories = categories
- adapter.itemsPerCategory = adapter.categories
- .map { (it.id ?: -1) to (mangaMap[it.id]?.size ?: 0) }
- .toMap()
- // Restore active category.
- binding.libraryPager.setCurrentItem(activeCat, false)
- // Trigger display of tabs
- onTabsSettingsChanged()
- // Delay the scroll position to allow the view to be properly measured.
- view.post {
- if (isAttached) {
- (activity as? MainActivity)?.binding?.tabs?.setScrollPosition(binding.libraryPager.currentItem, 0f, true)
- }
- }
- // Send the manga map to child fragments after the adapter is updated.
- libraryMangaRelay.call(LibraryMangaEvent(mangaMap))
- // Finally update the title
- updateTitle()
- }
- /**
- * Returns a preference for the number of manga per row based on the current orientation.
- *
- * @return the preference.
- */
- private fun getColumnsPreferenceForCurrentOrientation(): Preference<Int> {
- return if (resources?.configuration?.orientation == Configuration.ORIENTATION_PORTRAIT) {
- preferences.portraitColumns()
- } else {
- preferences.landscapeColumns()
- }
- }
- private fun onFilterChanged() {
- presenter.requestFilterUpdate()
- activity?.invalidateOptionsMenu()
- }
- private fun onBadgeSettingChanged() {
- presenter.requestBadgesUpdate()
- }
- private fun onTabsSettingsChanged() {
- tabsVisibilityRelay.call(preferences.categoryTabs().get() && adapter?.categories?.size ?: 0 > 1)
- mangaCountVisibilityRelay.call(preferences.categoryNumberOfItems().get())
- updateTitle()
- }
- /**
- * Called when the sorting mode is changed.
- */
- private fun onSortChanged() {
- presenter.requestSortUpdate()
- }
- /**
- * Reattaches the adapter to the view pager to recreate fragments
- */
- private fun reattachAdapter() {
- val adapter = adapter ?: return
- val position = binding.libraryPager.currentItem
- adapter.recycle = false
- binding.libraryPager.adapter = adapter
- binding.libraryPager.currentItem = position
- adapter.recycle = true
- }
- /**
- * Creates the action mode if it's not created already.
- */
- fun createActionModeIfNeeded() {
- if (actionMode == null) {
- actionMode = (activity as AppCompatActivity).startSupportActionMode(this)
- binding.actionToolbar.show(
- actionMode!!,
- R.menu.library_selection
- ) { onActionItemClicked(it!!) }
- (activity as? MainActivity)?.showNav(visible = false, collapse = true)
- }
- }
- /**
- * Destroys the action mode.
- */
- private fun destroyActionModeIfNeeded() {
- actionMode?.finish()
- }
- override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
- createOptionsMenu(menu, inflater, R.menu.library, R.id.action_search)
- // Mutate the filter icon because it needs to be tinted and the resource is shared.
- menu.findItem(R.id.action_filter).icon.mutate()
- }
- fun search(query: String) {
- presenter.query = query
- }
- private fun performSearch() {
- searchRelay.call(presenter.query)
- if (presenter.query.isNotEmpty()) {
- binding.btnGlobalSearch.isVisible = true
- binding.btnGlobalSearch.text =
- resources?.getString(R.string.action_global_search_query, presenter.query)
- } else {
- binding.btnGlobalSearch.isVisible = false
- }
- }
- override fun onPrepareOptionsMenu(menu: Menu) {
- val settingsSheet = settingsSheet ?: return
- val filterItem = menu.findItem(R.id.action_filter)
- // Tint icon if there's a filter active
- if (settingsSheet.filters.hasActiveFilters()) {
- val filterColor = activity!!.getResourceColor(R.attr.colorFilterActive)
- DrawableCompat.setTint(filterItem.icon, filterColor)
- }
- }
- override fun onOptionsItemSelected(item: MenuItem): Boolean {
- when (item.itemId) {
- R.id.action_search -> expandActionViewFromInteraction = true
- R.id.action_filter -> showSettingsSheet()
- R.id.action_update_library -> {
- activity?.let {
- if (LibraryUpdateService.start(it)) {
- it.toast(R.string.updating_library)
- }
- }
- }
- }
- return super.onOptionsItemSelected(item)
- }
- /**
- * Invalidates the action mode, forcing it to refresh its content.
- */
- fun invalidateActionMode() {
- actionMode?.invalidate()
- }
- override fun onCreateActionMode(mode: ActionMode, menu: Menu): Boolean {
- mode.menuInflater.inflate(R.menu.generic_selection, menu)
- return true
- }
- override fun onPrepareActionMode(mode: ActionMode, menu: Menu): Boolean {
- val count = selectedMangas.size
- if (count == 0) {
- // Destroy action mode if there are no items selected.
- destroyActionModeIfNeeded()
- } else {
- mode.title = count.toString()
- binding.actionToolbar.findItem(R.id.action_download_unread)?.isVisible = selectedMangas.any { it.source != LocalSource.ID }
- }
- return false
- }
- override fun onActionItemClicked(mode: ActionMode, item: MenuItem): Boolean {
- return onActionItemClicked(item)
- }
- private fun onActionItemClicked(item: MenuItem): Boolean {
- when (item.itemId) {
- R.id.action_move_to_category -> showChangeMangaCategoriesDialog()
- R.id.action_download_unread -> downloadUnreadChapters()
- R.id.action_mark_as_read -> markReadStatus(true)
- R.id.action_mark_as_unread -> markReadStatus(false)
- R.id.action_delete -> showDeleteMangaDialog()
- R.id.action_select_all -> selectAllCategoryManga()
- R.id.action_select_inverse -> selectInverseCategoryManga()
- else -> return false
- }
- return true
- }
- override fun onDestroyActionMode(mode: ActionMode?) {
- // Clear all the manga selections and notify child views.
- selectedMangas.clear()
- selectionRelay.call(LibrarySelectionEvent.Cleared())
- binding.actionToolbar.hide()
- (activity as? MainActivity)?.showNav(visible = true, collapse = true)
- actionMode = null
- }
- fun openManga(manga: Manga) {
- // Notify the presenter a manga is being opened.
- presenter.onOpenManga()
- router.pushController(MangaController(manga).withFadeTransaction())
- }
- /**
- * Sets the selection for a given manga.
- *
- * @param manga the manga whose selection has changed.
- * @param selected whether it's now selected or not.
- */
- fun setSelection(manga: Manga, selected: Boolean) {
- if (selected) {
- if (selectedMangas.add(manga)) {
- selectionRelay.call(LibrarySelectionEvent.Selected(manga))
- }
- } else {
- if (selectedMangas.remove(manga)) {
- selectionRelay.call(LibrarySelectionEvent.Unselected(manga))
- }
- }
- }
- /**
- * Toggles the current selection state for a given manga.
- *
- * @param manga the manga whose selection to change.
- */
- fun toggleSelection(manga: Manga) {
- if (selectedMangas.add(manga)) {
- selectionRelay.call(LibrarySelectionEvent.Selected(manga))
- } else if (selectedMangas.remove(manga)) {
- selectionRelay.call(LibrarySelectionEvent.Unselected(manga))
- }
- }
- /**
- * Move the selected manga to a list of categories.
- */
- private fun showChangeMangaCategoriesDialog() {
- // Create a copy of selected manga
- val mangas = selectedMangas.toList()
- // Hide the default category because it has a different behavior than the ones from db.
- val categories = presenter.categories.filter { it.id != 0 }
- // Get indexes of the common categories to preselect.
- val commonCategoriesIndexes = presenter.getCommonCategories(mangas)
- .map { categories.indexOf(it) }
- .toTypedArray()
- ChangeMangaCategoriesDialog(this, mangas, categories, commonCategoriesIndexes)
- .showDialog(router)
- }
- private fun downloadUnreadChapters() {
- val mangas = selectedMangas.toList()
- presenter.downloadUnreadChapters(mangas)
- destroyActionModeIfNeeded()
- }
- private fun markReadStatus(read: Boolean) {
- val mangas = selectedMangas.toList()
- presenter.markReadStatus(mangas, read)
- destroyActionModeIfNeeded()
- }
- private fun showDeleteMangaDialog() {
- DeleteLibraryMangasDialog(this, selectedMangas.toList()).showDialog(router)
- }
- override fun updateCategoriesForMangas(mangas: List<Manga>, categories: List<Category>) {
- presenter.moveMangasToCategories(categories, mangas)
- destroyActionModeIfNeeded()
- }
- override fun deleteMangas(mangas: List<Manga>, deleteFromLibrary: Boolean, deleteChapters: Boolean) {
- presenter.removeMangas(mangas, deleteFromLibrary, deleteChapters)
- destroyActionModeIfNeeded()
- }
- private fun selectAllCategoryManga() {
- adapter?.categories?.getOrNull(binding.libraryPager.currentItem)?.id?.let {
- selectAllRelay.call(it)
- }
- }
- private fun selectInverseCategoryManga() {
- adapter?.categories?.getOrNull(binding.libraryPager.currentItem)?.id?.let {
- selectInverseRelay.call(it)
- }
- }
- override fun onSearchViewQueryTextChange(newText: String?) {
- // Ignore events if this controller isn't at the top to avoid query being reset
- if (router.backstack.lastOrNull()?.controller() == this) {
- presenter.query = newText ?: ""
- performSearch()
- }
- }
- }
|