Add Qi Scans (#10596)
* Qi Scans * Update src/en/qiscans/src/eu/kanade/tachiyomi/extension/en/qiscans/QiScans.kt Co-authored-by: bapeey <90949336+bapeey@users.noreply.github.com> * Update src/en/qiscans/src/eu/kanade/tachiyomi/extension/en/qiscans/QiScans.kt Co-authored-by: bapeey <90949336+bapeey@users.noreply.github.com> * Update src/en/qiscans/build.gradle Co-authored-by: bapeey <90949336+bapeey@users.noreply.github.com> * Update src/en/qiscans/src/eu/kanade/tachiyomi/extension/en/qiscans/QiScans.kt Co-authored-by: Luqman <16263232+Riztard@users.noreply.github.com> * Update QiScans.kt --------- Co-authored-by: bapeey <90949336+bapeey@users.noreply.github.com> Co-authored-by: Luqman <16263232+Riztard@users.noreply.github.com>
This commit is contained in:
parent
8da4bc0fd0
commit
7b843a9396
10
src/en/qiscans/build.gradle
Normal file
10
src/en/qiscans/build.gradle
Normal file
@ -0,0 +1,10 @@
|
||||
ext {
|
||||
extName = 'Qi Scans'
|
||||
extClass = '.QiScans'
|
||||
themePkg = 'iken'
|
||||
baseUrl = 'https://qiscans.org'
|
||||
overrideVersionCode = 0
|
||||
isNsfw = false
|
||||
}
|
||||
|
||||
apply from: "$rootDir/common.gradle"
|
BIN
src/en/qiscans/res/mipmap-hdpi/ic_launcher.png
Normal file
BIN
src/en/qiscans/res/mipmap-hdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 4.6 KiB |
BIN
src/en/qiscans/res/mipmap-mdpi/ic_launcher.png
Normal file
BIN
src/en/qiscans/res/mipmap-mdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.5 KiB |
BIN
src/en/qiscans/res/mipmap-xhdpi/ic_launcher.png
Normal file
BIN
src/en/qiscans/res/mipmap-xhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 6.8 KiB |
BIN
src/en/qiscans/res/mipmap-xxhdpi/ic_launcher.png
Normal file
BIN
src/en/qiscans/res/mipmap-xxhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 14 KiB |
BIN
src/en/qiscans/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
BIN
src/en/qiscans/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 23 KiB |
@ -0,0 +1,9 @@
|
||||
package eu.kanade.tachiyomi.extension.en.qiscans
|
||||
|
||||
import kotlinx.serialization.Serializable
|
||||
|
||||
@Serializable
|
||||
class GenreDto(val id: Int, val name: String)
|
||||
|
||||
@Serializable
|
||||
class PageDto(val url: String, val order: Int)
|
@ -0,0 +1,127 @@
|
||||
package eu.kanade.tachiyomi.extension.en.qiscans
|
||||
|
||||
import eu.kanade.tachiyomi.multisrc.iken.GenreFilter
|
||||
import eu.kanade.tachiyomi.multisrc.iken.Iken
|
||||
import eu.kanade.tachiyomi.multisrc.iken.SelectFilter
|
||||
import eu.kanade.tachiyomi.network.GET
|
||||
import eu.kanade.tachiyomi.network.interceptor.rateLimit
|
||||
import eu.kanade.tachiyomi.source.model.Filter
|
||||
import eu.kanade.tachiyomi.source.model.FilterList
|
||||
import eu.kanade.tachiyomi.source.model.MangasPage
|
||||
import eu.kanade.tachiyomi.source.model.Page
|
||||
import eu.kanade.tachiyomi.source.model.SChapter
|
||||
import keiyoushi.utils.parseAs
|
||||
import okhttp3.HttpUrl.Companion.toHttpUrl
|
||||
import okhttp3.Request
|
||||
import okhttp3.Response
|
||||
import rx.Observable
|
||||
import rx.schedulers.Schedulers
|
||||
import java.util.concurrent.TimeUnit
|
||||
|
||||
class QiScans : Iken(
|
||||
"Qi Scans",
|
||||
"en",
|
||||
"https://qiscans.org",
|
||||
"https://api.qiscans.org",
|
||||
) {
|
||||
|
||||
override val client = super.client.newBuilder()
|
||||
.rateLimit(3, 1, TimeUnit.SECONDS)
|
||||
.build()
|
||||
|
||||
override fun popularMangaRequest(page: Int): Request {
|
||||
val url = "$apiUrl/api/query".toHttpUrl().newBuilder().apply {
|
||||
addQueryParameter("page", page.toString())
|
||||
addQueryParameter("perPage", "18")
|
||||
addQueryParameter("orderBy", "totalViews")
|
||||
}.build()
|
||||
return GET(url, headers)
|
||||
}
|
||||
|
||||
override fun popularMangaParse(response: Response): MangasPage {
|
||||
return searchMangaParse(response)
|
||||
}
|
||||
|
||||
override fun latestUpdatesRequest(page: Int): Request {
|
||||
val url = "$apiUrl/api/query".toHttpUrl().newBuilder().apply { // 'query' instead of 'posts'
|
||||
addQueryParameter("page", page.toString())
|
||||
addQueryParameter("perPage", "18")
|
||||
addQueryParameter("orderBy", "updatedAt")
|
||||
}.build()
|
||||
return GET(url, headers)
|
||||
}
|
||||
|
||||
override fun pageListRequest(chapter: SChapter): Request {
|
||||
return GET(baseUrl + chapter.url, headersBuilder().add("rsc", "1").build())
|
||||
}
|
||||
|
||||
override fun pageListParse(response: Response): List<Page> {
|
||||
return response.body.string().lines()
|
||||
.mapNotNull { line ->
|
||||
val jsonStartIndex = line.indexOf('{').takeIf { it != -1 } ?: return@mapNotNull null
|
||||
val jsonString = line.substring(jsonStartIndex)
|
||||
try {
|
||||
jsonString.parseAs<PageDto>().takeIf { it.url.isNotEmpty() }
|
||||
} catch (e: Exception) {
|
||||
null
|
||||
}
|
||||
}
|
||||
.sortedBy { it.order }
|
||||
.mapIndexed { i, p -> Page(i, imageUrl = p.url) }
|
||||
}
|
||||
|
||||
private var genresList: List<Pair<String, String>> = emptyList()
|
||||
private var fetchGenresAttempts = 0
|
||||
|
||||
private fun fetchGenres() {
|
||||
try {
|
||||
val response = client.newCall(GET("$apiUrl/api/genres", headers)).execute()
|
||||
genresList = response.parseAs<List<GenreDto>>()
|
||||
.map { Pair(it.name, it.id.toString()) }
|
||||
} catch (e: Throwable) {
|
||||
} finally {
|
||||
fetchGenresAttempts++
|
||||
}
|
||||
}
|
||||
|
||||
override fun getFilterList(): FilterList {
|
||||
if (genresList.isEmpty() && fetchGenresAttempts < 3) {
|
||||
Observable.fromCallable { fetchGenres() }
|
||||
.subscribeOn(Schedulers.io())
|
||||
.subscribe()
|
||||
}
|
||||
|
||||
val filters = mutableListOf<Filter<*>>(
|
||||
SortFilter(),
|
||||
StatusFilter(),
|
||||
)
|
||||
|
||||
if (genresList.isNotEmpty()) {
|
||||
filters.add(GenreFilter(genresList))
|
||||
} else {
|
||||
filters.add(Filter.Header("Press 'Reset' to attempt to load genres"))
|
||||
}
|
||||
return FilterList(filters)
|
||||
}
|
||||
|
||||
private class SortFilter : SelectFilter(
|
||||
"Sort",
|
||||
"orderBy",
|
||||
listOf(
|
||||
Pair("Popularity", "totalViews"),
|
||||
Pair("Latest", "updatedAt"),
|
||||
),
|
||||
)
|
||||
|
||||
private class StatusFilter : SelectFilter(
|
||||
"Status",
|
||||
"seriesStatus",
|
||||
listOf(
|
||||
Pair("All", ""),
|
||||
Pair("Ongoing", "ONGOING"),
|
||||
Pair("Hiatus", "HIATUS"),
|
||||
Pair("Completed", "COMPLETED"),
|
||||
Pair("Dropped", "DROPPED"),
|
||||
),
|
||||
)
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user