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