|  | @@ -17,6 +17,7 @@ import uy.kohesive.injekt.Injekt
 | 
	
		
			
				|  |  |  import uy.kohesive.injekt.api.get
 | 
	
		
			
				|  |  |  import java.util.concurrent.PriorityBlockingQueue
 | 
	
		
			
				|  |  |  import java.util.concurrent.atomic.AtomicInteger
 | 
	
		
			
				|  |  | +import kotlin.math.min
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  /**
 | 
	
		
			
				|  |  |   * Loader used to load chapters from an online source.
 | 
	
	
		
			
				|  | @@ -37,18 +38,20 @@ class HttpPageLoader(
 | 
	
		
			
				|  |  |       */
 | 
	
		
			
				|  |  |      private val subscriptions = CompositeSubscription()
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +    private val preloadSize = 4
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |      init {
 | 
	
		
			
				|  |  |          subscriptions += Observable.defer { Observable.just(queue.take().page) }
 | 
	
		
			
				|  |  | -            .filter { it.status == Page.QUEUE }
 | 
	
		
			
				|  |  | -            .concatMap { source.fetchImageFromCacheThenNet(it) }
 | 
	
		
			
				|  |  | -            .repeat()
 | 
	
		
			
				|  |  | -            .subscribeOn(Schedulers.io())
 | 
	
		
			
				|  |  | -            .subscribe({
 | 
	
		
			
				|  |  | -            }, { error ->
 | 
	
		
			
				|  |  | -                if (error !is InterruptedException) {
 | 
	
		
			
				|  |  | -                    Timber.e(error)
 | 
	
		
			
				|  |  | -                }
 | 
	
		
			
				|  |  | -            })
 | 
	
		
			
				|  |  | +                .filter { it.status == Page.QUEUE }
 | 
	
		
			
				|  |  | +                .concatMap { source.fetchImageFromCacheThenNet(it) }
 | 
	
		
			
				|  |  | +                .repeat()
 | 
	
		
			
				|  |  | +                .subscribeOn(Schedulers.io())
 | 
	
		
			
				|  |  | +                .subscribe({
 | 
	
		
			
				|  |  | +                }, { error ->
 | 
	
		
			
				|  |  | +                    if (error !is InterruptedException) {
 | 
	
		
			
				|  |  | +                        Timber.e(error)
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                })
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      /**
 | 
	
	
		
			
				|  | @@ -80,13 +83,13 @@ class HttpPageLoader(
 | 
	
		
			
				|  |  |       */
 | 
	
		
			
				|  |  |      override fun getPages(): Observable<List<ReaderPage>> {
 | 
	
		
			
				|  |  |          return chapterCache
 | 
	
		
			
				|  |  | -            .getPageListFromCache(chapter.chapter)
 | 
	
		
			
				|  |  | -            .onErrorResumeNext { source.fetchPageList(chapter.chapter) }
 | 
	
		
			
				|  |  | -            .map { pages ->
 | 
	
		
			
				|  |  | -                pages.mapIndexed { index, page -> // Don't trust sources and use our own indexing
 | 
	
		
			
				|  |  | -                    ReaderPage(index, page.url, page.imageUrl)
 | 
	
		
			
				|  |  | +                .getPageListFromCache(chapter.chapter)
 | 
	
		
			
				|  |  | +                .onErrorResumeNext { source.fetchPageList(chapter.chapter) }
 | 
	
		
			
				|  |  | +                .map { pages ->
 | 
	
		
			
				|  |  | +                    pages.mapIndexed { index, page -> // Don't trust sources and use our own indexing
 | 
	
		
			
				|  |  | +                        ReaderPage(index, page.url, page.imageUrl)
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  |                  }
 | 
	
		
			
				|  |  | -            }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      /**
 | 
	
	
		
			
				|  | @@ -110,29 +113,41 @@ class HttpPageLoader(
 | 
	
		
			
				|  |  |              val statusSubject = SerializedSubject(PublishSubject.create<Int>())
 | 
	
		
			
				|  |  |              page.setStatusSubject(statusSubject)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +            val queuedPages = mutableListOf<PriorityPage>()
 | 
	
		
			
				|  |  |              if (page.status == Page.QUEUE) {
 | 
	
		
			
				|  |  | -                queue.offer(PriorityPage(page, 1))
 | 
	
		
			
				|  |  | +                queuedPages += PriorityPage(page, 1).also { queue.offer(it) }
 | 
	
		
			
				|  |  |              }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -            preloadNextPages(page, 4)
 | 
	
		
			
				|  |  | +            queuedPages += preloadNextPages(page, preloadSize)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |              statusSubject.startWith(page.status)
 | 
	
		
			
				|  |  | +                    .doOnUnsubscribe {
 | 
	
		
			
				|  |  | +                        queuedPages.forEach {
 | 
	
		
			
				|  |  | +                            if (it.page.status == Page.QUEUE) {
 | 
	
		
			
				|  |  | +                                queue.remove(it)
 | 
	
		
			
				|  |  | +                            }
 | 
	
		
			
				|  |  | +                        }
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  | +                .subscribeOn(Schedulers.io())
 | 
	
		
			
				|  |  | +                .unsubscribeOn(Schedulers.io())
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      /**
 | 
	
		
			
				|  |  |       * Preloads the given [amount] of pages after the [currentPage] with a lower priority.
 | 
	
		
			
				|  |  | +     * @return a list of [PriorityPage] that were added to the [queue]
 | 
	
		
			
				|  |  |       */
 | 
	
		
			
				|  |  | -    private fun preloadNextPages(currentPage: ReaderPage, amount: Int) {
 | 
	
		
			
				|  |  | +    private fun preloadNextPages(currentPage: ReaderPage, amount: Int): List<PriorityPage> {
 | 
	
		
			
				|  |  |          val pageIndex = currentPage.index
 | 
	
		
			
				|  |  | -        val pages = currentPage.chapter.pages ?: return
 | 
	
		
			
				|  |  | -        if (pageIndex == pages.lastIndex) return
 | 
	
		
			
				|  |  | -        val nextPages = pages.subList(pageIndex + 1, Math.min(pageIndex + 1 + amount, pages.size))
 | 
	
		
			
				|  |  | -        for (nextPage in nextPages) {
 | 
	
		
			
				|  |  | -            if (nextPage.status == Page.QUEUE) {
 | 
	
		
			
				|  |  | -                queue.offer(PriorityPage(nextPage, 0))
 | 
	
		
			
				|  |  | -            }
 | 
	
		
			
				|  |  | -        }
 | 
	
		
			
				|  |  | +        val pages = currentPage.chapter.pages ?: return emptyList()
 | 
	
		
			
				|  |  | +        if (pageIndex == pages.lastIndex) return emptyList()
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        return pages
 | 
	
		
			
				|  |  | +                .subList(pageIndex + 1, min(pageIndex + 1 + amount, pages.size))
 | 
	
		
			
				|  |  | +                .mapNotNull {
 | 
	
		
			
				|  |  | +                    if (it.status == Page.QUEUE) {
 | 
	
		
			
				|  |  | +                        PriorityPage(it, 0).apply { queue.offer(this) }
 | 
	
		
			
				|  |  | +                    } else null
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      /**
 | 
	
	
		
			
				|  | @@ -148,7 +163,7 @@ class HttpPageLoader(
 | 
	
		
			
				|  |  |      /**
 | 
	
		
			
				|  |  |       * Data class used to keep ordering of pages in order to maintain priority.
 | 
	
		
			
				|  |  |       */
 | 
	
		
			
				|  |  | -    private data class PriorityPage(
 | 
	
		
			
				|  |  | +    private class PriorityPage(
 | 
	
		
			
				|  |  |              val page: ReaderPage,
 | 
	
		
			
				|  |  |              val priority: Int
 | 
	
		
			
				|  |  |      ): Comparable<PriorityPage> {
 |