LibraryPresenter.kt 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. package eu.kanade.tachiyomi.ui.library
  2. import android.os.Bundle
  3. import android.util.Pair
  4. import eu.kanade.tachiyomi.data.cache.CoverCache
  5. import eu.kanade.tachiyomi.data.database.DatabaseHelper
  6. import eu.kanade.tachiyomi.data.database.models.Category
  7. import eu.kanade.tachiyomi.data.database.models.Manga
  8. import eu.kanade.tachiyomi.data.database.models.MangaCategory
  9. import eu.kanade.tachiyomi.data.download.DownloadManager
  10. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  11. import eu.kanade.tachiyomi.data.preference.getOrDefault
  12. import eu.kanade.tachiyomi.data.source.SourceManager
  13. import eu.kanade.tachiyomi.ui.base.presenter.BasePresenter
  14. import rx.Observable
  15. import rx.android.schedulers.AndroidSchedulers
  16. import rx.schedulers.Schedulers
  17. import rx.subjects.BehaviorSubject
  18. import java.io.IOException
  19. import java.io.InputStream
  20. import java.util.*
  21. import javax.inject.Inject
  22. /**
  23. * Presenter of [LibraryFragment].
  24. */
  25. class LibraryPresenter : BasePresenter<LibraryFragment>() {
  26. /**
  27. * Categories of the library.
  28. */
  29. lateinit var categories: List<Category>
  30. /**
  31. * Currently selected manga.
  32. */
  33. var selectedMangas = mutableListOf<Manga>()
  34. /**
  35. * Search query of the library.
  36. */
  37. val searchSubject = BehaviorSubject.create<String>()
  38. /**
  39. * Subject to notify the library's viewpager for updates.
  40. */
  41. val libraryMangaSubject = BehaviorSubject.create<LibraryMangaEvent>()
  42. /**
  43. * Database.
  44. */
  45. @Inject lateinit var db: DatabaseHelper
  46. /**
  47. * Preferences.
  48. */
  49. @Inject lateinit var preferences: PreferencesHelper
  50. /**
  51. * Cover cache.
  52. */
  53. @Inject lateinit var coverCache: CoverCache
  54. /**
  55. * Source manager.
  56. */
  57. @Inject lateinit var sourceManager: SourceManager
  58. /**
  59. * Download manager.
  60. */
  61. @Inject lateinit var downloadManager: DownloadManager
  62. companion object {
  63. /**
  64. * Id of the restartable that listens for library updates.
  65. */
  66. const val GET_LIBRARY = 1
  67. }
  68. override fun onCreate(savedState: Bundle?) {
  69. super.onCreate(savedState)
  70. restartableLatestCache(GET_LIBRARY,
  71. { getLibraryObservable() },
  72. { view, pair -> view.onNextLibraryUpdate(pair.first, pair.second) })
  73. if (savedState == null) {
  74. start(GET_LIBRARY)
  75. }
  76. }
  77. /**
  78. * Get the categories and all its manga from the database.
  79. *
  80. * @return an observable of the categories and its manga.
  81. */
  82. fun getLibraryObservable(): Observable<Pair<List<Category>, Map<Int, List<Manga>>>> {
  83. return Observable.combineLatest(getCategoriesObservable(), getLibraryMangasObservable(),
  84. { dbCategories, libraryManga ->
  85. val categories = if (libraryManga.containsKey(0))
  86. arrayListOf(Category.createDefault()) + dbCategories
  87. else
  88. dbCategories
  89. this.categories = categories
  90. Pair(categories, libraryManga)
  91. })
  92. .observeOn(AndroidSchedulers.mainThread())
  93. }
  94. /**
  95. * Get the categories from the database.
  96. *
  97. * @return an observable of the categories.
  98. */
  99. fun getCategoriesObservable(): Observable<List<Category>> {
  100. return db.getCategories().asRxObservable()
  101. }
  102. /**
  103. * Get the manga grouped by categories.
  104. *
  105. * @return an observable containing a map with the category id as key and a list of manga as the
  106. * value.
  107. */
  108. fun getLibraryMangasObservable(): Observable<Map<Int, List<Manga>>> {
  109. return db.getLibraryMangas().asRxObservable()
  110. .flatMap { mangas ->
  111. Observable.from(mangas)
  112. // Filter library by options
  113. .filter { filterManga(it) }
  114. .groupBy { it.category }
  115. .flatMap { group -> group.toList().map { Pair(group.key, it) } }
  116. .toMap({ it.first }, { it.second })
  117. }
  118. }
  119. /**
  120. * Resubscribes to library if needed.
  121. */
  122. fun subscribeLibrary() {
  123. if (isUnsubscribed(GET_LIBRARY)) {
  124. start(GET_LIBRARY)
  125. }
  126. }
  127. /**
  128. * Resubscribes to library.
  129. */
  130. fun updateLibrary() {
  131. start(GET_LIBRARY)
  132. }
  133. /**
  134. * Filters an entry of the library.
  135. *
  136. * @param manga a favorite manga from the database.
  137. * @return true if the entry is included, false otherwise.
  138. */
  139. fun filterManga(manga: Manga): Boolean {
  140. // Filter out manga without source
  141. val source = sourceManager.get(manga.source) ?: return false
  142. val prefFilterDownloaded = preferences.filterDownloaded().getOrDefault()
  143. val prefFilterUnread = preferences.filterUnread().getOrDefault()
  144. // Check if filter option is selected
  145. if (prefFilterDownloaded || prefFilterUnread) {
  146. // Does it have downloaded chapters.
  147. var hasDownloaded = false
  148. var hasUnread = false
  149. if (prefFilterUnread) {
  150. // Does it have unread chapters.
  151. hasUnread = manga.unread > 0
  152. }
  153. if (prefFilterDownloaded) {
  154. val mangaDir = downloadManager.getAbsoluteMangaDirectory(source, manga)
  155. if (mangaDir.exists()) {
  156. for (file in mangaDir.listFiles()) {
  157. if (file.isDirectory && file.listFiles().isNotEmpty()) {
  158. hasDownloaded = true
  159. break
  160. }
  161. }
  162. }
  163. }
  164. // Return correct filter status
  165. if (prefFilterDownloaded && prefFilterUnread) {
  166. return (hasDownloaded && hasUnread)
  167. } else {
  168. return (hasDownloaded || hasUnread)
  169. }
  170. } else {
  171. return true
  172. }
  173. }
  174. /**
  175. * Called when a manga is opened.
  176. */
  177. fun onOpenManga() {
  178. // Avoid further db updates for the library when it's not needed
  179. stop(GET_LIBRARY)
  180. }
  181. /**
  182. * Sets the selection for a given manga.
  183. *
  184. * @param manga the manga whose selection has changed.
  185. * @param selected whether it's now selected or not.
  186. */
  187. fun setSelection(manga: Manga, selected: Boolean) {
  188. if (selected) {
  189. selectedMangas.add(manga)
  190. } else {
  191. selectedMangas.remove(manga)
  192. }
  193. }
  194. /**
  195. * Returns the common categories for the given list of manga.
  196. *
  197. * @param mangas the list of manga.
  198. */
  199. fun getCommonCategories(mangas: List<Manga>) = mangas.toSet()
  200. .map { db.getCategoriesForManga(it).executeAsBlocking() }
  201. .reduce { set1: Iterable<Category>, set2 -> set1.intersect(set2) }
  202. /**
  203. * Remove the selected manga from the library.
  204. */
  205. fun deleteMangas() {
  206. // Create a set of the list
  207. val mangaToDelete = selectedMangas.toSet()
  208. Observable.from(mangaToDelete)
  209. .subscribeOn(Schedulers.io())
  210. .doOnNext {
  211. it.favorite = false
  212. coverCache.deleteFromCache(it.thumbnail_url)
  213. }
  214. .toList()
  215. .flatMap { db.insertMangas(it).asRxObservable() }
  216. .subscribe()
  217. }
  218. /**
  219. * Move the given list of manga to categories.
  220. *
  221. * @param categories the selected categories.
  222. * @param mangas the list of manga to move.
  223. */
  224. fun moveMangasToCategories(categories: List<Category>, mangas: List<Manga>) {
  225. val mc = ArrayList<MangaCategory>()
  226. for (manga in mangas) {
  227. for (cat in categories) {
  228. mc.add(MangaCategory.create(manga, cat))
  229. }
  230. }
  231. db.setMangaCategories(mc, mangas)
  232. }
  233. /**
  234. * Update cover with local file.
  235. *
  236. * @param inputStream the new cover.
  237. * @param manga the manga edited.
  238. * @return true if the cover is updated, false otherwise
  239. */
  240. @Throws(IOException::class)
  241. fun editCoverWithStream(inputStream: InputStream, manga: Manga): Boolean {
  242. if (manga.thumbnail_url != null && manga.favorite) {
  243. coverCache.copyToCache(manga.thumbnail_url, inputStream)
  244. return true
  245. }
  246. return false
  247. }
  248. }