LibraryScreenModel.kt 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790
  1. package eu.kanade.tachiyomi.ui.library
  2. import androidx.compose.runtime.Immutable
  3. import androidx.compose.runtime.getValue
  4. import androidx.compose.runtime.setValue
  5. import androidx.compose.ui.util.fastAny
  6. import androidx.compose.ui.util.fastMap
  7. import cafe.adriel.voyager.core.model.StateScreenModel
  8. import cafe.adriel.voyager.core.model.coroutineScope
  9. import eu.kanade.core.prefs.CheckboxState
  10. import eu.kanade.core.prefs.PreferenceMutableState
  11. import eu.kanade.core.prefs.asState
  12. import eu.kanade.core.util.fastDistinctBy
  13. import eu.kanade.core.util.fastFilter
  14. import eu.kanade.core.util.fastFilterNot
  15. import eu.kanade.core.util.fastMapNotNull
  16. import eu.kanade.core.util.fastPartition
  17. import eu.kanade.domain.base.BasePreferences
  18. import eu.kanade.domain.category.interactor.GetCategories
  19. import eu.kanade.domain.category.interactor.SetMangaCategories
  20. import eu.kanade.domain.category.model.Category
  21. import eu.kanade.domain.chapter.interactor.GetChapterByMangaId
  22. import eu.kanade.domain.chapter.interactor.SetReadStatus
  23. import eu.kanade.domain.chapter.model.Chapter
  24. import eu.kanade.domain.history.interactor.GetNextChapters
  25. import eu.kanade.domain.library.model.LibraryManga
  26. import eu.kanade.domain.library.model.LibrarySort
  27. import eu.kanade.domain.library.model.sort
  28. import eu.kanade.domain.library.service.LibraryPreferences
  29. import eu.kanade.domain.manga.interactor.GetLibraryManga
  30. import eu.kanade.domain.manga.interactor.UpdateManga
  31. import eu.kanade.domain.manga.model.Manga
  32. import eu.kanade.domain.manga.model.MangaUpdate
  33. import eu.kanade.domain.manga.model.isLocal
  34. import eu.kanade.domain.track.interactor.GetTracksPerManga
  35. import eu.kanade.presentation.library.components.LibraryToolbarTitle
  36. import eu.kanade.presentation.manga.DownloadAction
  37. import eu.kanade.tachiyomi.data.cache.CoverCache
  38. import eu.kanade.tachiyomi.data.download.DownloadCache
  39. import eu.kanade.tachiyomi.data.download.DownloadManager
  40. import eu.kanade.tachiyomi.data.track.TrackManager
  41. import eu.kanade.tachiyomi.source.SourceManager
  42. import eu.kanade.tachiyomi.source.model.SManga
  43. import eu.kanade.tachiyomi.source.online.HttpSource
  44. import eu.kanade.tachiyomi.util.chapter.getNextUnread
  45. import eu.kanade.tachiyomi.util.lang.launchIO
  46. import eu.kanade.tachiyomi.util.lang.launchNonCancellable
  47. import eu.kanade.tachiyomi.util.lang.withIOContext
  48. import eu.kanade.tachiyomi.util.removeCovers
  49. import eu.kanade.tachiyomi.widget.ExtendedNavigationView.Item.TriStateGroup
  50. import kotlinx.coroutines.flow.Flow
  51. import kotlinx.coroutines.flow.collectLatest
  52. import kotlinx.coroutines.flow.combine
  53. import kotlinx.coroutines.flow.distinctUntilChanged
  54. import kotlinx.coroutines.flow.first
  55. import kotlinx.coroutines.flow.flowOf
  56. import kotlinx.coroutines.flow.launchIn
  57. import kotlinx.coroutines.flow.map
  58. import kotlinx.coroutines.flow.onEach
  59. import kotlinx.coroutines.flow.update
  60. import uy.kohesive.injekt.Injekt
  61. import uy.kohesive.injekt.api.get
  62. import java.text.Collator
  63. import java.util.Collections
  64. import java.util.Locale
  65. /**
  66. * Typealias for the library manga, using the category as keys, and list of manga as values.
  67. */
  68. typealias LibraryMap = Map<Category, List<LibraryItem>>
  69. class LibraryScreenModel(
  70. private val getLibraryManga: GetLibraryManga = Injekt.get(),
  71. private val getCategories: GetCategories = Injekt.get(),
  72. private val getTracksPerManga: GetTracksPerManga = Injekt.get(),
  73. private val getNextChapters: GetNextChapters = Injekt.get(),
  74. private val getChaptersByMangaId: GetChapterByMangaId = Injekt.get(),
  75. private val setReadStatus: SetReadStatus = Injekt.get(),
  76. private val updateManga: UpdateManga = Injekt.get(),
  77. private val setMangaCategories: SetMangaCategories = Injekt.get(),
  78. private val preferences: BasePreferences = Injekt.get(),
  79. private val libraryPreferences: LibraryPreferences = Injekt.get(),
  80. private val coverCache: CoverCache = Injekt.get(),
  81. private val sourceManager: SourceManager = Injekt.get(),
  82. private val downloadManager: DownloadManager = Injekt.get(),
  83. private val downloadCache: DownloadCache = Injekt.get(),
  84. private val trackManager: TrackManager = Injekt.get(),
  85. ) : StateScreenModel<LibraryScreenModel.State>(State()) {
  86. // This is active category INDEX NUMBER
  87. var activeCategory: Int by libraryPreferences.lastUsedCategory().asState(coroutineScope)
  88. val isDownloadOnly: Boolean by preferences.downloadedOnly().asState(coroutineScope)
  89. val isIncognitoMode: Boolean by preferences.incognitoMode().asState(coroutineScope)
  90. init {
  91. coroutineScope.launchIO {
  92. combine(
  93. state.map { it.searchQuery }.distinctUntilChanged(),
  94. getLibraryFlow(),
  95. getTracksPerManga.subscribe(),
  96. getTrackingFilterFlow(),
  97. ) { searchQuery, library, tracks, loggedInTrackServices ->
  98. library
  99. .applyFilters(tracks, loggedInTrackServices)
  100. .applySort()
  101. .mapValues { (_, value) ->
  102. if (searchQuery != null) {
  103. // Filter query
  104. value.filter { it.matches(searchQuery) }
  105. } else {
  106. // Don't do anything
  107. value
  108. }
  109. }
  110. }
  111. .collectLatest {
  112. mutableState.update { state ->
  113. state.copy(
  114. isLoading = false,
  115. library = it,
  116. )
  117. }
  118. }
  119. }
  120. combine(
  121. libraryPreferences.categoryTabs().changes(),
  122. libraryPreferences.categoryNumberOfItems().changes(),
  123. libraryPreferences.showContinueReadingButton().changes(),
  124. ) { a, b, c -> arrayOf(a, b, c) }
  125. .onEach { (showCategoryTabs, showMangaCount, showMangaContinueButton) ->
  126. mutableState.update { state ->
  127. state.copy(
  128. showCategoryTabs = showCategoryTabs,
  129. showMangaCount = showMangaCount,
  130. showMangaContinueButton = showMangaContinueButton,
  131. )
  132. }
  133. }
  134. .launchIn(coroutineScope)
  135. combine(
  136. getLibraryItemPreferencesFlow(),
  137. getTrackingFilterFlow(),
  138. ) { prefs, trackFilter ->
  139. val a = (
  140. prefs.filterDownloaded or
  141. prefs.filterUnread or
  142. prefs.filterStarted or
  143. prefs.filterBookmarked or
  144. prefs.filterCompleted
  145. ) != TriStateGroup.State.IGNORE.value
  146. val b = trackFilter.values.any { it != TriStateGroup.State.IGNORE.value }
  147. a || b
  148. }
  149. .distinctUntilChanged()
  150. .onEach {
  151. mutableState.update { state ->
  152. state.copy(hasActiveFilters = it)
  153. }
  154. }
  155. .launchIn(coroutineScope)
  156. }
  157. /**
  158. * Applies library filters to the given map of manga.
  159. */
  160. private suspend fun LibraryMap.applyFilters(
  161. trackMap: Map<Long, List<Long>>,
  162. loggedInTrackServices: Map<Long, Int>,
  163. ): LibraryMap {
  164. val prefs = getLibraryItemPreferencesFlow().first()
  165. val downloadedOnly = prefs.globalFilterDownloaded
  166. val filterDownloaded = prefs.filterDownloaded
  167. val filterUnread = prefs.filterUnread
  168. val filterStarted = prefs.filterStarted
  169. val filterBookmarked = prefs.filterBookmarked
  170. val filterCompleted = prefs.filterCompleted
  171. val isNotLoggedInAnyTrack = loggedInTrackServices.isEmpty()
  172. val excludedTracks = loggedInTrackServices.mapNotNull { if (it.value == TriStateGroup.State.EXCLUDE.value) it.key else null }
  173. val includedTracks = loggedInTrackServices.mapNotNull { if (it.value == TriStateGroup.State.INCLUDE.value) it.key else null }
  174. val trackFiltersIsIgnored = includedTracks.isEmpty() && excludedTracks.isEmpty()
  175. val filterFnDownloaded: (LibraryItem) -> Boolean = downloaded@{ item ->
  176. if (!downloadedOnly && filterDownloaded == TriStateGroup.State.IGNORE.value) return@downloaded true
  177. val isDownloaded = when {
  178. item.libraryManga.manga.isLocal() -> true
  179. item.downloadCount != -1L -> item.downloadCount > 0
  180. else -> downloadManager.getDownloadCount(item.libraryManga.manga) > 0
  181. }
  182. return@downloaded if (downloadedOnly || filterDownloaded == TriStateGroup.State.INCLUDE.value) {
  183. isDownloaded
  184. } else {
  185. !isDownloaded
  186. }
  187. }
  188. val filterFnUnread: (LibraryItem) -> Boolean = unread@{ item ->
  189. if (filterUnread == TriStateGroup.State.IGNORE.value) return@unread true
  190. val isUnread = item.libraryManga.unreadCount > 0
  191. return@unread if (filterUnread == TriStateGroup.State.INCLUDE.value) {
  192. isUnread
  193. } else {
  194. !isUnread
  195. }
  196. }
  197. val filterFnStarted: (LibraryItem) -> Boolean = started@{ item ->
  198. if (filterStarted == TriStateGroup.State.IGNORE.value) return@started true
  199. val hasStarted = item.libraryManga.hasStarted
  200. return@started if (filterStarted == TriStateGroup.State.INCLUDE.value) {
  201. hasStarted
  202. } else {
  203. !hasStarted
  204. }
  205. }
  206. val filterFnBookmarked: (LibraryItem) -> Boolean = bookmarked@{ item ->
  207. if (filterBookmarked == TriStateGroup.State.IGNORE.value) return@bookmarked true
  208. val hasBookmarks = item.libraryManga.hasBookmarks
  209. return@bookmarked if (filterBookmarked == TriStateGroup.State.INCLUDE.value) {
  210. hasBookmarks
  211. } else {
  212. !hasBookmarks
  213. }
  214. }
  215. val filterFnCompleted: (LibraryItem) -> Boolean = completed@{ item ->
  216. if (filterCompleted == TriStateGroup.State.IGNORE.value) return@completed true
  217. val isCompleted = item.libraryManga.manga.status.toInt() == SManga.COMPLETED
  218. return@completed if (filterCompleted == TriStateGroup.State.INCLUDE.value) {
  219. isCompleted
  220. } else {
  221. !isCompleted
  222. }
  223. }
  224. val filterFnTracking: (LibraryItem) -> Boolean = tracking@{ item ->
  225. if (isNotLoggedInAnyTrack || trackFiltersIsIgnored) return@tracking true
  226. val mangaTracks = trackMap[item.libraryManga.id].orEmpty()
  227. val exclude = mangaTracks.fastFilter { it in excludedTracks }
  228. val include = mangaTracks.fastFilter { it in includedTracks }
  229. // TODO: Simplify the filter logic
  230. if (includedTracks.isNotEmpty() && excludedTracks.isNotEmpty()) {
  231. return@tracking if (exclude.isNotEmpty()) false else include.isNotEmpty()
  232. }
  233. if (excludedTracks.isNotEmpty()) return@tracking exclude.isEmpty()
  234. if (includedTracks.isNotEmpty()) return@tracking include.isNotEmpty()
  235. return@tracking false
  236. }
  237. val filterFn: (LibraryItem) -> Boolean = filter@{ item ->
  238. return@filter !(
  239. !filterFnDownloaded(item) ||
  240. !filterFnUnread(item) ||
  241. !filterFnStarted(item) ||
  242. !filterFnBookmarked(item) ||
  243. !filterFnCompleted(item) ||
  244. !filterFnTracking(item)
  245. )
  246. }
  247. return this.mapValues { entry -> entry.value.fastFilter(filterFn) }
  248. }
  249. /**
  250. * Applies library sorting to the given map of manga.
  251. */
  252. private fun LibraryMap.applySort(): LibraryMap {
  253. val locale = Locale.getDefault()
  254. val collator = Collator.getInstance(locale).apply {
  255. strength = Collator.PRIMARY
  256. }
  257. val sortAlphabetically: (LibraryItem, LibraryItem) -> Int = { i1, i2 ->
  258. collator.compare(i1.libraryManga.manga.title.lowercase(locale), i2.libraryManga.manga.title.lowercase(locale))
  259. }
  260. val sortFn: (LibraryItem, LibraryItem) -> Int = { i1, i2 ->
  261. val sort = keys.find { it.id == i1.libraryManga.category }!!.sort
  262. when (sort.type) {
  263. LibrarySort.Type.Alphabetical -> {
  264. sortAlphabetically(i1, i2)
  265. }
  266. LibrarySort.Type.LastRead -> {
  267. i1.libraryManga.lastRead.compareTo(i2.libraryManga.lastRead)
  268. }
  269. LibrarySort.Type.LastUpdate -> {
  270. i1.libraryManga.manga.lastUpdate.compareTo(i2.libraryManga.manga.lastUpdate)
  271. }
  272. LibrarySort.Type.UnreadCount -> when {
  273. // Ensure unread content comes first
  274. i1.libraryManga.unreadCount == i2.libraryManga.unreadCount -> 0
  275. i1.libraryManga.unreadCount == 0L -> if (sort.isAscending) 1 else -1
  276. i2.libraryManga.unreadCount == 0L -> if (sort.isAscending) -1 else 1
  277. else -> i1.libraryManga.unreadCount.compareTo(i2.libraryManga.unreadCount)
  278. }
  279. LibrarySort.Type.TotalChapters -> {
  280. i1.libraryManga.totalChapters.compareTo(i2.libraryManga.totalChapters)
  281. }
  282. LibrarySort.Type.LatestChapter -> {
  283. i1.libraryManga.latestUpload.compareTo(i2.libraryManga.latestUpload)
  284. }
  285. LibrarySort.Type.ChapterFetchDate -> {
  286. i1.libraryManga.chapterFetchedAt.compareTo(i2.libraryManga.chapterFetchedAt)
  287. }
  288. LibrarySort.Type.DateAdded -> {
  289. i1.libraryManga.manga.dateAdded.compareTo(i2.libraryManga.manga.dateAdded)
  290. }
  291. }
  292. }
  293. return this.mapValues { entry ->
  294. val comparator = if (keys.find { it.id == entry.key.id }!!.sort.isAscending) {
  295. Comparator(sortFn)
  296. } else {
  297. Collections.reverseOrder(sortFn)
  298. }
  299. entry.value.sortedWith(comparator.thenComparator(sortAlphabetically))
  300. }
  301. }
  302. private fun getLibraryItemPreferencesFlow(): Flow<ItemPreferences> {
  303. return combine(
  304. libraryPreferences.downloadBadge().changes(),
  305. libraryPreferences.unreadBadge().changes(),
  306. libraryPreferences.localBadge().changes(),
  307. libraryPreferences.languageBadge().changes(),
  308. preferences.downloadedOnly().changes(),
  309. libraryPreferences.filterDownloaded().changes(),
  310. libraryPreferences.filterUnread().changes(),
  311. libraryPreferences.filterStarted().changes(),
  312. libraryPreferences.filterBookmarked().changes(),
  313. libraryPreferences.filterCompleted().changes(),
  314. transform = {
  315. ItemPreferences(
  316. downloadBadge = it[0] as Boolean,
  317. unreadBadge = it[1] as Boolean,
  318. localBadge = it[2] as Boolean,
  319. languageBadge = it[3] as Boolean,
  320. globalFilterDownloaded = it[4] as Boolean,
  321. filterDownloaded = it[5] as Int,
  322. filterUnread = it[6] as Int,
  323. filterStarted = it[7] as Int,
  324. filterBookmarked = it[8] as Int,
  325. filterCompleted = it[9] as Int,
  326. )
  327. },
  328. )
  329. }
  330. /**
  331. * Get the categories and all its manga from the database.
  332. *
  333. * @return an observable of the categories and its manga.
  334. */
  335. private fun getLibraryFlow(): Flow<LibraryMap> {
  336. val libraryMangasFlow = combine(
  337. getLibraryManga.subscribe(),
  338. getLibraryItemPreferencesFlow(),
  339. downloadCache.changes,
  340. ) { libraryMangaList, prefs, _ ->
  341. libraryMangaList
  342. .map { libraryManga ->
  343. val needsDownloadCounts = prefs.downloadBadge ||
  344. prefs.filterDownloaded != TriStateGroup.State.IGNORE.value ||
  345. prefs.globalFilterDownloaded
  346. // Display mode based on user preference: take it from global library setting or category
  347. LibraryItem(libraryManga).apply {
  348. downloadCount = if (needsDownloadCounts) {
  349. downloadManager.getDownloadCount(libraryManga.manga).toLong()
  350. } else {
  351. 0
  352. }
  353. unreadCount = if (prefs.unreadBadge) libraryManga.unreadCount else 0
  354. isLocal = if (prefs.localBadge) libraryManga.manga.isLocal() else false
  355. sourceLanguage = if (prefs.languageBadge) {
  356. sourceManager.getOrStub(libraryManga.manga.source).lang
  357. } else {
  358. ""
  359. }
  360. }
  361. }
  362. .groupBy { it.libraryManga.category }
  363. }
  364. return combine(getCategories.subscribe(), libraryMangasFlow) { categories, libraryManga ->
  365. val displayCategories = if (libraryManga.isNotEmpty() && !libraryManga.containsKey(0)) {
  366. categories.fastFilterNot { it.isSystemCategory }
  367. } else {
  368. categories
  369. }
  370. displayCategories.associateWith { libraryManga[it.id] ?: emptyList() }
  371. }
  372. }
  373. /**
  374. * Flow of tracking filter preferences
  375. *
  376. * @return map of track id with the filter value
  377. */
  378. private fun getTrackingFilterFlow(): Flow<Map<Long, Int>> {
  379. val loggedServices = trackManager.services.filter { it.isLogged }
  380. return if (loggedServices.isNotEmpty()) {
  381. val prefFlows = loggedServices
  382. .map { libraryPreferences.filterTracking(it.id.toInt()).changes() }
  383. .toTypedArray()
  384. combine(*prefFlows) {
  385. loggedServices
  386. .mapIndexed { index, trackService -> trackService.id to it[index] }
  387. .toMap()
  388. }
  389. } else {
  390. flowOf(emptyMap())
  391. }
  392. }
  393. /**
  394. * Returns the common categories for the given list of manga.
  395. *
  396. * @param mangas the list of manga.
  397. */
  398. private suspend fun getCommonCategories(mangas: List<Manga>): Collection<Category> {
  399. if (mangas.isEmpty()) return emptyList()
  400. return mangas
  401. .map { getCategories.await(it.id).toSet() }
  402. .reduce { set1, set2 -> set1.intersect(set2) }
  403. }
  404. suspend fun getNextUnreadChapter(manga: Manga): Chapter? {
  405. return getChaptersByMangaId.await(manga.id).getNextUnread(manga, downloadManager)
  406. }
  407. /**
  408. * Returns the mix (non-common) categories for the given list of manga.
  409. *
  410. * @param mangas the list of manga.
  411. */
  412. private suspend fun getMixCategories(mangas: List<Manga>): Collection<Category> {
  413. if (mangas.isEmpty()) return emptyList()
  414. val mangaCategories = mangas.map { getCategories.await(it.id).toSet() }
  415. val common = mangaCategories.reduce { set1, set2 -> set1.intersect(set2) }
  416. return mangaCategories.flatten().distinct().subtract(common)
  417. }
  418. fun runDownloadActionSelection(action: DownloadAction) {
  419. val selection = state.value.selection
  420. val mangas = selection.map { it.manga }.toList()
  421. when (action) {
  422. DownloadAction.NEXT_1_CHAPTER -> downloadUnreadChapters(mangas, 1)
  423. DownloadAction.NEXT_5_CHAPTERS -> downloadUnreadChapters(mangas, 5)
  424. DownloadAction.NEXT_10_CHAPTERS -> downloadUnreadChapters(mangas, 10)
  425. DownloadAction.UNREAD_CHAPTERS -> downloadUnreadChapters(mangas, null)
  426. DownloadAction.CUSTOM -> {
  427. mutableState.update { state ->
  428. state.copy(
  429. dialog = Dialog.DownloadCustomAmount(
  430. mangas,
  431. selection.maxOf { it.unreadCount }.toInt(),
  432. ),
  433. )
  434. }
  435. return
  436. }
  437. else -> {}
  438. }
  439. clearSelection()
  440. }
  441. /**
  442. * Queues the amount specified of unread chapters from the list of mangas given.
  443. *
  444. * @param mangas the list of manga.
  445. * @param amount the amount to queue or null to queue all
  446. */
  447. fun downloadUnreadChapters(mangas: List<Manga>, amount: Int?) {
  448. coroutineScope.launchNonCancellable {
  449. mangas.forEach { manga ->
  450. val chapters = getNextChapters.await(manga.id)
  451. .fastFilterNot { chapter ->
  452. downloadManager.getQueuedDownloadOrNull(chapter.id) != null ||
  453. downloadManager.isChapterDownloaded(
  454. chapter.name,
  455. chapter.scanlator,
  456. manga.title,
  457. manga.source,
  458. )
  459. }
  460. .let { if (amount != null) it.take(amount) else it }
  461. downloadManager.downloadChapters(manga, chapters)
  462. }
  463. }
  464. }
  465. /**
  466. * Marks mangas' chapters read status.
  467. */
  468. fun markReadSelection(read: Boolean) {
  469. val mangas = state.value.selection.toList()
  470. coroutineScope.launchNonCancellable {
  471. mangas.forEach { manga ->
  472. setReadStatus.await(
  473. manga = manga.manga,
  474. read = read,
  475. )
  476. }
  477. }
  478. clearSelection()
  479. }
  480. /**
  481. * Remove the selected manga.
  482. *
  483. * @param mangaList the list of manga to delete.
  484. * @param deleteFromLibrary whether to delete manga from library.
  485. * @param deleteChapters whether to delete downloaded chapters.
  486. */
  487. fun removeMangas(mangaList: List<Manga>, deleteFromLibrary: Boolean, deleteChapters: Boolean) {
  488. coroutineScope.launchNonCancellable {
  489. val mangaToDelete = mangaList.distinctBy { it.id }
  490. if (deleteFromLibrary) {
  491. val toDelete = mangaToDelete.map {
  492. it.removeCovers(coverCache)
  493. MangaUpdate(
  494. favorite = false,
  495. id = it.id,
  496. )
  497. }
  498. updateManga.awaitAll(toDelete)
  499. }
  500. if (deleteChapters) {
  501. mangaToDelete.forEach { manga ->
  502. val source = sourceManager.get(manga.source) as? HttpSource
  503. if (source != null) {
  504. downloadManager.deleteManga(manga, source)
  505. }
  506. }
  507. }
  508. }
  509. }
  510. /**
  511. * Bulk update categories of manga using old and new common categories.
  512. *
  513. * @param mangaList the list of manga to move.
  514. * @param addCategories the categories to add for all mangas.
  515. * @param removeCategories the categories to remove in all mangas.
  516. */
  517. fun setMangaCategories(mangaList: List<Manga>, addCategories: List<Long>, removeCategories: List<Long>) {
  518. coroutineScope.launchNonCancellable {
  519. mangaList.forEach { manga ->
  520. val categoryIds = getCategories.await(manga.id)
  521. .map { it.id }
  522. .subtract(removeCategories.toSet())
  523. .plus(addCategories)
  524. .toList()
  525. setMangaCategories.await(manga.id, categoryIds)
  526. }
  527. }
  528. }
  529. fun getColumnsPreferenceForCurrentOrientation(isLandscape: Boolean): PreferenceMutableState<Int> {
  530. return (if (isLandscape) libraryPreferences.landscapeColumns() else libraryPreferences.portraitColumns()).asState(coroutineScope)
  531. }
  532. suspend fun getRandomLibraryItemForCurrentCategory(): LibraryItem? {
  533. return withIOContext {
  534. state.value
  535. .getLibraryItemsByCategoryId(activeCategory.toLong())
  536. .randomOrNull()
  537. }
  538. }
  539. fun clearSelection() {
  540. mutableState.update { it.copy(selection = emptyList()) }
  541. }
  542. fun toggleSelection(manga: LibraryManga) {
  543. mutableState.update { state ->
  544. val newSelection = state.selection.toMutableList().apply {
  545. if (fastAny { it.id == manga.id }) {
  546. removeAll { it.id == manga.id }
  547. } else {
  548. add(manga)
  549. }
  550. }
  551. state.copy(selection = newSelection)
  552. }
  553. }
  554. /**
  555. * Selects all mangas between and including the given manga and the last pressed manga from the
  556. * same category as the given manga
  557. */
  558. fun toggleRangeSelection(manga: LibraryManga) {
  559. mutableState.update { state ->
  560. val newSelection = state.selection.toMutableList().apply {
  561. val lastSelected = lastOrNull()
  562. if (lastSelected?.category != manga.category) {
  563. add(manga)
  564. return@apply
  565. }
  566. val items = state.getLibraryItemsByCategoryId(manga.category)
  567. .fastMap { it.libraryManga }
  568. val lastMangaIndex = items.indexOf(lastSelected)
  569. val curMangaIndex = items.indexOf(manga)
  570. val selectedIds = fastMap { it.id }
  571. val selectionRange = when {
  572. lastMangaIndex < curMangaIndex -> IntRange(lastMangaIndex, curMangaIndex)
  573. curMangaIndex < lastMangaIndex -> IntRange(curMangaIndex, lastMangaIndex)
  574. // We shouldn't reach this point
  575. else -> return@apply
  576. }
  577. val newSelections = selectionRange.mapNotNull { index ->
  578. items[index].takeUnless { it.id in selectedIds }
  579. }
  580. addAll(newSelections)
  581. }
  582. state.copy(selection = newSelection)
  583. }
  584. }
  585. fun selectAll(index: Int) {
  586. mutableState.update { state ->
  587. val newSelection = state.selection.toMutableList().apply {
  588. val categoryId = state.categories.getOrNull(index)?.id ?: -1
  589. val selectedIds = fastMap { it.id }
  590. val newSelections = state.getLibraryItemsByCategoryId(categoryId)
  591. .fastMapNotNull { item ->
  592. item.libraryManga.takeUnless { it.id in selectedIds }
  593. }
  594. addAll(newSelections)
  595. }
  596. state.copy(selection = newSelection)
  597. }
  598. }
  599. fun invertSelection(index: Int) {
  600. mutableState.update { state ->
  601. val newSelection = state.selection.toMutableList().apply {
  602. val categoryId = state.categories[index].id
  603. val items = state.getLibraryItemsByCategoryId(categoryId).fastMap { it.libraryManga }
  604. val selectedIds = fastMap { it.id }
  605. val (toRemove, toAdd) = items.fastPartition { it.id in selectedIds }
  606. val toRemoveIds = toRemove.fastMap { it.id }
  607. removeAll { it.id in toRemoveIds }
  608. addAll(toAdd)
  609. }
  610. state.copy(selection = newSelection)
  611. }
  612. }
  613. fun search(query: String?) {
  614. mutableState.update { it.copy(searchQuery = query) }
  615. }
  616. fun openChangeCategoryDialog() {
  617. coroutineScope.launchIO {
  618. // Create a copy of selected manga
  619. val mangaList = state.value.selection.map { it.manga }
  620. // Hide the default category because it has a different behavior than the ones from db.
  621. val categories = state.value.categories.filter { it.id != 0L }
  622. // Get indexes of the common categories to preselect.
  623. val common = getCommonCategories(mangaList)
  624. // Get indexes of the mix categories to preselect.
  625. val mix = getMixCategories(mangaList)
  626. val preselected = categories.map {
  627. when (it) {
  628. in common -> CheckboxState.State.Checked(it)
  629. in mix -> CheckboxState.TriState.Exclude(it)
  630. else -> CheckboxState.State.None(it)
  631. }
  632. }
  633. mutableState.update { it.copy(dialog = Dialog.ChangeCategory(mangaList, preselected)) }
  634. }
  635. }
  636. fun openDeleteMangaDialog() {
  637. val mangaList = state.value.selection.map { it.manga }
  638. mutableState.update { it.copy(dialog = Dialog.DeleteManga(mangaList)) }
  639. }
  640. fun closeDialog() {
  641. mutableState.update { it.copy(dialog = null) }
  642. }
  643. sealed class Dialog {
  644. data class ChangeCategory(val manga: List<Manga>, val initialSelection: List<CheckboxState<Category>>) : Dialog()
  645. data class DeleteManga(val manga: List<Manga>) : Dialog()
  646. data class DownloadCustomAmount(val manga: List<Manga>, val max: Int) : Dialog()
  647. }
  648. @Immutable
  649. private data class ItemPreferences(
  650. val downloadBadge: Boolean,
  651. val unreadBadge: Boolean,
  652. val localBadge: Boolean,
  653. val languageBadge: Boolean,
  654. val globalFilterDownloaded: Boolean,
  655. val filterDownloaded: Int,
  656. val filterUnread: Int,
  657. val filterStarted: Int,
  658. val filterBookmarked: Int,
  659. val filterCompleted: Int,
  660. )
  661. @Immutable
  662. data class State(
  663. val isLoading: Boolean = true,
  664. val library: LibraryMap = emptyMap(),
  665. val searchQuery: String? = null,
  666. val selection: List<LibraryManga> = emptyList(),
  667. val hasActiveFilters: Boolean = false,
  668. val showCategoryTabs: Boolean = false,
  669. val showMangaCount: Boolean = false,
  670. val showMangaContinueButton: Boolean = false,
  671. val dialog: Dialog? = null,
  672. ) {
  673. val selectionMode = selection.isNotEmpty()
  674. val categories = library.keys.toList()
  675. val libraryCount by lazy {
  676. library.values
  677. .flatten()
  678. .fastDistinctBy { it.libraryManga.manga.id }
  679. .size
  680. }
  681. fun getLibraryItemsByCategoryId(categoryId: Long): List<LibraryItem> {
  682. return library.firstNotNullOf { (k, v) -> v.takeIf { k.id == categoryId } }
  683. }
  684. fun getLibraryItemsByPage(page: Int): List<LibraryItem> {
  685. return library.values.toTypedArray().getOrNull(page) ?: emptyList()
  686. }
  687. fun getMangaCountForCategory(category: Category): Int? {
  688. return if (showMangaCount || !searchQuery.isNullOrEmpty()) library[category]?.size else null
  689. }
  690. fun getToolbarTitle(
  691. defaultTitle: String,
  692. defaultCategoryTitle: String,
  693. page: Int,
  694. ): LibraryToolbarTitle {
  695. val category = categories.getOrNull(page) ?: return LibraryToolbarTitle(defaultTitle)
  696. val categoryName = category.let {
  697. if (it.isSystemCategory) defaultCategoryTitle else it.name
  698. }
  699. val title = if (showCategoryTabs) defaultTitle else categoryName
  700. val count = when {
  701. !showMangaCount -> null
  702. !showCategoryTabs -> getMangaCountForCategory(category)
  703. // Whole library count
  704. else -> libraryCount
  705. }
  706. return LibraryToolbarTitle(title, count)
  707. }
  708. }
  709. }