LibraryFragment.kt 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. package eu.kanade.tachiyomi.ui.library
  2. import android.app.Activity
  3. import android.content.Intent
  4. import android.os.Bundle
  5. import android.support.design.widget.AppBarLayout
  6. import android.support.design.widget.TabLayout
  7. import android.support.v7.view.ActionMode
  8. import android.support.v7.widget.SearchView
  9. import android.view.*
  10. import com.afollestad.materialdialogs.MaterialDialog
  11. import eu.davidea.flexibleadapter.FlexibleAdapter
  12. import eu.kanade.tachiyomi.R
  13. import eu.kanade.tachiyomi.data.database.models.Category
  14. import eu.kanade.tachiyomi.data.database.models.Manga
  15. import eu.kanade.tachiyomi.data.library.LibraryUpdateService
  16. import eu.kanade.tachiyomi.event.LibraryMangaEvent
  17. import eu.kanade.tachiyomi.ui.base.fragment.BaseRxFragment
  18. import eu.kanade.tachiyomi.ui.category.CategoryActivity
  19. import eu.kanade.tachiyomi.ui.main.MainActivity
  20. import eu.kanade.tachiyomi.util.inflate
  21. import eu.kanade.tachiyomi.util.toast
  22. import kotlinx.android.synthetic.main.activity_main.*
  23. import kotlinx.android.synthetic.main.fragment_library.*
  24. import nucleus.factory.RequiresPresenter
  25. import java.io.IOException
  26. /**
  27. * Fragment that shows the manga from the library.
  28. * Uses R.layout.fragment_library.
  29. */
  30. @RequiresPresenter(LibraryPresenter::class)
  31. class LibraryFragment : BaseRxFragment<LibraryPresenter>(), ActionMode.Callback {
  32. /**
  33. * Adapter containing the categories of the library.
  34. */
  35. lateinit var adapter: LibraryAdapter
  36. private set
  37. /**
  38. * TabLayout of the categories.
  39. */
  40. private lateinit var tabs: TabLayout
  41. /**
  42. * AppBarLayout from [MainActivity].
  43. */
  44. private val appbar: AppBarLayout
  45. get() = (activity as MainActivity).appbar
  46. /**
  47. * Position of the active category.
  48. */
  49. private var activeCategory: Int = 0
  50. /**
  51. * Query of the search box.
  52. */
  53. private var query: String? = null
  54. /**
  55. * Action mode for manga selection.
  56. */
  57. var actionMode: ActionMode? = null
  58. private set
  59. /**
  60. * Selected manga for editing its cover.
  61. */
  62. private var selectedCoverManga: Manga? = null
  63. /**
  64. * Status of isFilterDownloaded
  65. */
  66. var isFilterDownloaded = false
  67. /**
  68. * Status of isFilterUnread
  69. */
  70. var isFilterUnread = false
  71. companion object {
  72. /**
  73. * Key to change the cover of a manga in [onActivityResult].
  74. */
  75. const val REQUEST_IMAGE_OPEN = 101
  76. /**
  77. * Key to save and restore [query] from a [Bundle].
  78. */
  79. const val QUERY_KEY = "query_key"
  80. /**
  81. * Key to save and restore [activeCategory] from a [Bundle].
  82. */
  83. const val CATEGORY_KEY = "category_key"
  84. /**
  85. * Creates a new instance of this fragment.
  86. *
  87. * @return a new instance of [LibraryFragment].
  88. */
  89. @JvmStatic
  90. fun newInstance(): LibraryFragment {
  91. return LibraryFragment()
  92. }
  93. }
  94. override fun onCreate(savedState: Bundle?) {
  95. super.onCreate(savedState)
  96. setHasOptionsMenu(true)
  97. isFilterDownloaded = presenter.preferences.filterDownloaded().get() as Boolean
  98. isFilterUnread = presenter.preferences.filterUnread().get() as Boolean
  99. }
  100. override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedState: Bundle?): View? {
  101. return inflater.inflate(R.layout.fragment_library, container, false)
  102. }
  103. override fun onViewCreated(view: View, savedState: Bundle?) {
  104. setToolbarTitle(getString(R.string.label_library))
  105. tabs = appbar.inflate(R.layout.library_tab_layout) as TabLayout
  106. appbar.addView(tabs)
  107. adapter = LibraryAdapter(childFragmentManager)
  108. view_pager.adapter = adapter
  109. tabs.setupWithViewPager(view_pager)
  110. if (savedState != null) {
  111. activeCategory = savedState.getInt(CATEGORY_KEY)
  112. query = savedState.getString(QUERY_KEY)
  113. presenter.searchSubject.onNext(query)
  114. }
  115. }
  116. override fun onDestroyView() {
  117. appbar.removeView(tabs)
  118. super.onDestroyView()
  119. }
  120. override fun onSaveInstanceState(outState: Bundle) {
  121. outState.putInt(CATEGORY_KEY, view_pager.currentItem)
  122. outState.putString(QUERY_KEY, query)
  123. super.onSaveInstanceState(outState)
  124. }
  125. override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
  126. inflater.inflate(R.menu.library, menu)
  127. // Initialize search menu
  128. val filterDownloadedItem = menu.findItem(R.id.action_filter_downloaded)
  129. val filterUnreadItem = menu.findItem(R.id.action_filter_unread)
  130. val searchItem = menu.findItem(R.id.action_search)
  131. val searchView = searchItem.actionView as SearchView
  132. if (!query.isNullOrEmpty()) {
  133. searchItem.expandActionView()
  134. searchView.setQuery(query, true)
  135. searchView.clearFocus()
  136. }
  137. filterDownloadedItem.isChecked = isFilterDownloaded;
  138. filterUnreadItem.isChecked = isFilterUnread;
  139. searchView.setOnQueryTextListener(object : SearchView.OnQueryTextListener {
  140. override fun onQueryTextSubmit(query: String): Boolean {
  141. onSearchTextChange(query)
  142. return true
  143. }
  144. override fun onQueryTextChange(newText: String): Boolean {
  145. onSearchTextChange(newText)
  146. return true
  147. }
  148. })
  149. }
  150. override fun onOptionsItemSelected(item: MenuItem): Boolean {
  151. when (item.itemId) {
  152. R.id.action_filter_unread -> {
  153. // Change unread filter status.
  154. isFilterUnread = !isFilterUnread
  155. // Update settings.
  156. presenter.preferences.filterUnread().set(isFilterUnread)
  157. // Apply filter.
  158. onFilterCheckboxChanged()
  159. }
  160. R.id.action_filter_downloaded -> {
  161. // Change downloaded filter status.
  162. isFilterDownloaded = !isFilterDownloaded
  163. // Update settings.
  164. presenter.preferences.filterDownloaded().set(isFilterDownloaded)
  165. // Apply filter.
  166. onFilterCheckboxChanged()
  167. }
  168. R.id.action_filter_empty -> {
  169. // Remove filter status.
  170. isFilterUnread = false
  171. isFilterDownloaded = false
  172. // Update settings.
  173. presenter.preferences.filterUnread().set(isFilterUnread)
  174. presenter.preferences.filterDownloaded().set(isFilterDownloaded)
  175. // Apply filter
  176. onFilterCheckboxChanged()
  177. }
  178. R.id.action_refresh -> LibraryUpdateService.start(activity)
  179. R.id.action_edit_categories -> {
  180. val intent = CategoryActivity.newIntent(activity)
  181. startActivity(intent)
  182. }
  183. else -> return super.onOptionsItemSelected(item)
  184. }
  185. return true
  186. }
  187. /**
  188. * Applies filter change
  189. */
  190. private fun onFilterCheckboxChanged() {
  191. presenter.updateLibrary()
  192. adapter.notifyDataSetChanged()
  193. adapter.refreshRegisteredAdapters()
  194. activity.supportInvalidateOptionsMenu();
  195. }
  196. /**
  197. * Updates the query.
  198. *
  199. * @param query the new value of the query.
  200. */
  201. private fun onSearchTextChange(query: String?) {
  202. this.query = query
  203. // Notify the subject the query has changed.
  204. presenter.searchSubject.onNext(query)
  205. }
  206. /**
  207. * Called when the library is updated. It sets the new data and updates the view.
  208. *
  209. * @param categories the categories of the library.
  210. * @param mangaMap a map containing the manga for each category.
  211. */
  212. fun onNextLibraryUpdate(categories: List<Category>, mangaMap: Map<Int, List<Manga>>) {
  213. // Check if library is empty and update information accordingly.
  214. (activity as MainActivity).updateEmptyView(mangaMap.isEmpty(),
  215. R.string.information_empty_library, R.drawable.ic_book_black_128dp)
  216. // Get the current active category.
  217. val activeCat = if (adapter.categories != null) view_pager.currentItem else activeCategory
  218. // Add the default category if it contains manga.
  219. if (mangaMap[0] != null) {
  220. setCategories(arrayListOf(Category.createDefault()) + categories)
  221. } else {
  222. setCategories(categories)
  223. }
  224. // Restore active category.
  225. view_pager.setCurrentItem(activeCat, false)
  226. if (tabs.tabCount > 0) {
  227. // Prevent IndexOutOfBoundsException
  228. if (tabs.tabCount <= view_pager.currentItem) {
  229. view_pager.currentItem = (tabs.tabCount - 1)
  230. }
  231. tabs.getTabAt(view_pager.currentItem)?.select()
  232. }
  233. // Send the manga map to child fragments after the adapter is updated.
  234. presenter.libraryMangaSubject.onNext(LibraryMangaEvent(mangaMap))
  235. }
  236. /**
  237. * Sets the categories in the adapter and the tab layout.
  238. *
  239. * @param categories the categories to set.
  240. */
  241. private fun setCategories(categories: List<Category>) {
  242. adapter.categories = categories
  243. tabs.setupWithViewPager(view_pager)
  244. tabs.visibility = if (categories.size <= 1) View.GONE else View.VISIBLE
  245. }
  246. /**
  247. * Sets the title of the action mode.
  248. *
  249. * @param count the number of items selected.
  250. */
  251. fun setContextTitle(count: Int) {
  252. actionMode?.title = getString(R.string.label_selected, count)
  253. }
  254. /**
  255. * Sets the visibility of the edit cover item.
  256. *
  257. * @param count the number of items selected.
  258. */
  259. fun setVisibilityOfCoverEdit(count: Int) {
  260. // If count = 1 display edit button
  261. actionMode?.menu?.findItem(R.id.action_edit_cover)?.isVisible = count == 1
  262. }
  263. override fun onCreateActionMode(mode: ActionMode, menu: Menu): Boolean {
  264. mode.menuInflater.inflate(R.menu.library_selection, menu)
  265. adapter.setSelectionMode(FlexibleAdapter.MODE_MULTI)
  266. return true
  267. }
  268. override fun onPrepareActionMode(mode: ActionMode, menu: Menu): Boolean {
  269. return false
  270. }
  271. override fun onActionItemClicked(mode: ActionMode, item: MenuItem): Boolean {
  272. when (item.itemId) {
  273. R.id.action_edit_cover -> {
  274. changeSelectedCover(presenter.selectedMangas)
  275. destroyActionModeIfNeeded()
  276. }
  277. R.id.action_move_to_category -> {
  278. moveMangasToCategories(presenter.selectedMangas)
  279. }
  280. R.id.action_delete -> {
  281. presenter.deleteMangas()
  282. destroyActionModeIfNeeded()
  283. }
  284. else -> return false
  285. }
  286. return true
  287. }
  288. override fun onDestroyActionMode(mode: ActionMode) {
  289. adapter.setSelectionMode(FlexibleAdapter.MODE_SINGLE)
  290. presenter.selectedMangas.clear()
  291. actionMode = null
  292. }
  293. /**
  294. * Destroys the action mode.
  295. */
  296. fun destroyActionModeIfNeeded() {
  297. actionMode?.finish()
  298. }
  299. /**
  300. * Changes the cover for the selected manga.
  301. *
  302. * @param mangas a list of selected manga.
  303. */
  304. private fun changeSelectedCover(mangas: List<Manga>) {
  305. if (mangas.size == 1) {
  306. selectedCoverManga = mangas[0]
  307. if (selectedCoverManga?.favorite ?: false) {
  308. val intent = Intent(Intent.ACTION_GET_CONTENT)
  309. intent.type = "image/*"
  310. startActivityForResult(Intent.createChooser(intent,
  311. getString(R.string.file_select_cover)), REQUEST_IMAGE_OPEN)
  312. } else {
  313. context.toast(R.string.notification_first_add_to_library)
  314. }
  315. }
  316. }
  317. override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
  318. if (data != null && resultCode == Activity.RESULT_OK && requestCode == REQUEST_IMAGE_OPEN) {
  319. selectedCoverManga?.let { manga ->
  320. try {
  321. // Get the file's input stream from the incoming Intent
  322. context.contentResolver.openInputStream(data.data).use {
  323. // Update cover to selected file, show error if something went wrong
  324. if (presenter.editCoverWithStream(it, manga)) {
  325. adapter.refreshRegisteredAdapters()
  326. } else {
  327. context.toast(R.string.notification_manga_update_failed)
  328. }
  329. }
  330. } catch (e: IOException) {
  331. context.toast(R.string.notification_manga_update_failed)
  332. e.printStackTrace()
  333. }
  334. }
  335. }
  336. }
  337. /**
  338. * Move the selected manga to a list of categories.
  339. *
  340. * @param mangas the manga list to move.
  341. */
  342. private fun moveMangasToCategories(mangas: List<Manga>) {
  343. MaterialDialog.Builder(activity)
  344. .title(R.string.action_move_category)
  345. .items(presenter.getCategoryNames())
  346. .itemsCallbackMultiChoice(null) { dialog, positions, text ->
  347. presenter.moveMangasToCategories(positions, mangas)
  348. destroyActionModeIfNeeded()
  349. true
  350. }
  351. .positiveText(android.R.string.ok)
  352. .negativeText(android.R.string.cancel)
  353. .show()
  354. }
  355. /**
  356. * Creates the action mode if it's not created already.
  357. */
  358. fun createActionModeIfNeeded() {
  359. if (actionMode == null) {
  360. actionMode = baseActivity.startSupportActionMode(this)
  361. }
  362. }
  363. /**
  364. * Invalidates the action mode, forcing it to refresh its content.
  365. */
  366. fun invalidateActionMode() {
  367. actionMode?.invalidate()
  368. }
  369. }