Add small sources, fix , rename sources , remove code not used, remove dead sources, typo.
parent
81004b53a9
commit
e43c9f6489
@ -0,0 +1,10 @@
|
|||||||
|
package org.koitharu.kotatsu.parsers.site.mangareader.en
|
||||||
|
|
||||||
|
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
||||||
|
import org.koitharu.kotatsu.parsers.MangaSourceParser
|
||||||
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
|
import org.koitharu.kotatsu.parsers.site.mangareader.MangaReaderParser
|
||||||
|
|
||||||
|
@MangaSourceParser("MANJANOON_EN", "Manjanoon", "en")
|
||||||
|
internal class Manjanoon(context: MangaLoaderContext) :
|
||||||
|
MangaReaderParser(context, MangaSource.MANJANOON_EN, "manjanoon.net", pageSize = 20, searchPageSize = 10)
|
||||||
@ -0,0 +1,12 @@
|
|||||||
|
package org.koitharu.kotatsu.parsers.site.mangareader.en
|
||||||
|
|
||||||
|
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
||||||
|
import org.koitharu.kotatsu.parsers.MangaSourceParser
|
||||||
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
|
import org.koitharu.kotatsu.parsers.site.mangareader.MangaReaderParser
|
||||||
|
|
||||||
|
@MangaSourceParser("RAISCANS", "Rai Scans", "en")
|
||||||
|
internal class RaiScans(context: MangaLoaderContext) :
|
||||||
|
MangaReaderParser(context, MangaSource.RAISCANS, "www.raiscans.com", pageSize = 20, searchPageSize = 10) {
|
||||||
|
override val listUrl = "/Series"
|
||||||
|
}
|
||||||
@ -1,139 +0,0 @@
|
|||||||
package org.koitharu.kotatsu.parsers.site.pt
|
|
||||||
|
|
||||||
import okhttp3.Headers
|
|
||||||
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
|
||||||
import org.koitharu.kotatsu.parsers.MangaSourceParser
|
|
||||||
import org.koitharu.kotatsu.parsers.PagedMangaParser
|
|
||||||
import org.koitharu.kotatsu.parsers.config.ConfigKey
|
|
||||||
import org.koitharu.kotatsu.parsers.model.*
|
|
||||||
import org.koitharu.kotatsu.parsers.network.UserAgents
|
|
||||||
import org.koitharu.kotatsu.parsers.util.*
|
|
||||||
import java.text.SimpleDateFormat
|
|
||||||
import java.util.*
|
|
||||||
|
|
||||||
@MangaSourceParser("GOLDENMANGA", "Golden Manga", "pt")
|
|
||||||
internal class GoldenManga(context: MangaLoaderContext) : PagedMangaParser(context, MangaSource.GOLDENMANGA, 36) {
|
|
||||||
|
|
||||||
override val sortOrders: Set<SortOrder> = EnumSet.of(SortOrder.ALPHABETICAL)
|
|
||||||
override val configKeyDomain = ConfigKey.Domain("goldenmanga.top")
|
|
||||||
override val headers: Headers = Headers.Builder()
|
|
||||||
.add("User-Agent", UserAgents.CHROME_MOBILE)
|
|
||||||
.build()
|
|
||||||
|
|
||||||
override suspend fun getListPage(
|
|
||||||
page: Int,
|
|
||||||
query: String?,
|
|
||||||
tags: Set<MangaTag>?,
|
|
||||||
sortOrder: SortOrder,
|
|
||||||
): List<Manga> {
|
|
||||||
val url = buildString {
|
|
||||||
append("https://")
|
|
||||||
append(domain)
|
|
||||||
if (!query.isNullOrEmpty()) {
|
|
||||||
append("/mangabr?pagina=")
|
|
||||||
append(page.toString())
|
|
||||||
append("&busca=")
|
|
||||||
append(query.urlEncoded())
|
|
||||||
} else {
|
|
||||||
append("/mangas")
|
|
||||||
append("?pagina=")
|
|
||||||
append(page.toString())
|
|
||||||
}
|
|
||||||
if (!tags.isNullOrEmpty()) {
|
|
||||||
append("&genero=")
|
|
||||||
for (tag in tags) {
|
|
||||||
append(tag.key)
|
|
||||||
append(",")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val doc = webClient.httpGet(url).parseHtml()
|
|
||||||
return doc.select("section.row div.mangas")
|
|
||||||
.map { div ->
|
|
||||||
val href = div.selectFirstOrThrow("a").attrAsRelativeUrl("href")
|
|
||||||
Manga(
|
|
||||||
id = generateUid(href),
|
|
||||||
title = div.selectFirstOrThrow("a h3").text(),
|
|
||||||
altTitle = null,
|
|
||||||
url = href,
|
|
||||||
publicUrl = href.toAbsoluteUrl(domain),
|
|
||||||
rating = RATING_UNKNOWN,
|
|
||||||
isNsfw = div.selectFirst("div.MangaAdulto") != null,
|
|
||||||
coverUrl = div.selectFirstOrThrow("img").attrAsAbsoluteUrl("src"),
|
|
||||||
tags = setOf(),
|
|
||||||
state = null,
|
|
||||||
author = null,
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun getTags(): Set<MangaTag> {
|
|
||||||
val doc = webClient.httpGet("https://$domain/mangas").parseHtml()
|
|
||||||
return doc.select("div.container a.btn.btn-warning ").mapNotNullToSet { a ->
|
|
||||||
MangaTag(
|
|
||||||
key = a.attr("href").substringAfterLast("="),
|
|
||||||
title = a.text(),
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun getDetails(manga: Manga): Manga {
|
|
||||||
val root = webClient.httpGet(manga.url.toAbsoluteUrl(domain)).parseHtml()
|
|
||||||
val dateFormat = SimpleDateFormat("(dd/MM/yyyy)", Locale.ENGLISH)
|
|
||||||
return manga.copy(
|
|
||||||
altTitle = null,
|
|
||||||
state = when (root.select("h5.cg_color")[3].select("a").text()) {
|
|
||||||
"ativo", "Ativo" -> MangaState.ONGOING
|
|
||||||
"Completo" -> MangaState.FINISHED
|
|
||||||
else -> null
|
|
||||||
},
|
|
||||||
tags = root.select("h5.cg_color")[0].select("a").mapNotNullToSet { a ->
|
|
||||||
|
|
||||||
if (a.text().isNullOrEmpty()) {
|
|
||||||
return@mapNotNullToSet null
|
|
||||||
} else {
|
|
||||||
MangaTag(
|
|
||||||
key = a.attr("href").substringAfterLast("="),
|
|
||||||
title = a.text().toTitleCase(),
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
author = root.select("h5.cg_color a")[1].text(),
|
|
||||||
description = root.getElementById("manga_capitulo_descricao")?.html(),
|
|
||||||
chapters = root.requireElementById("capitulos").select("li")
|
|
||||||
.mapChapters(reversed = true) { i, div ->
|
|
||||||
val href = div.selectFirstOrThrow("a").attrAsRelativeUrl("href")
|
|
||||||
val dateText = div.selectFirstOrThrow("div.col-sm-5 span").text()
|
|
||||||
val name = div.selectFirstOrThrow("div.col-sm-5").text().substringBeforeLast("(")
|
|
||||||
MangaChapter(
|
|
||||||
id = generateUid(href),
|
|
||||||
name = name,
|
|
||||||
number = i,
|
|
||||||
url = href,
|
|
||||||
scanlator = null,
|
|
||||||
uploadDate = dateFormat.tryParse(dateText),
|
|
||||||
branch = null,
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun getPages(chapter: MangaChapter): List<MangaPage> {
|
|
||||||
val fullUrl = chapter.url.toAbsoluteUrl(domain)
|
|
||||||
val doc = webClient.httpGet(fullUrl).parseHtml()
|
|
||||||
val root = doc.body().requireElementById("capitulos_images")
|
|
||||||
return root.select("img").map { img ->
|
|
||||||
val url = img.src()?.toRelativeUrl(domain) ?: img.parseFailed("Image src not found")
|
|
||||||
MangaPage(
|
|
||||||
id = generateUid(url),
|
|
||||||
url = url,
|
|
||||||
preview = null,
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,174 +0,0 @@
|
|||||||
package org.koitharu.kotatsu.parsers.site.pt
|
|
||||||
|
|
||||||
import org.jsoup.nodes.Element
|
|
||||||
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
|
||||||
import org.koitharu.kotatsu.parsers.MangaSourceParser
|
|
||||||
import org.koitharu.kotatsu.parsers.PagedMangaParser
|
|
||||||
import org.koitharu.kotatsu.parsers.config.ConfigKey
|
|
||||||
import org.koitharu.kotatsu.parsers.model.*
|
|
||||||
import org.koitharu.kotatsu.parsers.util.*
|
|
||||||
import org.koitharu.kotatsu.parsers.util.json.getStringOrNull
|
|
||||||
import org.koitharu.kotatsu.parsers.util.json.mapJSON
|
|
||||||
import java.text.SimpleDateFormat
|
|
||||||
import java.util.*
|
|
||||||
|
|
||||||
@MangaSourceParser("UNION_MANGAS", "Union Mangás", "pt")
|
|
||||||
class UnionMangasParser(context: MangaLoaderContext) : PagedMangaParser(context, MangaSource.UNION_MANGAS, 40) {
|
|
||||||
|
|
||||||
override val sortOrders: Set<SortOrder> = EnumSet.of(
|
|
||||||
SortOrder.ALPHABETICAL,
|
|
||||||
SortOrder.POPULARITY,
|
|
||||||
)
|
|
||||||
|
|
||||||
override val configKeyDomain = ConfigKey.Domain("unionmangasbr.top", "guimah.com")
|
|
||||||
|
|
||||||
override suspend fun getListPage(
|
|
||||||
page: Int,
|
|
||||||
query: String?,
|
|
||||||
tags: Set<MangaTag>?,
|
|
||||||
sortOrder: SortOrder,
|
|
||||||
): List<Manga> {
|
|
||||||
if (!query.isNullOrEmpty()) {
|
|
||||||
return if (page == searchPaginator.firstPage) {
|
|
||||||
search(query)
|
|
||||||
} else {
|
|
||||||
emptyList()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val tag = tags.oneOrThrowIfMany()
|
|
||||||
val url = urlBuilder()
|
|
||||||
.addPathSegment("lista-mangas")
|
|
||||||
.addPathSegment(
|
|
||||||
when {
|
|
||||||
tag != null -> tag.key
|
|
||||||
sortOrder == SortOrder.ALPHABETICAL -> "a-z"
|
|
||||||
else -> "visualizacoes"
|
|
||||||
},
|
|
||||||
).addPathSegment(page.toString())
|
|
||||||
val doc = webClient.httpGet(url.build()).parseHtml()
|
|
||||||
val root = doc.selectFirstOrThrow("div.tamanho-bloco-perfil")
|
|
||||||
return root.select(".lista-itens").map { div ->
|
|
||||||
val a = div.selectFirstOrThrow("a")
|
|
||||||
val img = div.selectFirstOrThrow("img")
|
|
||||||
val href = a.attrAsRelativeUrl("href")
|
|
||||||
Manga(
|
|
||||||
id = generateUid(href),
|
|
||||||
url = href,
|
|
||||||
publicUrl = a.attrAsAbsoluteUrl("href"),
|
|
||||||
title = div.selectLastOrThrow("a").text(),
|
|
||||||
coverUrl = img.attrAsAbsoluteUrl("src"),
|
|
||||||
altTitle = null,
|
|
||||||
rating = RATING_UNKNOWN,
|
|
||||||
tags = emptySet(),
|
|
||||||
description = div.selectLast("div")?.ownText(),
|
|
||||||
state = null,
|
|
||||||
author = null,
|
|
||||||
isNsfw = false,
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun getDetails(manga: Manga): Manga {
|
|
||||||
val doc = webClient.httpGet(manga.url.toAbsoluteUrl(domain)).parseHtml()
|
|
||||||
val dateFormat = SimpleDateFormat("dd/MM/yyyy", Locale.ROOT)
|
|
||||||
return manga.copy(
|
|
||||||
rating = doc.select("h2")
|
|
||||||
.find { it.ownText().startsWith('#') }
|
|
||||||
?.ownText()?.drop(1)?.toFloatOrNull()?.div(10f) ?: manga.rating,
|
|
||||||
largeCoverUrl = doc.selectFirst("img.img-thumbnail")?.attrAsAbsoluteUrlOrNull("src"),
|
|
||||||
description = doc.selectFirst(".panel-default")?.selectFirst(".panel-body")?.html(),
|
|
||||||
author = doc.tableValue("Autor")?.ownText(),
|
|
||||||
altTitle = doc.tableValue("Título(s) Alternativo(s)")?.ownText(),
|
|
||||||
state = when (doc.tableValue("Status")?.selectLast(".label")?.text()) {
|
|
||||||
"Completo" -> MangaState.FINISHED
|
|
||||||
"Ativo" -> MangaState.ONGOING
|
|
||||||
else -> null
|
|
||||||
},
|
|
||||||
tags = doc.tableValue("Gênero(s)")?.select("a")?.mapToSet {
|
|
||||||
it.toMangaTag()
|
|
||||||
} ?: manga.tags,
|
|
||||||
isNsfw = doc.selectFirst(".alert-danger")?.html()?.contains("18 anos") == true,
|
|
||||||
chapters = doc.select("div.row.capitulos").mapChapters(reversed = true) { i, div ->
|
|
||||||
val a = div.selectFirstOrThrow("a")
|
|
||||||
val href = a.attrAsRelativeUrl("href")
|
|
||||||
val title = a.text()
|
|
||||||
MangaChapter(
|
|
||||||
id = generateUid(href),
|
|
||||||
name = title,
|
|
||||||
number = i + 1,
|
|
||||||
url = href,
|
|
||||||
scanlator = div.selectLast("a")?.text()?.takeUnless { it == title },
|
|
||||||
uploadDate = dateFormat.tryParse(
|
|
||||||
a.nextElementSibling()?.text()?.removeSurrounding("(", ")"),
|
|
||||||
),
|
|
||||||
branch = null,
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun getPages(chapter: MangaChapter): List<MangaPage> {
|
|
||||||
val fullUrl = chapter.url.toAbsoluteUrl(domain)
|
|
||||||
val doc = webClient.httpGet(fullUrl).parseHtml()
|
|
||||||
return doc.body().selectOrThrow("img[pag]").mapNotNull { img ->
|
|
||||||
val href = img.attrAsRelativeUrl("src")
|
|
||||||
if (href.startsWith("/images/banner")) {
|
|
||||||
return@mapNotNull null
|
|
||||||
}
|
|
||||||
MangaPage(
|
|
||||||
id = generateUid(href),
|
|
||||||
url = href,
|
|
||||||
preview = null,
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun getTags(): Set<MangaTag> {
|
|
||||||
val doc = webClient.httpGet(urlBuilder().addPathSegment("lista-mangas").build()).parseHtml()
|
|
||||||
val ul = doc.body().selectFirstOrThrow(".nav-tabs").selectFirstOrThrow("ul.dropdown-menu")
|
|
||||||
return ul.select("li").mapToSet { li ->
|
|
||||||
li.selectFirstOrThrow("a").toMangaTag()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun search(query: String): List<Manga> {
|
|
||||||
val domain = domain
|
|
||||||
val json = webClient.httpGet(
|
|
||||||
urlBuilder()
|
|
||||||
.addPathSegments("assets/busca.php")
|
|
||||||
.addQueryParameter("nomeManga", query)
|
|
||||||
.build(),
|
|
||||||
).parseJson()
|
|
||||||
return json.getJSONArray("items").mapJSON { jo ->
|
|
||||||
val href = "/perfil/" + jo.getString("url")
|
|
||||||
Manga(
|
|
||||||
id = generateUid(href),
|
|
||||||
url = href,
|
|
||||||
publicUrl = href.toAbsoluteUrl(domain),
|
|
||||||
title = jo.getString("titulo"),
|
|
||||||
rating = RATING_UNKNOWN,
|
|
||||||
tags = emptySet(),
|
|
||||||
author = jo.getStringOrNull("autor"),
|
|
||||||
coverUrl = jo.getString("imagem"),
|
|
||||||
state = null,
|
|
||||||
isNsfw = false,
|
|
||||||
altTitle = null,
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun Element.tableValue(title: String): Element? {
|
|
||||||
return select("h4.media-heading")
|
|
||||||
.find { it.selectFirst("label.subtit-manga")?.text()?.contains(title, ignoreCase = true) == true }
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun Element.toMangaTag() = MangaTag(
|
|
||||||
title = text().toTitleCase(sourceLocale),
|
|
||||||
key = attr("href").removeSuffix('/').substringAfterLast('/'),
|
|
||||||
source = source,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
Loading…
Reference in New Issue