LibraryFragment.kt 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492
  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.os.Bundle
  6. import android.support.design.widget.TabLayout
  7. import android.support.v4.view.ViewPager
  8. import android.support.v7.view.ActionMode
  9. import android.support.v7.widget.RecyclerView
  10. import android.support.v7.widget.SearchView
  11. import android.view.*
  12. import com.afollestad.materialdialogs.MaterialDialog
  13. import com.f2prateek.rx.preferences.Preference
  14. import eu.davidea.flexibleadapter.FlexibleAdapter
  15. import eu.kanade.tachiyomi.R
  16. import eu.kanade.tachiyomi.data.database.models.Category
  17. import eu.kanade.tachiyomi.data.database.models.Manga
  18. import eu.kanade.tachiyomi.data.library.LibraryUpdateService
  19. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  20. import eu.kanade.tachiyomi.data.preference.getOrDefault
  21. import eu.kanade.tachiyomi.ui.base.fragment.BaseRxFragment
  22. import eu.kanade.tachiyomi.ui.category.CategoryActivity
  23. import eu.kanade.tachiyomi.ui.main.MainActivity
  24. import eu.kanade.tachiyomi.util.toast
  25. import kotlinx.android.synthetic.main.activity_main.*
  26. import kotlinx.android.synthetic.main.fragment_library.*
  27. import nucleus.factory.RequiresPresenter
  28. import rx.Subscription
  29. import uy.kohesive.injekt.injectLazy
  30. import java.io.IOException
  31. /**
  32. * Fragment that shows the manga from the library.
  33. * Uses R.layout.fragment_library.
  34. */
  35. @RequiresPresenter(LibraryPresenter::class)
  36. class LibraryFragment : BaseRxFragment<LibraryPresenter>(), ActionMode.Callback {
  37. /**
  38. * Adapter containing the categories of the library.
  39. */
  40. lateinit var adapter: LibraryAdapter
  41. private set
  42. /**
  43. * Preferences.
  44. */
  45. val preferences: PreferencesHelper by injectLazy()
  46. /**
  47. * TabLayout of the categories.
  48. */
  49. private val tabs: TabLayout
  50. get() = (activity as MainActivity).tabs
  51. /**
  52. * Position of the active category.
  53. */
  54. private var activeCategory: Int = 0
  55. /**
  56. * Query of the search box.
  57. */
  58. private var query: String? = null
  59. /**
  60. * Action mode for manga selection.
  61. */
  62. var actionMode: ActionMode? = null
  63. private set
  64. /**
  65. * Selected manga for editing its cover.
  66. */
  67. private var selectedCoverManga: Manga? = null
  68. /**
  69. * Status of isFilterDownloaded
  70. */
  71. var isFilterDownloaded = false
  72. /**
  73. * Status of isFilterUnread
  74. */
  75. var isFilterUnread = false
  76. /**
  77. * Number of manga per row in grid mode.
  78. */
  79. var mangaPerRow = 0
  80. private set
  81. /**
  82. * A pool to share view holders between all the registered categories (fragments).
  83. */
  84. var pool = RecyclerView.RecycledViewPool()
  85. private var numColumnsSubscription: Subscription? = null
  86. companion object {
  87. /**
  88. * Key to change the cover of a manga in [onActivityResult].
  89. */
  90. const val REQUEST_IMAGE_OPEN = 101
  91. /**
  92. * Key to save and restore [query] from a [Bundle].
  93. */
  94. const val QUERY_KEY = "query_key"
  95. /**
  96. * Key to save and restore [activeCategory] from a [Bundle].
  97. */
  98. const val CATEGORY_KEY = "category_key"
  99. /**
  100. * Creates a new instance of this fragment.
  101. *
  102. * @return a new instance of [LibraryFragment].
  103. */
  104. fun newInstance(): LibraryFragment {
  105. return LibraryFragment()
  106. }
  107. }
  108. override fun onCreate(savedState: Bundle?) {
  109. super.onCreate(savedState)
  110. setHasOptionsMenu(true)
  111. isFilterDownloaded = presenter.preferences.filterDownloaded().get() as Boolean
  112. isFilterUnread = presenter.preferences.filterUnread().get() as Boolean
  113. }
  114. override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedState: Bundle?): View? {
  115. return inflater.inflate(R.layout.fragment_library, container, false)
  116. }
  117. override fun onViewCreated(view: View, savedState: Bundle?) {
  118. setToolbarTitle(getString(R.string.label_library))
  119. adapter = LibraryAdapter(childFragmentManager)
  120. view_pager.adapter = adapter
  121. view_pager.addOnPageChangeListener(object : ViewPager.SimpleOnPageChangeListener() {
  122. override fun onPageSelected(position: Int) {
  123. presenter.preferences.lastUsedCategory().set(position)
  124. }
  125. })
  126. tabs.setupWithViewPager(view_pager)
  127. if (savedState != null) {
  128. activeCategory = savedState.getInt(CATEGORY_KEY)
  129. query = savedState.getString(QUERY_KEY)
  130. presenter.searchSubject.onNext(query)
  131. } else {
  132. activeCategory = presenter.preferences.lastUsedCategory().getOrDefault()
  133. }
  134. numColumnsSubscription = getColumnsPreferenceForCurrentOrientation().asObservable()
  135. .doOnNext { mangaPerRow = it }
  136. .skip(1)
  137. // Set again the adapter to recalculate the covers height
  138. .subscribe { reattachAdapter() }
  139. }
  140. override fun onResume() {
  141. super.onResume()
  142. presenter.subscribeLibrary()
  143. }
  144. override fun onDestroyView() {
  145. numColumnsSubscription?.unsubscribe()
  146. tabs.setupWithViewPager(null)
  147. tabs.visibility = View.GONE
  148. super.onDestroyView()
  149. }
  150. override fun onSaveInstanceState(outState: Bundle) {
  151. outState.putInt(CATEGORY_KEY, view_pager.currentItem)
  152. outState.putString(QUERY_KEY, query)
  153. super.onSaveInstanceState(outState)
  154. }
  155. override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
  156. inflater.inflate(R.menu.library, menu)
  157. // Initialize search menu
  158. val filterDownloadedItem = menu.findItem(R.id.action_filter_downloaded)
  159. val filterUnreadItem = menu.findItem(R.id.action_filter_unread)
  160. val searchItem = menu.findItem(R.id.action_search)
  161. val searchView = searchItem.actionView as SearchView
  162. if (!query.isNullOrEmpty()) {
  163. searchItem.expandActionView()
  164. searchView.setQuery(query, true)
  165. searchView.clearFocus()
  166. }
  167. filterDownloadedItem.isChecked = isFilterDownloaded
  168. filterUnreadItem.isChecked = isFilterUnread
  169. searchView.setOnQueryTextListener(object : SearchView.OnQueryTextListener {
  170. override fun onQueryTextSubmit(query: String): Boolean {
  171. onSearchTextChange(query)
  172. return true
  173. }
  174. override fun onQueryTextChange(newText: String): Boolean {
  175. onSearchTextChange(newText)
  176. return true
  177. }
  178. })
  179. }
  180. override fun onOptionsItemSelected(item: MenuItem): Boolean {
  181. when (item.itemId) {
  182. R.id.action_filter_unread -> {
  183. // Change unread filter status.
  184. isFilterUnread = !isFilterUnread
  185. // Update settings.
  186. presenter.preferences.filterUnread().set(isFilterUnread)
  187. // Apply filter.
  188. onFilterCheckboxChanged()
  189. }
  190. R.id.action_filter_downloaded -> {
  191. // Change downloaded filter status.
  192. isFilterDownloaded = !isFilterDownloaded
  193. // Update settings.
  194. presenter.preferences.filterDownloaded().set(isFilterDownloaded)
  195. // Apply filter.
  196. onFilterCheckboxChanged()
  197. }
  198. R.id.action_filter_empty -> {
  199. // Remove filter status.
  200. isFilterUnread = false
  201. isFilterDownloaded = false
  202. // Update settings.
  203. presenter.preferences.filterUnread().set(isFilterUnread)
  204. presenter.preferences.filterDownloaded().set(isFilterDownloaded)
  205. // Apply filter
  206. onFilterCheckboxChanged()
  207. }
  208. R.id.action_library_display_mode -> swapDisplayMode()
  209. R.id.action_update_library -> {
  210. LibraryUpdateService.start(activity, true)
  211. }
  212. R.id.action_edit_categories -> {
  213. val intent = CategoryActivity.newIntent(activity)
  214. startActivity(intent)
  215. }
  216. else -> return super.onOptionsItemSelected(item)
  217. }
  218. return true
  219. }
  220. /**
  221. * Applies filter change
  222. */
  223. private fun onFilterCheckboxChanged() {
  224. presenter.updateLibrary()
  225. adapter.refreshRegisteredAdapters()
  226. activity.supportInvalidateOptionsMenu()
  227. }
  228. /**
  229. * Swap display mode
  230. */
  231. private fun swapDisplayMode() {
  232. presenter.swapDisplayMode()
  233. reattachAdapter()
  234. }
  235. /**
  236. * Reattaches the adapter to the view pager to recreate fragments
  237. */
  238. private fun reattachAdapter() {
  239. pool.clear()
  240. pool = RecyclerView.RecycledViewPool()
  241. val position = view_pager.currentItem
  242. view_pager.adapter = adapter
  243. view_pager.currentItem = position
  244. }
  245. /**
  246. * Returns a preference for the number of manga per row based on the current orientation.
  247. *
  248. * @return the preference.
  249. */
  250. private fun getColumnsPreferenceForCurrentOrientation(): Preference<Int> {
  251. return if (resources.configuration.orientation == Configuration.ORIENTATION_PORTRAIT)
  252. preferences.portraitColumns()
  253. else
  254. preferences.landscapeColumns()
  255. }
  256. /**
  257. * Updates the query.
  258. *
  259. * @param query the new value of the query.
  260. */
  261. private fun onSearchTextChange(query: String?) {
  262. this.query = query
  263. // Notify the subject the query has changed.
  264. if (isResumed) {
  265. presenter.searchSubject.onNext(query)
  266. }
  267. }
  268. /**
  269. * Called when the library is updated. It sets the new data and updates the view.
  270. *
  271. * @param categories the categories of the library.
  272. * @param mangaMap a map containing the manga for each category.
  273. */
  274. fun onNextLibraryUpdate(categories: List<Category>, mangaMap: Map<Int, List<Manga>>) {
  275. // Check if library is empty and update information accordingly.
  276. (activity as MainActivity).updateEmptyView(mangaMap.isEmpty(),
  277. R.string.information_empty_library, R.drawable.ic_book_black_128dp)
  278. // Get the current active category.
  279. val activeCat = if (adapter.categories != null) view_pager.currentItem else activeCategory
  280. // Set the categories
  281. adapter.categories = categories
  282. tabs.visibility = if (categories.size <= 1) View.GONE else View.VISIBLE
  283. // Restore active category.
  284. view_pager.setCurrentItem(activeCat, false)
  285. // Delay the scroll position to allow the view to be properly measured.
  286. view_pager.post { if (isAdded) tabs.setScrollPosition(view_pager.currentItem, 0f, true) }
  287. // Send the manga map to child fragments after the adapter is updated.
  288. presenter.libraryMangaSubject.onNext(LibraryMangaEvent(mangaMap))
  289. }
  290. /**
  291. * Sets the title of the action mode.
  292. *
  293. * @param count the number of items selected.
  294. */
  295. fun setContextTitle(count: Int) {
  296. actionMode?.title = getString(R.string.label_selected, count)
  297. }
  298. /**
  299. * Sets the visibility of the edit cover item.
  300. *
  301. * @param count the number of items selected.
  302. */
  303. fun setVisibilityOfCoverEdit(count: Int) {
  304. // If count = 1 display edit button
  305. actionMode?.menu?.findItem(R.id.action_edit_cover)?.isVisible = count == 1
  306. }
  307. override fun onCreateActionMode(mode: ActionMode, menu: Menu): Boolean {
  308. mode.menuInflater.inflate(R.menu.library_selection, menu)
  309. adapter.setSelectionMode(FlexibleAdapter.MODE_MULTI)
  310. return true
  311. }
  312. override fun onPrepareActionMode(mode: ActionMode, menu: Menu): Boolean {
  313. return false
  314. }
  315. override fun onActionItemClicked(mode: ActionMode, item: MenuItem): Boolean {
  316. when (item.itemId) {
  317. R.id.action_edit_cover -> {
  318. changeSelectedCover(presenter.selectedMangas)
  319. destroyActionModeIfNeeded()
  320. }
  321. R.id.action_move_to_category -> moveMangasToCategories(presenter.selectedMangas)
  322. R.id.action_delete -> showDeleteMangaDialog()
  323. else -> return false
  324. }
  325. return true
  326. }
  327. override fun onDestroyActionMode(mode: ActionMode) {
  328. adapter.setSelectionMode(FlexibleAdapter.MODE_SINGLE)
  329. presenter.selectedMangas.clear()
  330. actionMode = null
  331. }
  332. /**
  333. * Destroys the action mode.
  334. */
  335. fun destroyActionModeIfNeeded() {
  336. actionMode?.finish()
  337. }
  338. /**
  339. * Changes the cover for the selected manga.
  340. *
  341. * @param mangas a list of selected manga.
  342. */
  343. private fun changeSelectedCover(mangas: List<Manga>) {
  344. if (mangas.size == 1) {
  345. selectedCoverManga = mangas[0]
  346. if (selectedCoverManga?.favorite ?: false) {
  347. val intent = Intent(Intent.ACTION_GET_CONTENT)
  348. intent.type = "image/*"
  349. startActivityForResult(Intent.createChooser(intent,
  350. getString(R.string.file_select_cover)), REQUEST_IMAGE_OPEN)
  351. } else {
  352. context.toast(R.string.notification_first_add_to_library)
  353. }
  354. }
  355. }
  356. override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
  357. if (data != null && resultCode == Activity.RESULT_OK && requestCode == REQUEST_IMAGE_OPEN) {
  358. selectedCoverManga?.let { manga ->
  359. try {
  360. // Get the file's input stream from the incoming Intent
  361. context.contentResolver.openInputStream(data.data).use {
  362. // Update cover to selected file, show error if something went wrong
  363. if (presenter.editCoverWithStream(it, manga)) {
  364. adapter.refreshRegisteredAdapters()
  365. } else {
  366. context.toast(R.string.notification_manga_update_failed)
  367. }
  368. }
  369. } catch (e: IOException) {
  370. context.toast(R.string.notification_manga_update_failed)
  371. e.printStackTrace()
  372. }
  373. }
  374. }
  375. }
  376. /**
  377. * Move the selected manga to a list of categories.
  378. *
  379. * @param mangas the manga list to move.
  380. */
  381. private fun moveMangasToCategories(mangas: List<Manga>) {
  382. // Hide the default category because it has a different behavior than the ones from db.
  383. val categories = presenter.categories.filter { it.id != 0 }
  384. // Get indexes of the common categories to preselect.
  385. val commonCategoriesIndexes = presenter.getCommonCategories(mangas)
  386. .map { categories.indexOf(it) }
  387. .toTypedArray()
  388. MaterialDialog.Builder(activity)
  389. .title(R.string.action_move_category)
  390. .items(categories.map { it.name })
  391. .itemsCallbackMultiChoice(commonCategoriesIndexes) { dialog, positions, text ->
  392. val selectedCategories = positions.map { categories[it] }
  393. presenter.moveMangasToCategories(selectedCategories, mangas)
  394. destroyActionModeIfNeeded()
  395. true
  396. }
  397. .positiveText(android.R.string.ok)
  398. .negativeText(android.R.string.cancel)
  399. .show()
  400. }
  401. private fun showDeleteMangaDialog() {
  402. MaterialDialog.Builder(activity)
  403. .content(R.string.confirm_delete_manga)
  404. .positiveText(android.R.string.yes)
  405. .negativeText(android.R.string.no)
  406. .onPositive { dialog, action ->
  407. presenter.removeMangaFromLibrary()
  408. destroyActionModeIfNeeded()
  409. }
  410. .show()
  411. }
  412. /**
  413. * Creates the action mode if it's not created already.
  414. */
  415. fun createActionModeIfNeeded() {
  416. if (actionMode == null) {
  417. actionMode = activity.startSupportActionMode(this)
  418. }
  419. }
  420. /**
  421. * Invalidates the action mode, forcing it to refresh its content.
  422. */
  423. fun invalidateActionMode() {
  424. actionMode?.invalidate()
  425. }
  426. }