LibraryController.kt 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586
  1. package eu.kanade.tachiyomi.ui.library
  2. import android.content.res.Configuration
  3. import android.os.Bundle
  4. import android.view.LayoutInflater
  5. import android.view.Menu
  6. import android.view.MenuInflater
  7. import android.view.MenuItem
  8. import android.view.View
  9. import androidx.appcompat.app.AppCompatActivity
  10. import androidx.appcompat.view.ActionMode
  11. import androidx.core.graphics.drawable.DrawableCompat
  12. import androidx.core.view.isVisible
  13. import com.bluelinelabs.conductor.ControllerChangeHandler
  14. import com.bluelinelabs.conductor.ControllerChangeType
  15. import com.google.android.material.tabs.TabLayout
  16. import com.jakewharton.rxrelay.BehaviorRelay
  17. import com.jakewharton.rxrelay.PublishRelay
  18. import com.tfcporciuncula.flow.Preference
  19. import dev.chrisbanes.insetter.applyInsetter
  20. import eu.kanade.tachiyomi.R
  21. import eu.kanade.tachiyomi.data.database.models.Category
  22. import eu.kanade.tachiyomi.data.database.models.Manga
  23. import eu.kanade.tachiyomi.data.library.LibraryUpdateService
  24. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  25. import eu.kanade.tachiyomi.data.preference.asImmediateFlow
  26. import eu.kanade.tachiyomi.databinding.LibraryControllerBinding
  27. import eu.kanade.tachiyomi.source.LocalSource
  28. import eu.kanade.tachiyomi.ui.base.controller.RootController
  29. import eu.kanade.tachiyomi.ui.base.controller.SearchableNucleusController
  30. import eu.kanade.tachiyomi.ui.base.controller.TabbedController
  31. import eu.kanade.tachiyomi.ui.base.controller.withFadeTransaction
  32. import eu.kanade.tachiyomi.ui.browse.source.globalsearch.GlobalSearchController
  33. import eu.kanade.tachiyomi.ui.main.MainActivity
  34. import eu.kanade.tachiyomi.ui.manga.MangaController
  35. import eu.kanade.tachiyomi.util.system.getResourceColor
  36. import eu.kanade.tachiyomi.util.system.toast
  37. import kotlinx.coroutines.flow.drop
  38. import kotlinx.coroutines.flow.launchIn
  39. import kotlinx.coroutines.flow.onEach
  40. import reactivecircus.flowbinding.android.view.clicks
  41. import reactivecircus.flowbinding.viewpager.pageSelections
  42. import rx.Subscription
  43. import uy.kohesive.injekt.Injekt
  44. import uy.kohesive.injekt.api.get
  45. class LibraryController(
  46. bundle: Bundle? = null,
  47. private val preferences: PreferencesHelper = Injekt.get()
  48. ) : SearchableNucleusController<LibraryControllerBinding, LibraryPresenter>(bundle),
  49. RootController,
  50. TabbedController,
  51. ActionMode.Callback,
  52. ChangeMangaCategoriesDialog.Listener,
  53. DeleteLibraryMangasDialog.Listener {
  54. /**
  55. * Position of the active category.
  56. */
  57. private var activeCategory: Int = preferences.lastUsedCategory().get()
  58. /**
  59. * Action mode for selections.
  60. */
  61. private var actionMode: ActionMode? = null
  62. /**
  63. * Currently selected mangas.
  64. */
  65. val selectedMangas = mutableSetOf<Manga>()
  66. /**
  67. * Relay to notify the UI of selection updates.
  68. */
  69. val selectionRelay: PublishRelay<LibrarySelectionEvent> = PublishRelay.create()
  70. /**
  71. * Relay to notify search query changes.
  72. */
  73. val searchRelay: BehaviorRelay<String> = BehaviorRelay.create()
  74. /**
  75. * Relay to notify the library's viewpager for updates.
  76. */
  77. val libraryMangaRelay: BehaviorRelay<LibraryMangaEvent> = BehaviorRelay.create()
  78. /**
  79. * Relay to notify the library's viewpager to select all manga
  80. */
  81. val selectAllRelay: PublishRelay<Int> = PublishRelay.create()
  82. /**
  83. * Relay to notify the library's viewpager to select the inverse
  84. */
  85. val selectInverseRelay: PublishRelay<Int> = PublishRelay.create()
  86. /**
  87. * Number of manga per row in grid mode.
  88. */
  89. var mangaPerRow = 0
  90. private set
  91. /**
  92. * Adapter of the view pager.
  93. */
  94. private var adapter: LibraryAdapter? = null
  95. /**
  96. * Sheet containing filter/sort/display items.
  97. */
  98. private var settingsSheet: LibrarySettingsSheet? = null
  99. private var tabsVisibilityRelay: BehaviorRelay<Boolean> = BehaviorRelay.create(false)
  100. private var mangaCountVisibilityRelay: BehaviorRelay<Boolean> = BehaviorRelay.create(false)
  101. private var tabsVisibilitySubscription: Subscription? = null
  102. private var mangaCountVisibilitySubscription: Subscription? = null
  103. init {
  104. setHasOptionsMenu(true)
  105. retainViewMode = RetainViewMode.RETAIN_DETACH
  106. }
  107. private var currentTitle: String? = null
  108. set(value) {
  109. if (field != value) {
  110. field = value
  111. setTitle()
  112. }
  113. }
  114. override fun getTitle(): String? {
  115. return currentTitle ?: resources?.getString(R.string.label_library)
  116. }
  117. private fun updateTitle() {
  118. val showCategoryTabs = preferences.categoryTabs().get()
  119. val currentCategory = adapter?.categories?.getOrNull(binding.libraryPager.currentItem)
  120. var title = if (showCategoryTabs) {
  121. resources?.getString(R.string.label_library)
  122. } else {
  123. currentCategory?.name
  124. }
  125. if (preferences.categoryNumberOfItems().get() && libraryMangaRelay.hasValue()) {
  126. libraryMangaRelay.value.mangas.let { mangaMap ->
  127. if (!showCategoryTabs) {
  128. title += " (${mangaMap[currentCategory?.id]?.size ?: 0})"
  129. } else if (adapter?.categories?.size == 1) {
  130. // Only "Default" category
  131. title += " (${mangaMap[0]?.size ?: 0})"
  132. }
  133. }
  134. }
  135. currentTitle = title
  136. }
  137. override fun createPresenter(): LibraryPresenter {
  138. return LibraryPresenter()
  139. }
  140. override fun createBinding(inflater: LayoutInflater) = LibraryControllerBinding.inflate(inflater)
  141. override fun onViewCreated(view: View) {
  142. super.onViewCreated(view)
  143. binding.actionToolbar.applyInsetter {
  144. type(navigationBars = true) {
  145. margin(bottom = true)
  146. }
  147. }
  148. adapter = LibraryAdapter(this)
  149. binding.libraryPager.adapter = adapter
  150. binding.libraryPager.pageSelections()
  151. .onEach {
  152. preferences.lastUsedCategory().set(it)
  153. activeCategory = it
  154. updateTitle()
  155. }
  156. .launchIn(viewScope)
  157. getColumnsPreferenceForCurrentOrientation().asImmediateFlow { mangaPerRow = it }
  158. .drop(1)
  159. // Set again the adapter to recalculate the covers height
  160. .onEach { reattachAdapter() }
  161. .launchIn(viewScope)
  162. if (selectedMangas.isNotEmpty()) {
  163. createActionModeIfNeeded()
  164. }
  165. settingsSheet = LibrarySettingsSheet(router) { group ->
  166. when (group) {
  167. is LibrarySettingsSheet.Filter.FilterGroup -> onFilterChanged()
  168. is LibrarySettingsSheet.Sort.SortGroup -> onSortChanged()
  169. is LibrarySettingsSheet.Display.DisplayGroup -> reattachAdapter()
  170. is LibrarySettingsSheet.Display.BadgeGroup -> onBadgeSettingChanged()
  171. is LibrarySettingsSheet.Display.TabsGroup -> onTabsSettingsChanged()
  172. }
  173. }
  174. binding.btnGlobalSearch.clicks()
  175. .onEach {
  176. router.pushController(
  177. GlobalSearchController(presenter.query).withFadeTransaction()
  178. )
  179. }
  180. .launchIn(viewScope)
  181. (activity as? MainActivity)?.fixViewToBottom(binding.actionToolbar)
  182. }
  183. override fun onChangeStarted(handler: ControllerChangeHandler, type: ControllerChangeType) {
  184. super.onChangeStarted(handler, type)
  185. if (type.isEnter) {
  186. (activity as? MainActivity)?.binding?.tabs?.setupWithViewPager(binding.libraryPager)
  187. presenter.subscribeLibrary()
  188. }
  189. }
  190. override fun onDestroyView(view: View) {
  191. destroyActionModeIfNeeded()
  192. (activity as? MainActivity)?.clearFixViewToBottom(binding.actionToolbar)
  193. binding.actionToolbar.destroy()
  194. adapter?.onDestroy()
  195. adapter = null
  196. settingsSheet = null
  197. tabsVisibilitySubscription?.unsubscribe()
  198. tabsVisibilitySubscription = null
  199. super.onDestroyView(view)
  200. }
  201. override fun configureTabs(tabs: TabLayout) {
  202. with(tabs) {
  203. tabGravity = TabLayout.GRAVITY_START
  204. tabMode = TabLayout.MODE_SCROLLABLE
  205. }
  206. tabsVisibilitySubscription?.unsubscribe()
  207. tabsVisibilitySubscription = tabsVisibilityRelay.subscribe { visible ->
  208. val tabAnimator = (activity as? MainActivity)?.tabAnimator
  209. if (visible) {
  210. tabAnimator?.expand()
  211. } else {
  212. tabAnimator?.collapse()
  213. }
  214. }
  215. mangaCountVisibilitySubscription?.unsubscribe()
  216. mangaCountVisibilitySubscription = mangaCountVisibilityRelay.subscribe {
  217. adapter?.notifyDataSetChanged()
  218. }
  219. }
  220. override fun cleanupTabs(tabs: TabLayout) {
  221. tabsVisibilitySubscription?.unsubscribe()
  222. tabsVisibilitySubscription = null
  223. }
  224. fun showSettingsSheet() {
  225. settingsSheet?.show()
  226. }
  227. fun onNextLibraryUpdate(categories: List<Category>, mangaMap: Map<Int, List<LibraryItem>>) {
  228. val view = view ?: return
  229. val adapter = adapter ?: return
  230. // Show empty view if needed
  231. if (mangaMap.isNotEmpty()) {
  232. binding.emptyView.hide()
  233. } else {
  234. binding.emptyView.show(R.string.information_empty_library)
  235. }
  236. // Get the current active category.
  237. val activeCat = if (adapter.categories.isNotEmpty()) {
  238. binding.libraryPager.currentItem
  239. } else {
  240. activeCategory
  241. }
  242. // Set the categories
  243. adapter.categories = categories
  244. adapter.itemsPerCategory = adapter.categories
  245. .map { (it.id ?: -1) to (mangaMap[it.id]?.size ?: 0) }
  246. .toMap()
  247. // Restore active category.
  248. binding.libraryPager.setCurrentItem(activeCat, false)
  249. // Trigger display of tabs
  250. onTabsSettingsChanged()
  251. // Delay the scroll position to allow the view to be properly measured.
  252. view.post {
  253. if (isAttached) {
  254. (activity as? MainActivity)?.binding?.tabs?.setScrollPosition(binding.libraryPager.currentItem, 0f, true)
  255. }
  256. }
  257. // Send the manga map to child fragments after the adapter is updated.
  258. libraryMangaRelay.call(LibraryMangaEvent(mangaMap))
  259. // Finally update the title
  260. updateTitle()
  261. }
  262. /**
  263. * Returns a preference for the number of manga per row based on the current orientation.
  264. *
  265. * @return the preference.
  266. */
  267. private fun getColumnsPreferenceForCurrentOrientation(): Preference<Int> {
  268. return if (resources?.configuration?.orientation == Configuration.ORIENTATION_PORTRAIT) {
  269. preferences.portraitColumns()
  270. } else {
  271. preferences.landscapeColumns()
  272. }
  273. }
  274. private fun onFilterChanged() {
  275. presenter.requestFilterUpdate()
  276. activity?.invalidateOptionsMenu()
  277. }
  278. private fun onBadgeSettingChanged() {
  279. presenter.requestBadgesUpdate()
  280. }
  281. private fun onTabsSettingsChanged() {
  282. tabsVisibilityRelay.call(preferences.categoryTabs().get() && adapter?.categories?.size ?: 0 > 1)
  283. mangaCountVisibilityRelay.call(preferences.categoryNumberOfItems().get())
  284. updateTitle()
  285. }
  286. /**
  287. * Called when the sorting mode is changed.
  288. */
  289. private fun onSortChanged() {
  290. presenter.requestSortUpdate()
  291. }
  292. /**
  293. * Reattaches the adapter to the view pager to recreate fragments
  294. */
  295. private fun reattachAdapter() {
  296. val adapter = adapter ?: return
  297. val position = binding.libraryPager.currentItem
  298. adapter.recycle = false
  299. binding.libraryPager.adapter = adapter
  300. binding.libraryPager.currentItem = position
  301. adapter.recycle = true
  302. }
  303. /**
  304. * Creates the action mode if it's not created already.
  305. */
  306. fun createActionModeIfNeeded() {
  307. if (actionMode == null) {
  308. actionMode = (activity as AppCompatActivity).startSupportActionMode(this)
  309. binding.actionToolbar.show(
  310. actionMode!!,
  311. R.menu.library_selection
  312. ) { onActionItemClicked(it!!) }
  313. (activity as? MainActivity)?.showNav(visible = false, collapse = true)
  314. }
  315. }
  316. /**
  317. * Destroys the action mode.
  318. */
  319. private fun destroyActionModeIfNeeded() {
  320. actionMode?.finish()
  321. }
  322. override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
  323. createOptionsMenu(menu, inflater, R.menu.library, R.id.action_search)
  324. // Mutate the filter icon because it needs to be tinted and the resource is shared.
  325. menu.findItem(R.id.action_filter).icon.mutate()
  326. }
  327. fun search(query: String) {
  328. presenter.query = query
  329. }
  330. private fun performSearch() {
  331. searchRelay.call(presenter.query)
  332. if (presenter.query.isNotEmpty()) {
  333. binding.btnGlobalSearch.isVisible = true
  334. binding.btnGlobalSearch.text =
  335. resources?.getString(R.string.action_global_search_query, presenter.query)
  336. } else {
  337. binding.btnGlobalSearch.isVisible = false
  338. }
  339. }
  340. override fun onPrepareOptionsMenu(menu: Menu) {
  341. val settingsSheet = settingsSheet ?: return
  342. val filterItem = menu.findItem(R.id.action_filter)
  343. // Tint icon if there's a filter active
  344. if (settingsSheet.filters.hasActiveFilters()) {
  345. val filterColor = activity!!.getResourceColor(R.attr.colorFilterActive)
  346. DrawableCompat.setTint(filterItem.icon, filterColor)
  347. }
  348. }
  349. override fun onOptionsItemSelected(item: MenuItem): Boolean {
  350. when (item.itemId) {
  351. R.id.action_search -> expandActionViewFromInteraction = true
  352. R.id.action_filter -> showSettingsSheet()
  353. R.id.action_update_library -> {
  354. activity?.let {
  355. if (LibraryUpdateService.start(it)) {
  356. it.toast(R.string.updating_library)
  357. }
  358. }
  359. }
  360. }
  361. return super.onOptionsItemSelected(item)
  362. }
  363. /**
  364. * Invalidates the action mode, forcing it to refresh its content.
  365. */
  366. fun invalidateActionMode() {
  367. actionMode?.invalidate()
  368. }
  369. override fun onCreateActionMode(mode: ActionMode, menu: Menu): Boolean {
  370. mode.menuInflater.inflate(R.menu.generic_selection, menu)
  371. return true
  372. }
  373. override fun onPrepareActionMode(mode: ActionMode, menu: Menu): Boolean {
  374. val count = selectedMangas.size
  375. if (count == 0) {
  376. // Destroy action mode if there are no items selected.
  377. destroyActionModeIfNeeded()
  378. } else {
  379. mode.title = count.toString()
  380. binding.actionToolbar.findItem(R.id.action_download_unread)?.isVisible = selectedMangas.any { it.source != LocalSource.ID }
  381. }
  382. return false
  383. }
  384. override fun onActionItemClicked(mode: ActionMode, item: MenuItem): Boolean {
  385. return onActionItemClicked(item)
  386. }
  387. private fun onActionItemClicked(item: MenuItem): Boolean {
  388. when (item.itemId) {
  389. R.id.action_move_to_category -> showChangeMangaCategoriesDialog()
  390. R.id.action_download_unread -> downloadUnreadChapters()
  391. R.id.action_mark_as_read -> markReadStatus(true)
  392. R.id.action_mark_as_unread -> markReadStatus(false)
  393. R.id.action_delete -> showDeleteMangaDialog()
  394. R.id.action_select_all -> selectAllCategoryManga()
  395. R.id.action_select_inverse -> selectInverseCategoryManga()
  396. else -> return false
  397. }
  398. return true
  399. }
  400. override fun onDestroyActionMode(mode: ActionMode?) {
  401. // Clear all the manga selections and notify child views.
  402. selectedMangas.clear()
  403. selectionRelay.call(LibrarySelectionEvent.Cleared())
  404. binding.actionToolbar.hide()
  405. (activity as? MainActivity)?.showNav(visible = true, collapse = true)
  406. actionMode = null
  407. }
  408. fun openManga(manga: Manga) {
  409. // Notify the presenter a manga is being opened.
  410. presenter.onOpenManga()
  411. router.pushController(MangaController(manga).withFadeTransaction())
  412. }
  413. /**
  414. * Sets the selection for a given manga.
  415. *
  416. * @param manga the manga whose selection has changed.
  417. * @param selected whether it's now selected or not.
  418. */
  419. fun setSelection(manga: Manga, selected: Boolean) {
  420. if (selected) {
  421. if (selectedMangas.add(manga)) {
  422. selectionRelay.call(LibrarySelectionEvent.Selected(manga))
  423. }
  424. } else {
  425. if (selectedMangas.remove(manga)) {
  426. selectionRelay.call(LibrarySelectionEvent.Unselected(manga))
  427. }
  428. }
  429. }
  430. /**
  431. * Toggles the current selection state for a given manga.
  432. *
  433. * @param manga the manga whose selection to change.
  434. */
  435. fun toggleSelection(manga: Manga) {
  436. if (selectedMangas.add(manga)) {
  437. selectionRelay.call(LibrarySelectionEvent.Selected(manga))
  438. } else if (selectedMangas.remove(manga)) {
  439. selectionRelay.call(LibrarySelectionEvent.Unselected(manga))
  440. }
  441. }
  442. /**
  443. * Move the selected manga to a list of categories.
  444. */
  445. private fun showChangeMangaCategoriesDialog() {
  446. // Create a copy of selected manga
  447. val mangas = selectedMangas.toList()
  448. // Hide the default category because it has a different behavior than the ones from db.
  449. val categories = presenter.categories.filter { it.id != 0 }
  450. // Get indexes of the common categories to preselect.
  451. val commonCategoriesIndexes = presenter.getCommonCategories(mangas)
  452. .map { categories.indexOf(it) }
  453. .toTypedArray()
  454. ChangeMangaCategoriesDialog(this, mangas, categories, commonCategoriesIndexes)
  455. .showDialog(router)
  456. }
  457. private fun downloadUnreadChapters() {
  458. val mangas = selectedMangas.toList()
  459. presenter.downloadUnreadChapters(mangas)
  460. destroyActionModeIfNeeded()
  461. }
  462. private fun markReadStatus(read: Boolean) {
  463. val mangas = selectedMangas.toList()
  464. presenter.markReadStatus(mangas, read)
  465. destroyActionModeIfNeeded()
  466. }
  467. private fun showDeleteMangaDialog() {
  468. DeleteLibraryMangasDialog(this, selectedMangas.toList()).showDialog(router)
  469. }
  470. override fun updateCategoriesForMangas(mangas: List<Manga>, categories: List<Category>) {
  471. presenter.moveMangasToCategories(categories, mangas)
  472. destroyActionModeIfNeeded()
  473. }
  474. override fun deleteMangas(mangas: List<Manga>, deleteFromLibrary: Boolean, deleteChapters: Boolean) {
  475. presenter.removeMangas(mangas, deleteFromLibrary, deleteChapters)
  476. destroyActionModeIfNeeded()
  477. }
  478. private fun selectAllCategoryManga() {
  479. adapter?.categories?.getOrNull(binding.libraryPager.currentItem)?.id?.let {
  480. selectAllRelay.call(it)
  481. }
  482. }
  483. private fun selectInverseCategoryManga() {
  484. adapter?.categories?.getOrNull(binding.libraryPager.currentItem)?.id?.let {
  485. selectInverseRelay.call(it)
  486. }
  487. }
  488. override fun onSearchViewQueryTextChange(newText: String?) {
  489. // Ignore events if this controller isn't at the top to avoid query being reset
  490. if (router.backstack.lastOrNull()?.controller() == this) {
  491. presenter.query = newText ?: ""
  492. performSearch()
  493. }
  494. }
  495. }