LibraryController.kt 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573
  1. package eu.kanade.tachiyomi.ui.library
  2. import android.app.Activity
  3. import android.content.Intent
  4. import android.content.res.Configuration
  5. import android.graphics.Color
  6. import android.net.Uri
  7. import android.os.Bundle
  8. import android.view.LayoutInflater
  9. import android.view.Menu
  10. import android.view.MenuInflater
  11. import android.view.MenuItem
  12. import android.view.View
  13. import android.view.ViewGroup
  14. import androidx.appcompat.app.AppCompatActivity
  15. import androidx.appcompat.view.ActionMode
  16. import androidx.appcompat.widget.SearchView
  17. import androidx.core.graphics.drawable.DrawableCompat
  18. import androidx.core.view.GravityCompat
  19. import androidx.drawerlayout.widget.DrawerLayout
  20. import com.bluelinelabs.conductor.ControllerChangeHandler
  21. import com.bluelinelabs.conductor.ControllerChangeType
  22. import com.f2prateek.rx.preferences.Preference
  23. import com.google.android.material.tabs.TabLayout
  24. import com.jakewharton.rxbinding.support.v4.view.pageSelections
  25. import com.jakewharton.rxbinding.support.v7.widget.queryTextChanges
  26. import com.jakewharton.rxrelay.BehaviorRelay
  27. import com.jakewharton.rxrelay.PublishRelay
  28. import eu.kanade.tachiyomi.R
  29. import eu.kanade.tachiyomi.data.database.models.Category
  30. import eu.kanade.tachiyomi.data.database.models.Manga
  31. import eu.kanade.tachiyomi.data.library.LibraryUpdateService
  32. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  33. import eu.kanade.tachiyomi.data.preference.getOrDefault
  34. import eu.kanade.tachiyomi.ui.base.controller.NucleusController
  35. import eu.kanade.tachiyomi.ui.base.controller.RootController
  36. import eu.kanade.tachiyomi.ui.base.controller.SecondaryDrawerController
  37. import eu.kanade.tachiyomi.ui.base.controller.TabbedController
  38. import eu.kanade.tachiyomi.ui.base.controller.withFadeTransaction
  39. import eu.kanade.tachiyomi.ui.main.MainActivity
  40. import eu.kanade.tachiyomi.ui.manga.MangaController
  41. import eu.kanade.tachiyomi.util.system.toast
  42. import eu.kanade.tachiyomi.util.view.inflate
  43. import java.io.IOException
  44. import kotlinx.android.synthetic.main.library_controller.action_toolbar
  45. import kotlinx.android.synthetic.main.library_controller.empty_view
  46. import kotlinx.android.synthetic.main.library_controller.library_pager
  47. import kotlinx.android.synthetic.main.main_activity.drawer
  48. import kotlinx.android.synthetic.main.main_activity.tabs
  49. import rx.Subscription
  50. import timber.log.Timber
  51. import uy.kohesive.injekt.Injekt
  52. import uy.kohesive.injekt.api.get
  53. class LibraryController(
  54. bundle: Bundle? = null,
  55. private val preferences: PreferencesHelper = Injekt.get()
  56. ) : NucleusController<LibraryPresenter>(bundle),
  57. RootController,
  58. TabbedController,
  59. SecondaryDrawerController,
  60. ActionMode.Callback,
  61. ChangeMangaCategoriesDialog.Listener,
  62. DeleteLibraryMangasDialog.Listener {
  63. /**
  64. * Position of the active category.
  65. */
  66. var activeCategory: Int = preferences.lastUsedCategory().getOrDefault()
  67. private set
  68. /**
  69. * Action mode for selections.
  70. */
  71. private var actionMode: ActionMode? = null
  72. /**
  73. * Library search query.
  74. */
  75. private var query = ""
  76. /**
  77. * Currently selected mangas.
  78. */
  79. val selectedMangas = mutableSetOf<Manga>()
  80. private var selectedCoverManga: Manga? = null
  81. /**
  82. * Relay to notify the UI of selection updates.
  83. */
  84. val selectionRelay: PublishRelay<LibrarySelectionEvent> = PublishRelay.create()
  85. /**
  86. * Relay to notify search query changes.
  87. */
  88. val searchRelay: BehaviorRelay<String> = BehaviorRelay.create()
  89. /**
  90. * Relay to notify the library's viewpager for updates.
  91. */
  92. val libraryMangaRelay: BehaviorRelay<LibraryMangaEvent> = BehaviorRelay.create()
  93. /**
  94. * Relay to notify the library's viewpager to select all manga
  95. */
  96. val selectAllRelay: PublishRelay<Int> = PublishRelay.create()
  97. /**
  98. * Number of manga per row in grid mode.
  99. */
  100. var mangaPerRow = 0
  101. private set
  102. /**
  103. * Adapter of the view pager.
  104. */
  105. private var adapter: LibraryAdapter? = null
  106. /**
  107. * Navigation view containing filter/sort/display items.
  108. */
  109. private var navView: LibraryNavigationView? = null
  110. /**
  111. * Drawer listener to allow swipe only for closing the drawer.
  112. */
  113. private var drawerListener: DrawerLayout.DrawerListener? = null
  114. private var tabsVisibilityRelay: BehaviorRelay<Boolean> = BehaviorRelay.create(false)
  115. private var tabsVisibilitySubscription: Subscription? = null
  116. private var searchViewSubscription: Subscription? = null
  117. init {
  118. setHasOptionsMenu(true)
  119. retainViewMode = RetainViewMode.RETAIN_DETACH
  120. }
  121. override fun getTitle(): String? {
  122. return resources?.getString(R.string.label_library)
  123. }
  124. override fun createPresenter(): LibraryPresenter {
  125. return LibraryPresenter()
  126. }
  127. override fun inflateView(inflater: LayoutInflater, container: ViewGroup): View {
  128. return inflater.inflate(R.layout.library_controller, container, false)
  129. }
  130. override fun onViewCreated(view: View) {
  131. super.onViewCreated(view)
  132. adapter = LibraryAdapter(this)
  133. library_pager.adapter = adapter
  134. library_pager.pageSelections().skip(1).subscribeUntilDestroy {
  135. preferences.lastUsedCategory().set(it)
  136. activeCategory = it
  137. }
  138. getColumnsPreferenceForCurrentOrientation().asObservable()
  139. .doOnNext { mangaPerRow = it }
  140. .skip(1)
  141. // Set again the adapter to recalculate the covers height
  142. .subscribeUntilDestroy { reattachAdapter() }
  143. if (selectedMangas.isNotEmpty()) {
  144. createActionModeIfNeeded()
  145. }
  146. }
  147. override fun onChangeStarted(handler: ControllerChangeHandler, type: ControllerChangeType) {
  148. super.onChangeStarted(handler, type)
  149. if (type.isEnter) {
  150. activity?.tabs?.setupWithViewPager(library_pager)
  151. presenter.subscribeLibrary()
  152. }
  153. }
  154. override fun onDestroyView(view: View) {
  155. destroyActionModeIfNeeded()
  156. action_toolbar.destroy()
  157. adapter?.onDestroy()
  158. adapter = null
  159. tabsVisibilitySubscription?.unsubscribe()
  160. tabsVisibilitySubscription = null
  161. super.onDestroyView(view)
  162. }
  163. override fun createSecondaryDrawer(drawer: DrawerLayout): ViewGroup {
  164. val view = drawer.inflate(R.layout.library_drawer) as LibraryNavigationView
  165. navView = view
  166. drawer.setDrawerLockMode(DrawerLayout.LOCK_MODE_UNLOCKED, GravityCompat.END)
  167. navView?.onGroupClicked = { group ->
  168. when (group) {
  169. is LibraryNavigationView.FilterGroup -> onFilterChanged()
  170. is LibraryNavigationView.SortGroup -> onSortChanged()
  171. }
  172. }
  173. return view
  174. }
  175. override fun cleanupSecondaryDrawer(drawer: DrawerLayout) {
  176. navView = null
  177. }
  178. override fun configureTabs(tabs: TabLayout) {
  179. with(tabs) {
  180. tabGravity = TabLayout.GRAVITY_CENTER
  181. tabMode = TabLayout.MODE_SCROLLABLE
  182. }
  183. tabsVisibilitySubscription?.unsubscribe()
  184. tabsVisibilitySubscription = tabsVisibilityRelay.subscribe { visible ->
  185. val tabAnimator = (activity as? MainActivity)?.tabAnimator
  186. if (visible) {
  187. tabAnimator?.expand()
  188. } else {
  189. tabAnimator?.collapse()
  190. }
  191. }
  192. }
  193. override fun cleanupTabs(tabs: TabLayout) {
  194. tabsVisibilitySubscription?.unsubscribe()
  195. tabsVisibilitySubscription = null
  196. }
  197. fun onNextLibraryUpdate(categories: List<Category>, mangaMap: Map<Int, List<LibraryItem>>) {
  198. val view = view ?: return
  199. val adapter = adapter ?: return
  200. // Show empty view if needed
  201. if (mangaMap.isNotEmpty()) {
  202. empty_view.hide()
  203. } else {
  204. empty_view.show(R.drawable.ic_book_black_128dp, R.string.information_empty_library)
  205. }
  206. // Get the current active category.
  207. val activeCat = if (adapter.categories.isNotEmpty())
  208. library_pager.currentItem
  209. else
  210. activeCategory
  211. // Set the categories
  212. adapter.categories = categories
  213. // Restore active category.
  214. library_pager.setCurrentItem(activeCat, false)
  215. tabsVisibilityRelay.call(categories.size > 1)
  216. // Delay the scroll position to allow the view to be properly measured.
  217. view.post {
  218. if (isAttached) {
  219. activity?.tabs?.setScrollPosition(library_pager.currentItem, 0f, true)
  220. }
  221. }
  222. // Send the manga map to child fragments after the adapter is updated.
  223. libraryMangaRelay.call(LibraryMangaEvent(mangaMap))
  224. }
  225. /**
  226. * Returns a preference for the number of manga per row based on the current orientation.
  227. *
  228. * @return the preference.
  229. */
  230. private fun getColumnsPreferenceForCurrentOrientation(): Preference<Int> {
  231. return if (resources?.configuration?.orientation == Configuration.ORIENTATION_PORTRAIT)
  232. preferences.portraitColumns()
  233. else
  234. preferences.landscapeColumns()
  235. }
  236. /**
  237. * Called when a filter is changed.
  238. */
  239. private fun onFilterChanged() {
  240. presenter.requestFilterUpdate()
  241. activity?.invalidateOptionsMenu()
  242. }
  243. private fun onDownloadBadgeChanged() {
  244. presenter.requestDownloadBadgesUpdate()
  245. }
  246. /**
  247. * Called when the sorting mode is changed.
  248. */
  249. private fun onSortChanged() {
  250. presenter.requestSortUpdate()
  251. }
  252. /**
  253. * Reattaches the adapter to the view pager to recreate fragments
  254. */
  255. private fun reattachAdapter() {
  256. val adapter = adapter ?: return
  257. val position = library_pager.currentItem
  258. adapter.recycle = false
  259. library_pager.adapter = adapter
  260. library_pager.currentItem = position
  261. adapter.recycle = true
  262. }
  263. /**
  264. * Creates the action mode if it's not created already.
  265. */
  266. fun createActionModeIfNeeded() {
  267. if (actionMode == null) {
  268. actionMode = (activity as AppCompatActivity).startSupportActionMode(this)
  269. action_toolbar.show(
  270. actionMode!!,
  271. R.menu.library_selection
  272. ) { onActionItemClicked(actionMode!!, it!!) }
  273. }
  274. }
  275. /**
  276. * Destroys the action mode.
  277. */
  278. private fun destroyActionModeIfNeeded() {
  279. actionMode?.finish()
  280. }
  281. override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
  282. inflater.inflate(R.menu.library, menu)
  283. val searchItem = menu.findItem(R.id.action_search)
  284. val searchView = searchItem.actionView as SearchView
  285. if (query.isNotEmpty()) {
  286. searchItem.expandActionView()
  287. searchView.setQuery(query, true)
  288. searchView.clearFocus()
  289. }
  290. // Mutate the filter icon because it needs to be tinted and the resource is shared.
  291. menu.findItem(R.id.action_filter).icon.mutate()
  292. searchViewSubscription?.unsubscribe()
  293. searchViewSubscription = searchView.queryTextChanges()
  294. // Ignore events if this controller isn't at the top
  295. .filter { router.backstack.lastOrNull()?.controller() == this }
  296. .subscribeUntilDestroy {
  297. query = it.toString()
  298. searchRelay.call(query)
  299. }
  300. searchItem.fixExpand(onExpand = { invalidateMenuOnExpand() })
  301. }
  302. fun search(query: String) {
  303. this.query = query
  304. }
  305. override fun onPrepareOptionsMenu(menu: Menu) {
  306. val navView = navView ?: return
  307. val filterItem = menu.findItem(R.id.action_filter)
  308. // Tint icon if there's a filter active
  309. val filterColor = if (navView.hasActiveFilters()) Color.rgb(255, 238, 7) else Color.WHITE
  310. DrawableCompat.setTint(filterItem.icon, filterColor)
  311. // Display submenu
  312. if (preferences.libraryAsList().getOrDefault()) {
  313. menu.findItem(R.id.action_display_list).isChecked = true
  314. } else {
  315. menu.findItem(R.id.action_display_grid).isChecked = true
  316. }
  317. if (preferences.downloadBadge().getOrDefault()) {
  318. menu.findItem(R.id.action_display_download_badge).isChecked = true
  319. }
  320. }
  321. override fun onOptionsItemSelected(item: MenuItem): Boolean {
  322. when (item.itemId) {
  323. R.id.action_search -> expandActionViewFromInteraction = true
  324. R.id.action_filter -> {
  325. navView?.let { activity?.drawer?.openDrawer(GravityCompat.END) }
  326. }
  327. R.id.action_update_library -> {
  328. activity?.let { LibraryUpdateService.start(it) }
  329. }
  330. // Display submenu
  331. R.id.action_display_grid -> {
  332. item.isChecked = true
  333. preferences.libraryAsList().set(false)
  334. reattachAdapter()
  335. }
  336. R.id.action_display_list -> {
  337. item.isChecked = true
  338. preferences.libraryAsList().set(true)
  339. reattachAdapter()
  340. }
  341. R.id.action_display_download_badge -> {
  342. item.isChecked = !item.isChecked
  343. preferences.downloadBadge().set(item.isChecked)
  344. onDownloadBadgeChanged()
  345. }
  346. }
  347. return super.onOptionsItemSelected(item)
  348. }
  349. /**
  350. * Invalidates the action mode, forcing it to refresh its content.
  351. */
  352. fun invalidateActionMode() {
  353. actionMode?.invalidate()
  354. }
  355. override fun onCreateActionMode(mode: ActionMode, menu: Menu): Boolean {
  356. mode.menuInflater.inflate(R.menu.generic_selection, menu)
  357. return true
  358. }
  359. override fun onPrepareActionMode(mode: ActionMode, menu: Menu): Boolean {
  360. val count = selectedMangas.size
  361. if (count == 0) {
  362. // Destroy action mode if there are no items selected.
  363. destroyActionModeIfNeeded()
  364. } else {
  365. mode.title = count.toString()
  366. action_toolbar.findItem(R.id.action_edit_cover)?.isVisible = count == 1
  367. }
  368. return false
  369. }
  370. override fun onActionItemClicked(mode: ActionMode, item: MenuItem): Boolean {
  371. when (item.itemId) {
  372. R.id.action_edit_cover -> {
  373. changeSelectedCover()
  374. destroyActionModeIfNeeded()
  375. }
  376. R.id.action_move_to_category -> showChangeMangaCategoriesDialog()
  377. R.id.action_delete -> showDeleteMangaDialog()
  378. R.id.action_select_all -> selectAllCategoryManga()
  379. else -> return false
  380. }
  381. return true
  382. }
  383. override fun onDestroyActionMode(mode: ActionMode?) {
  384. action_toolbar.hide()
  385. // Clear all the manga selections and notify child views.
  386. selectedMangas.clear()
  387. selectionRelay.call(LibrarySelectionEvent.Cleared())
  388. actionMode = null
  389. }
  390. fun openManga(manga: Manga) {
  391. // Notify the presenter a manga is being opened.
  392. presenter.onOpenManga()
  393. router.pushController(MangaController(manga).withFadeTransaction())
  394. }
  395. /**
  396. * Sets the selection for a given manga.
  397. *
  398. * @param manga the manga whose selection has changed.
  399. * @param selected whether it's now selected or not.
  400. */
  401. fun setSelection(manga: Manga, selected: Boolean) {
  402. if (selected) {
  403. if (selectedMangas.add(manga)) {
  404. selectionRelay.call(LibrarySelectionEvent.Selected(manga))
  405. }
  406. } else {
  407. if (selectedMangas.remove(manga)) {
  408. selectionRelay.call(LibrarySelectionEvent.Unselected(manga))
  409. }
  410. }
  411. }
  412. /**
  413. * Move the selected manga to a list of categories.
  414. */
  415. private fun showChangeMangaCategoriesDialog() {
  416. // Create a copy of selected manga
  417. val mangas = selectedMangas.toList()
  418. // Hide the default category because it has a different behavior than the ones from db.
  419. val categories = presenter.categories.filter { it.id != 0 }
  420. // Get indexes of the common categories to preselect.
  421. val commonCategoriesIndexes = presenter.getCommonCategories(mangas)
  422. .map { categories.indexOf(it) }
  423. .toTypedArray()
  424. ChangeMangaCategoriesDialog(this, mangas, categories, commonCategoriesIndexes)
  425. .showDialog(router)
  426. }
  427. private fun showDeleteMangaDialog() {
  428. DeleteLibraryMangasDialog(this, selectedMangas.toList()).showDialog(router)
  429. }
  430. override fun updateCategoriesForMangas(mangas: List<Manga>, categories: List<Category>) {
  431. presenter.moveMangasToCategories(categories, mangas)
  432. destroyActionModeIfNeeded()
  433. }
  434. override fun deleteMangasFromLibrary(mangas: List<Manga>, deleteChapters: Boolean) {
  435. presenter.removeMangaFromLibrary(mangas, deleteChapters)
  436. destroyActionModeIfNeeded()
  437. }
  438. /**
  439. * Changes the cover for the selected manga.
  440. */
  441. private fun changeSelectedCover() {
  442. val manga = selectedMangas.firstOrNull() ?: return
  443. selectedCoverManga = manga
  444. if (manga.favorite) {
  445. val intent = Intent(Intent.ACTION_GET_CONTENT)
  446. intent.type = "image/*"
  447. startActivityForResult(Intent.createChooser(intent,
  448. resources?.getString(R.string.file_select_cover)), REQUEST_IMAGE_OPEN)
  449. } else {
  450. activity?.toast(R.string.notification_first_add_to_library)
  451. }
  452. }
  453. private fun selectAllCategoryManga() {
  454. adapter?.categories?.getOrNull(library_pager.currentItem)?.id?.let {
  455. selectAllRelay.call(it)
  456. }
  457. }
  458. override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
  459. if (requestCode == REQUEST_IMAGE_OPEN) {
  460. if (data == null || resultCode != Activity.RESULT_OK) return
  461. val activity = activity ?: return
  462. val manga = selectedCoverManga ?: return
  463. try {
  464. // Get the file's input stream from the incoming Intent
  465. activity.contentResolver.openInputStream(data.data ?: Uri.EMPTY).use {
  466. // Update cover to selected file, show error if something went wrong
  467. if (it != null && presenter.editCoverWithStream(it, manga)) {
  468. // TODO refresh cover
  469. } else {
  470. activity.toast(R.string.notification_cover_update_failed)
  471. }
  472. }
  473. } catch (error: IOException) {
  474. activity.toast(R.string.notification_cover_update_failed)
  475. Timber.e(error)
  476. }
  477. selectedCoverManga = null
  478. }
  479. }
  480. private companion object {
  481. /**
  482. * Key to change the cover of a manga in [onActivityResult].
  483. */
  484. const val REQUEST_IMAGE_OPEN = 101
  485. }
  486. }