|
@@ -12,9 +12,10 @@ import eu.kanade.tachiyomi.data.source.SourceManager
|
|
|
import eu.kanade.tachiyomi.data.source.model.MangasPage
|
|
|
import eu.kanade.tachiyomi.data.source.online.LoginSource
|
|
|
import eu.kanade.tachiyomi.data.source.online.OnlineSource
|
|
|
+import eu.kanade.tachiyomi.data.source.online.OnlineSource.Filter
|
|
|
import eu.kanade.tachiyomi.ui.base.presenter.BasePresenter
|
|
|
-import eu.kanade.tachiyomi.util.RxPager
|
|
|
import rx.Observable
|
|
|
+import rx.Subscription
|
|
|
import rx.android.schedulers.AndroidSchedulers
|
|
|
import rx.schedulers.Schedulers
|
|
|
import rx.subjects.PublishSubject
|
|
@@ -64,14 +65,14 @@ class CataloguePresenter : BasePresenter<CatalogueFragment>() {
|
|
|
private set
|
|
|
|
|
|
/**
|
|
|
- * Pager containing a list of manga results.
|
|
|
+ * Active filters.
|
|
|
*/
|
|
|
- private var pager = RxPager<Manga>()
|
|
|
+ var filters: List<Filter> = emptyList()
|
|
|
|
|
|
/**
|
|
|
- * Last fetched page from network.
|
|
|
+ * Pager containing a list of manga results.
|
|
|
*/
|
|
|
- private var lastMangasPage: MangasPage? = null
|
|
|
+ private lateinit var pager: CataloguePager
|
|
|
|
|
|
/**
|
|
|
* Subject that initializes a list of manga.
|
|
@@ -84,27 +85,20 @@ class CataloguePresenter : BasePresenter<CatalogueFragment>() {
|
|
|
var isListMode: Boolean = false
|
|
|
private set
|
|
|
|
|
|
- companion object {
|
|
|
- /**
|
|
|
- * Id of the restartable that delivers a list of manga.
|
|
|
- */
|
|
|
- const val PAGER = 1
|
|
|
-
|
|
|
- /**
|
|
|
- * Id of the restartable that requests a page of manga from network.
|
|
|
- */
|
|
|
- const val REQUEST_PAGE = 2
|
|
|
-
|
|
|
- /**
|
|
|
- * Id of the restartable that initializes the details of manga.
|
|
|
- */
|
|
|
- const val GET_MANGA_DETAILS = 3
|
|
|
-
|
|
|
- /**
|
|
|
- * Key to save and restore [query] from a [Bundle].
|
|
|
- */
|
|
|
- const val QUERY_KEY = "query_key"
|
|
|
- }
|
|
|
+ /**
|
|
|
+ * Subscription for the pager.
|
|
|
+ */
|
|
|
+ private var pagerSubscription: Subscription? = null
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Subscription for one request from the pager.
|
|
|
+ */
|
|
|
+ private var pageSubscription: Subscription? = null
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Subscription to initialize manga details.
|
|
|
+ */
|
|
|
+ private var initializerSubscription: Subscription? = null
|
|
|
|
|
|
override fun onCreate(savedState: Bundle?) {
|
|
|
super.onCreate(savedState)
|
|
@@ -112,131 +106,138 @@ class CataloguePresenter : BasePresenter<CatalogueFragment>() {
|
|
|
source = getLastUsedSource()
|
|
|
|
|
|
if (savedState != null) {
|
|
|
- query = savedState.getString(QUERY_KEY, "")
|
|
|
+ query = savedState.getString(CataloguePresenter::query.name, "")
|
|
|
}
|
|
|
|
|
|
- startableLatestCache(GET_MANGA_DETAILS,
|
|
|
- { mangaDetailSubject.observeOn(Schedulers.io())
|
|
|
- .flatMap { Observable.from(it) }
|
|
|
- .filter { !it.initialized }
|
|
|
- .concatMap { getMangaDetailsObservable(it) }
|
|
|
- .onBackpressureBuffer()
|
|
|
- .observeOn(AndroidSchedulers.mainThread()) },
|
|
|
- { view, manga -> view.onMangaInitialized(manga) },
|
|
|
- { view, error -> Timber.e(error.message) })
|
|
|
-
|
|
|
add(prefs.catalogueAsList().asObservable()
|
|
|
.subscribe { setDisplayMode(it) })
|
|
|
|
|
|
- startableReplay(PAGER,
|
|
|
- { pager.results() },
|
|
|
- { view, pair -> view.onAddPage(pair.first, pair.second) })
|
|
|
-
|
|
|
- startableFirst(REQUEST_PAGE,
|
|
|
- { pager.request { page -> getMangasPageObservable(page + 1) } },
|
|
|
- { view, next -> },
|
|
|
- { view, error -> view.onAddPageError(error) })
|
|
|
-
|
|
|
- start(PAGER)
|
|
|
- start(REQUEST_PAGE)
|
|
|
+ restartPager()
|
|
|
}
|
|
|
|
|
|
override fun onSave(state: Bundle) {
|
|
|
- state.putString(QUERY_KEY, query)
|
|
|
+ state.putString(CataloguePresenter::query.name, query)
|
|
|
super.onSave(state)
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * Sets the display mode.
|
|
|
- *
|
|
|
- * @param asList whether the current mode is in list or not.
|
|
|
- */
|
|
|
- private fun setDisplayMode(asList: Boolean) {
|
|
|
- isListMode = asList
|
|
|
- if (asList) {
|
|
|
- stop(GET_MANGA_DETAILS)
|
|
|
- } else {
|
|
|
- start(GET_MANGA_DETAILS)
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Sets the active source and restarts the pager.
|
|
|
- *
|
|
|
- * @param source the new active source.
|
|
|
- */
|
|
|
- fun setActiveSource(source: OnlineSource) {
|
|
|
- prefs.lastUsedCatalogueSource().set(source.id)
|
|
|
- this.source = source
|
|
|
- restartPager()
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Restarts the request for the active source.
|
|
|
+ * Restarts the pager for the active source with the provided query and filters.
|
|
|
*
|
|
|
- * @param query the query, or null if searching popular manga.
|
|
|
+ * @param query the query.
|
|
|
+ * @param filters the list of active filters (for search mode).
|
|
|
*/
|
|
|
- fun restartPager(query: String = "") {
|
|
|
+ fun restartPager(query: String = this.query, filters: List<Filter> = this.filters) {
|
|
|
this.query = query
|
|
|
- stop(REQUEST_PAGE)
|
|
|
- lastMangasPage = null
|
|
|
+ this.filters = filters
|
|
|
|
|
|
if (!isListMode) {
|
|
|
- start(GET_MANGA_DETAILS)
|
|
|
+ subscribeToMangaInitializer()
|
|
|
}
|
|
|
- start(PAGER)
|
|
|
- start(REQUEST_PAGE)
|
|
|
+
|
|
|
+ // Create a new pager.
|
|
|
+ pager = CataloguePager(source, query, filters)
|
|
|
+
|
|
|
+ // Prepare the pager.
|
|
|
+ pagerSubscription?.let { remove(it) }
|
|
|
+ pagerSubscription = pager.results()
|
|
|
+ .subscribeReplay({ view, page ->
|
|
|
+ view.onAddPage(page.page, page.mangas)
|
|
|
+ }, { view, error ->
|
|
|
+ Timber.e(error, error.message)
|
|
|
+ })
|
|
|
+
|
|
|
+ // Request first page.
|
|
|
+ requestNext()
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Requests the next page for the active pager.
|
|
|
*/
|
|
|
fun requestNext() {
|
|
|
- if (hasNextPage()) {
|
|
|
- start(REQUEST_PAGE)
|
|
|
- }
|
|
|
+ if (!hasNextPage()) return
|
|
|
+
|
|
|
+ pageSubscription?.let { remove(it) }
|
|
|
+ pageSubscription = pager.requestNext { getPageTransformer(it) }
|
|
|
+ .subscribeFirst({ view, page ->
|
|
|
+ // Nothing to do when onNext is emitted.
|
|
|
+ }, CatalogueFragment::onAddPageError)
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Returns true if the last fetched page has a next page.
|
|
|
*/
|
|
|
fun hasNextPage(): Boolean {
|
|
|
- return lastMangasPage?.nextPageUrl != null
|
|
|
+ return pager.hasNextPage()
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * Retries the current request that failed.
|
|
|
+ * Sets the active source and restarts the pager.
|
|
|
+ *
|
|
|
+ * @param source the new active source.
|
|
|
*/
|
|
|
- fun retryPage() {
|
|
|
- start(REQUEST_PAGE)
|
|
|
+ fun setActiveSource(source: OnlineSource) {
|
|
|
+ prefs.lastUsedCatalogueSource().set(source.id)
|
|
|
+ this.source = source
|
|
|
+
|
|
|
+ restartPager(query = "", filters = emptyList())
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * Returns the observable of the network request for a page.
|
|
|
+ * Sets the display mode.
|
|
|
*
|
|
|
- * @param page the page number to request.
|
|
|
- * @return an observable of the network request.
|
|
|
+ * @param asList whether the current mode is in list or not.
|
|
|
*/
|
|
|
- private fun getMangasPageObservable(page: Int): Observable<List<Manga>> {
|
|
|
- val nextMangasPage = MangasPage(page)
|
|
|
- if (page != 1) {
|
|
|
- nextMangasPage.url = lastMangasPage!!.nextPageUrl!!
|
|
|
+ private fun setDisplayMode(asList: Boolean) {
|
|
|
+ isListMode = asList
|
|
|
+ if (asList) {
|
|
|
+ initializerSubscription?.let { remove(it) }
|
|
|
+ } else {
|
|
|
+ subscribeToMangaInitializer()
|
|
|
}
|
|
|
+ }
|
|
|
|
|
|
- val observable = if (query.isEmpty())
|
|
|
- source.fetchPopularManga(nextMangasPage)
|
|
|
- else
|
|
|
- source.fetchSearchManga(nextMangasPage, query)
|
|
|
+ /**
|
|
|
+ * Subscribes to the initializer of manga details and updates the view if needed.
|
|
|
+ */
|
|
|
+ private fun subscribeToMangaInitializer() {
|
|
|
+ initializerSubscription?.let { remove(it) }
|
|
|
+ initializerSubscription = mangaDetailSubject.observeOn(Schedulers.io())
|
|
|
+ .flatMap { Observable.from(it) }
|
|
|
+ .filter { !it.initialized }
|
|
|
+ .concatMap { getMangaDetailsObservable(it) }
|
|
|
+ .onBackpressureBuffer()
|
|
|
+ .observeOn(AndroidSchedulers.mainThread())
|
|
|
+ .subscribe({ manga ->
|
|
|
+ @Suppress("DEPRECATION")
|
|
|
+ view?.onMangaInitialized(manga)
|
|
|
+ }, { error ->
|
|
|
+ Timber.e(error, error.message)
|
|
|
+ })
|
|
|
+ .apply { add(this) }
|
|
|
+ }
|
|
|
|
|
|
+ /**
|
|
|
+ * Returns the function to apply to the observable of the list of manga from the source.
|
|
|
+ *
|
|
|
+ * @param observable the observable from the source.
|
|
|
+ * @return the function to apply.
|
|
|
+ */
|
|
|
+ fun getPageTransformer(observable: Observable<MangasPage>): Observable<MangasPage> {
|
|
|
return observable.subscribeOn(Schedulers.io())
|
|
|
- .doOnNext { lastMangasPage = it }
|
|
|
- .flatMap { Observable.from(it.mangas) }
|
|
|
- .map { networkToLocalManga(it) }
|
|
|
- .toList()
|
|
|
- .doOnNext { initializeMangas(it) }
|
|
|
+ .doOnNext { it.mangas.replace { networkToLocalManga(it) } }
|
|
|
+ .doOnNext { initializeMangas(it.mangas) }
|
|
|
.observeOn(AndroidSchedulers.mainThread())
|
|
|
}
|
|
|
|
|
|
+ /**
|
|
|
+ * Replaces an object in the list with another.
|
|
|
+ */
|
|
|
+ fun <T> MutableList<T>.replace(block: (T) -> T) {
|
|
|
+ forEachIndexed { i, obj ->
|
|
|
+ set(i, block(obj))
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
/**
|
|
|
* Returns a manga from the database for the given manga from network. It creates a new entry
|
|
|
* if the manga is not yet in the database.
|
|
@@ -354,4 +355,13 @@ class CataloguePresenter : BasePresenter<CatalogueFragment>() {
|
|
|
prefs.catalogueAsList().set(!isListMode)
|
|
|
}
|
|
|
|
|
|
+ /**
|
|
|
+ * Set the active filters for the current source.
|
|
|
+ *
|
|
|
+ * @param selectedFilters a list of active filters.
|
|
|
+ */
|
|
|
+ fun setSourceFilter(selectedFilters: List<Filter>) {
|
|
|
+ restartPager(filters = selectedFilters)
|
|
|
+ }
|
|
|
+
|
|
|
}
|