|
@@ -2,8 +2,6 @@ package eu.kanade.tachiyomi.ui.recent_updates
|
|
|
|
|
|
import android.os.Bundle
|
|
|
import eu.kanade.tachiyomi.data.database.DatabaseHelper
|
|
|
-import eu.kanade.tachiyomi.data.database.models.Chapter
|
|
|
-import eu.kanade.tachiyomi.data.database.models.Manga
|
|
|
import eu.kanade.tachiyomi.data.database.models.MangaChapter
|
|
|
import eu.kanade.tachiyomi.data.download.DownloadManager
|
|
|
import eu.kanade.tachiyomi.data.download.DownloadService
|
|
@@ -15,34 +13,34 @@ import rx.Observable
|
|
|
import rx.android.schedulers.AndroidSchedulers
|
|
|
import rx.schedulers.Schedulers
|
|
|
import timber.log.Timber
|
|
|
+import uy.kohesive.injekt.injectLazy
|
|
|
import java.util.*
|
|
|
-import javax.inject.Inject
|
|
|
|
|
|
class RecentChaptersPresenter : BasePresenter<RecentChaptersFragment>() {
|
|
|
/**
|
|
|
* Used to connect to database
|
|
|
*/
|
|
|
- @Inject lateinit var db: DatabaseHelper
|
|
|
+ val db: DatabaseHelper by injectLazy()
|
|
|
|
|
|
/**
|
|
|
* Used to get settings
|
|
|
*/
|
|
|
- @Inject lateinit var preferences: PreferencesHelper
|
|
|
+ val preferences: PreferencesHelper by injectLazy()
|
|
|
|
|
|
/**
|
|
|
* Used to get information from download manager
|
|
|
*/
|
|
|
- @Inject lateinit var downloadManager: DownloadManager
|
|
|
+ val downloadManager: DownloadManager by injectLazy()
|
|
|
|
|
|
/**
|
|
|
* Used to get source from source id
|
|
|
*/
|
|
|
- @Inject lateinit var sourceManager: SourceManager
|
|
|
+ val sourceManager: SourceManager by injectLazy()
|
|
|
|
|
|
/**
|
|
|
* List containing chapter and manga information
|
|
|
*/
|
|
|
- private var mangaChapters: List<MangaChapter>? = null
|
|
|
+ private var chapters: List<RecentChapter>? = null
|
|
|
|
|
|
/**
|
|
|
* The id of the restartable.
|
|
@@ -60,171 +58,114 @@ class RecentChaptersPresenter : BasePresenter<RecentChaptersFragment>() {
|
|
|
// Used to get recent chapters
|
|
|
restartableLatestCache(GET_RECENT_CHAPTERS,
|
|
|
{ getRecentChaptersObservable() },
|
|
|
- { fragment, chapters ->
|
|
|
+ { view, chapters ->
|
|
|
// Update adapter to show recent manga's
|
|
|
- fragment.onNextMangaChapters(chapters)
|
|
|
- // Update download status
|
|
|
- updateChapterStatus(convertToMangaChaptersList(chapters))
|
|
|
+ view.onNextRecentChapters(chapters)
|
|
|
}
|
|
|
)
|
|
|
|
|
|
// Used to update download status
|
|
|
- startableLatestCache(CHAPTER_STATUS_CHANGES,
|
|
|
- { getChapterStatusObs() },
|
|
|
- { recentChaptersFragment, download ->
|
|
|
+ restartableLatestCache(CHAPTER_STATUS_CHANGES,
|
|
|
+ { getChapterStatusObservable() },
|
|
|
+ { view, download ->
|
|
|
// Set chapter status
|
|
|
- recentChaptersFragment.onChapterStatusChange(download)
|
|
|
+ view.onChapterStatusChange(download)
|
|
|
},
|
|
|
{ view, error -> Timber.e(error.cause, error.message) }
|
|
|
)
|
|
|
|
|
|
-
|
|
|
if (savedState == null) {
|
|
|
// Start fetching recent chapters
|
|
|
start(GET_RECENT_CHAPTERS)
|
|
|
+ start(CHAPTER_STATUS_CHANGES)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+ /**
|
|
|
+ * Get observable containing recent chapters and date
|
|
|
+ * @return observable containing recent chapters and date
|
|
|
+ */
|
|
|
+ fun getRecentChaptersObservable(): Observable<ArrayList<Any>> {
|
|
|
+ // Set date for recent chapters
|
|
|
+ val cal = Calendar.getInstance().apply {
|
|
|
+ time = Date()
|
|
|
+ add(Calendar.MONTH, -1)
|
|
|
+ }
|
|
|
+
|
|
|
+ return db.getRecentChapters(cal.time).asRxObservable()
|
|
|
+ // Convert to a list of recent chapters.
|
|
|
+ .map { mangaChapters ->
|
|
|
+ mangaChapters.map { it.toModel() }
|
|
|
+ }
|
|
|
+ .doOnNext { chapters = it }
|
|
|
+ // Group chapters by the date they were fetched on a ordered map.
|
|
|
+ .flatMap { recentItems ->
|
|
|
+ Observable.from(recentItems)
|
|
|
+ .toMultimap(
|
|
|
+ { getMapKey(it.date_fetch) },
|
|
|
+ { it },
|
|
|
+ { TreeMap { d1, d2 -> d2.compareTo(d1) } })
|
|
|
+ }
|
|
|
+ // Add every day and all its chapters to a single list.
|
|
|
+ .map { recentItems ->
|
|
|
+ ArrayList<Any>().apply {
|
|
|
+ for ((key, value) in recentItems) {
|
|
|
+ add(key)
|
|
|
+ addAll(value)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ .observeOn(AndroidSchedulers.mainThread())
|
|
|
+ }
|
|
|
|
|
|
/**
|
|
|
* Returns observable containing chapter status.
|
|
|
* @return download object containing download progress.
|
|
|
*/
|
|
|
- private fun getChapterStatusObs(): Observable<Download> {
|
|
|
+ private fun getChapterStatusObservable(): Observable<Download> {
|
|
|
return downloadManager.queue.getStatusObservable()
|
|
|
.observeOn(AndroidSchedulers.mainThread())
|
|
|
- .filter { download: Download ->
|
|
|
- if (chapterIdEquals(download.chapter.id))
|
|
|
- true
|
|
|
- else
|
|
|
- false
|
|
|
- }
|
|
|
- .doOnNext { download1: Download -> updateChapterStatus(download1) }
|
|
|
-
|
|
|
+ .doOnNext { download -> onDownloadStatusChange(download) }
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * Function to check if chapter is in recent list
|
|
|
- * @param chaptersId id of chapter
|
|
|
- * @return exist in recent list
|
|
|
+ * Converts a chapter from the database to an extended model, allowing to store new fields.
|
|
|
*/
|
|
|
- private fun chapterIdEquals(chaptersId: Long): Boolean {
|
|
|
- mangaChapters!!.forEach { mangaChapter ->
|
|
|
- if (chaptersId == mangaChapter.chapter.id) {
|
|
|
- return true
|
|
|
- }
|
|
|
- }
|
|
|
- return false
|
|
|
- }
|
|
|
+ private fun MangaChapter.toModel(): RecentChapter {
|
|
|
+ // Create the model object.
|
|
|
+ val model = RecentChapter(this)
|
|
|
|
|
|
- /**
|
|
|
- * Returns a list only containing MangaChapter objects.
|
|
|
- * @param input the list that will be converted.
|
|
|
- * @return list containing MangaChapters objects.
|
|
|
- */
|
|
|
- private fun convertToMangaChaptersList(input: List<Any>): List<MangaChapter> {
|
|
|
- // Create temp list
|
|
|
- val tempMangaChapterList = ArrayList<MangaChapter>()
|
|
|
+ // Find an active download for this chapter.
|
|
|
+ val download = downloadManager.queue.find { it.chapter.id == chapter.id }
|
|
|
|
|
|
- // Only add MangaChapter objects
|
|
|
- //noinspection Convert2streamapi
|
|
|
- input.forEach { `object` ->
|
|
|
- if (`object` is MangaChapter) {
|
|
|
- tempMangaChapterList.add(`object`)
|
|
|
- }
|
|
|
- }
|
|
|
+ // If there's an active download, assign it, otherwise ask the manager if the chapter is
|
|
|
+ // downloaded and assign it to the status.
|
|
|
+ if (download != null) {
|
|
|
+ model.download = download
|
|
|
+ } else {
|
|
|
+ // Get source of chapter.
|
|
|
+ val source = sourceManager.get(manga.source)!!
|
|
|
|
|
|
- // Return temp list
|
|
|
- return tempMangaChapterList
|
|
|
+ model.status = if (downloadManager.isChapterDownloaded(source, manga, chapter))
|
|
|
+ Download.DOWNLOADED
|
|
|
+ else
|
|
|
+ Download.NOT_DOWNLOADED
|
|
|
+ }
|
|
|
+ return model
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Update status of chapters.
|
|
|
* @param download download object containing progress.
|
|
|
*/
|
|
|
- private fun updateChapterStatus(download: Download) {
|
|
|
- // Loop through list
|
|
|
- mangaChapters?.let {
|
|
|
- for (item in it) {
|
|
|
- if (download.chapter.id == item.chapter.id) {
|
|
|
- // Update status.
|
|
|
- item.chapter.status = download.status
|
|
|
- break
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Update status of chapters
|
|
|
- * @param mangaChapters list containing recent chapters
|
|
|
- */
|
|
|
- private fun updateChapterStatus(mangaChapters: List<MangaChapter>) {
|
|
|
- // Set global list of chapters.
|
|
|
- this.mangaChapters = mangaChapters
|
|
|
-
|
|
|
- // Update status.
|
|
|
- //noinspection Convert2streamapi
|
|
|
- for (mangaChapter in mangaChapters)
|
|
|
- setChapterStatus(mangaChapter)
|
|
|
-
|
|
|
- // Start onChapterStatusChange restartable.
|
|
|
- start(CHAPTER_STATUS_CHANGES)
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Set the chapter status
|
|
|
- * @param mangaChapter MangaChapter which status gets updated
|
|
|
- */
|
|
|
- private fun setChapterStatus(mangaChapter: MangaChapter) {
|
|
|
- // Check if chapter in queue
|
|
|
- for (download in downloadManager.queue) {
|
|
|
- if (mangaChapter.chapter.id == download.chapter.id) {
|
|
|
- mangaChapter.chapter.status = download.status
|
|
|
- return
|
|
|
+ private fun onDownloadStatusChange(download: Download) {
|
|
|
+ // Assign the download to the model object.
|
|
|
+ if (download.status == Download.QUEUE) {
|
|
|
+ val chapter = chapters?.find { it.id == download.chapter.id }
|
|
|
+ if (chapter != null && chapter.download == null) {
|
|
|
+ chapter.download = download
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
- // Get source of chapter
|
|
|
- val source = sourceManager.get(mangaChapter.manga.source)!!
|
|
|
-
|
|
|
- // Check if chapter is downloaded
|
|
|
- if (downloadManager.isChapterDownloaded(source, mangaChapter.manga, mangaChapter.chapter)) {
|
|
|
- mangaChapter.chapter.status = Download.DOWNLOADED
|
|
|
- } else {
|
|
|
- mangaChapter.chapter.status = Download.NOT_DOWNLOADED
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Get observable containing recent chapters and date
|
|
|
- * @return observable containing recent chapters and date
|
|
|
- */
|
|
|
- fun getRecentChaptersObservable(): Observable<ArrayList<Any>>? {
|
|
|
- // Set date for recent chapters
|
|
|
- val cal = Calendar.getInstance()
|
|
|
- cal.time = Date()
|
|
|
- cal.add(Calendar.MONTH, -1)
|
|
|
-
|
|
|
- return db.getRecentChapters(cal.time).asRxObservable()
|
|
|
- // Group chapters by the date they were fetched on a ordered map.
|
|
|
- .flatMap { recentItems ->
|
|
|
- Observable.from(recentItems)
|
|
|
- .toMultimap(
|
|
|
- { getMapKey(it.chapter.date_fetch) },
|
|
|
- { it },
|
|
|
- { TreeMap { d1, d2 -> d2.compareTo(d1) } })
|
|
|
- }
|
|
|
- // Add every day and all its chapters to a single list.
|
|
|
- .map { recentItems ->
|
|
|
- val items = ArrayList<Any>()
|
|
|
- recentItems.entries.forEach { recent ->
|
|
|
- items.add(recent.key)
|
|
|
- items.addAll(recent.value)
|
|
|
- }
|
|
|
- items
|
|
|
- }
|
|
|
- .observeOn(AndroidSchedulers.mainThread())
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -244,18 +185,17 @@ class RecentChaptersPresenter : BasePresenter<RecentChaptersFragment>() {
|
|
|
|
|
|
/**
|
|
|
* Mark selected chapter as read
|
|
|
- * @param mangaChapters list of selected MangaChapters
|
|
|
+ * @param chapters list of selected chapters
|
|
|
* @param read read status
|
|
|
*/
|
|
|
- fun markChapterRead(mangaChapters: List<MangaChapter>, read: Boolean) {
|
|
|
- Observable.from(mangaChapters)
|
|
|
- .doOnNext { mangaChapter ->
|
|
|
- mangaChapter.chapter.read = read
|
|
|
+ fun markChapterRead(chapters: List<RecentChapter>, read: Boolean) {
|
|
|
+ Observable.from(chapters)
|
|
|
+ .doOnNext { chapter ->
|
|
|
+ chapter.read = read
|
|
|
if (!read) {
|
|
|
- mangaChapter.chapter.last_page_read = 0
|
|
|
+ chapter.last_page_read = 0
|
|
|
}
|
|
|
}
|
|
|
- .map { mangaChapter -> mangaChapter.chapter }
|
|
|
.toList()
|
|
|
.flatMap { db.updateChaptersProgress(it).asRxObservable() }
|
|
|
.subscribeOn(Schedulers.io())
|
|
@@ -264,9 +204,9 @@ class RecentChaptersPresenter : BasePresenter<RecentChaptersFragment>() {
|
|
|
|
|
|
/**
|
|
|
* Delete selected chapters
|
|
|
- * @param chapters list of MangaChapters
|
|
|
+ * @param chapters list of chapters
|
|
|
*/
|
|
|
- fun deleteChapters(chapters: List<MangaChapter>) {
|
|
|
+ fun deleteChapters(chapters: List<RecentChapter>) {
|
|
|
val wasRunning = downloadManager.isRunning
|
|
|
if (wasRunning) {
|
|
|
DownloadService.stop(context)
|
|
@@ -288,11 +228,11 @@ class RecentChaptersPresenter : BasePresenter<RecentChaptersFragment>() {
|
|
|
|
|
|
/**
|
|
|
* Download selected chapters
|
|
|
- * @param mangaChapters [MangaChapter] that is selected
|
|
|
+ * @param chapters list of recent chapters seleted.
|
|
|
*/
|
|
|
- fun downloadChapters(mangaChapters: List<MangaChapter>) {
|
|
|
+ fun downloadChapters(chapters: List<RecentChapter>) {
|
|
|
DownloadService.start(context)
|
|
|
- Observable.from(mangaChapters)
|
|
|
+ Observable.from(chapters)
|
|
|
.doOnNext { downloadChapter(it) }
|
|
|
.subscribeOn(AndroidSchedulers.mainThread())
|
|
|
.subscribe()
|
|
@@ -300,47 +240,23 @@ class RecentChaptersPresenter : BasePresenter<RecentChaptersFragment>() {
|
|
|
|
|
|
/**
|
|
|
* Download selected chapter
|
|
|
- * @param item chapter that is selected
|
|
|
+ * @param chapter chapter that is selected
|
|
|
*/
|
|
|
- fun downloadChapter(item: MangaChapter) {
|
|
|
+ fun downloadChapter(chapter: RecentChapter) {
|
|
|
DownloadService.start(context)
|
|
|
- downloadManager.downloadChapters(item.manga, listOf(item.chapter))
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Delete selected chapter
|
|
|
- * @param item chapter that are selected
|
|
|
- */
|
|
|
- fun deleteChapter(item: MangaChapter) {
|
|
|
- val wasRunning = downloadManager.isRunning
|
|
|
- if (wasRunning) {
|
|
|
- DownloadService.stop(context)
|
|
|
- }
|
|
|
- Observable.just(item)
|
|
|
- .doOnNext { deleteChapter(it.chapter, it.manga) }
|
|
|
- .subscribeOn(Schedulers.io())
|
|
|
- .observeOn(AndroidSchedulers.mainThread())
|
|
|
- .subscribeFirst({ view, result ->
|
|
|
- view.onChaptersDeleted()
|
|
|
- if (wasRunning) {
|
|
|
- DownloadService.start(context)
|
|
|
- }
|
|
|
- }, { view, error ->
|
|
|
- view.onChaptersDeletedError(error)
|
|
|
- })
|
|
|
+ downloadManager.downloadChapters(chapter.manga, listOf(chapter))
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Delete selected chapter
|
|
|
* @param chapter chapter that is selected
|
|
|
- * @param manga manga that belongs to chapter
|
|
|
*/
|
|
|
- private fun deleteChapter(chapter: Chapter, manga: Manga) {
|
|
|
- val source = sourceManager.get(manga.source) ?: return
|
|
|
+ private fun deleteChapter(chapter: RecentChapter) {
|
|
|
+ val source = sourceManager.get(chapter.manga.source) ?: return
|
|
|
downloadManager.queue.del(chapter)
|
|
|
- downloadManager.deleteChapter(source, manga, chapter)
|
|
|
+ downloadManager.deleteChapter(source, chapter.manga, chapter)
|
|
|
chapter.status = Download.NOT_DOWNLOADED
|
|
|
+ chapter.download = null
|
|
|
}
|
|
|
|
|
|
-
|
|
|
}
|