|
|
|
@ -10,58 +10,61 @@ import java.text.SimpleDateFormat
|
|
|
|
import java.util.*
|
|
|
|
import java.util.*
|
|
|
|
|
|
|
|
|
|
|
|
@MangaSourceParser("MANGAWORLD", "mangaworld.ac", "it")
|
|
|
|
@MangaSourceParser("MANGAWORLD", "mangaworld.ac", "it")
|
|
|
|
internal class MangaWorld(context: MangaLoaderContext) :
|
|
|
|
internal class MangaWorld(
|
|
|
|
PagedMangaParser(context, MangaSource.MANGAWORLD, pageSize = 16) {
|
|
|
|
context: MangaLoaderContext,
|
|
|
|
|
|
|
|
) : PagedMangaParser(context, MangaSource.MANGAWORLD, pageSize = 16) {
|
|
|
|
override val availableSortOrders: Set<SortOrder> = EnumSet.of(SortOrder.POPULARITY, SortOrder.ALPHABETICAL, SortOrder.NEWEST, SortOrder.ALPHABETICAL_DESC)
|
|
|
|
override val availableSortOrders: Set<SortOrder> =
|
|
|
|
|
|
|
|
EnumSet.of(SortOrder.POPULARITY, SortOrder.ALPHABETICAL, SortOrder.NEWEST, SortOrder.ALPHABETICAL_DESC)
|
|
|
|
|
|
|
|
|
|
|
|
override val configKeyDomain = ConfigKey.Domain("mangaworld.ac")
|
|
|
|
override val configKeyDomain = ConfigKey.Domain("mangaworld.ac")
|
|
|
|
|
|
|
|
|
|
|
|
override val isMultipleTagsSupported = true
|
|
|
|
override val isMultipleTagsSupported = true
|
|
|
|
|
|
|
|
|
|
|
|
override suspend fun getListPage(page: Int, filter: MangaListFilter?): List<Manga> {
|
|
|
|
override suspend fun getListPage(
|
|
|
|
val url = buildString {
|
|
|
|
page: Int,
|
|
|
|
append("https://")
|
|
|
|
filter: MangaListFilter?,
|
|
|
|
append(domain)
|
|
|
|
): List<Manga> {
|
|
|
|
append("/archive?")
|
|
|
|
val url =
|
|
|
|
when (filter) {
|
|
|
|
buildString {
|
|
|
|
is MangaListFilter.Search -> {
|
|
|
|
append("https://")
|
|
|
|
append("keyword=")
|
|
|
|
append(domain)
|
|
|
|
append(filter.query.urlEncoded())
|
|
|
|
append("/archive?")
|
|
|
|
}
|
|
|
|
when (filter) {
|
|
|
|
is MangaListFilter.Advanced -> {
|
|
|
|
is MangaListFilter.Search -> {
|
|
|
|
|
|
|
|
append("keyword=")
|
|
|
|
if (filter.tags.isNotEmpty()) {
|
|
|
|
append(filter.query.urlEncoded())
|
|
|
|
println(filter.tags)
|
|
|
|
|
|
|
|
val tags = filter.tags.joinToString("&") {it.key.substringAfter("archive?") }
|
|
|
|
|
|
|
|
append(tags)
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
when (filter.sortOrder) {
|
|
|
|
is MangaListFilter.Advanced -> {
|
|
|
|
SortOrder.POPULARITY -> append("&sort=most_read")
|
|
|
|
if (filter.tags.isNotEmpty()) {
|
|
|
|
SortOrder.ALPHABETICAL -> append("&sort=a-z")
|
|
|
|
filter.tags.joinTo(this, "&") { it.key.substringAfter("archive?") }
|
|
|
|
SortOrder.NEWEST -> append("&sort=newest")
|
|
|
|
}
|
|
|
|
SortOrder.ALPHABETICAL_DESC -> append("&sort=z-a")
|
|
|
|
|
|
|
|
else -> append("&sort=a-z")
|
|
|
|
when (filter.sortOrder) {
|
|
|
|
}
|
|
|
|
SortOrder.POPULARITY -> append("&sort=most_read")
|
|
|
|
when(filter.states.oneOrThrowIfMany()){
|
|
|
|
SortOrder.ALPHABETICAL -> append("&sort=a-z")
|
|
|
|
MangaState.ONGOING -> append("&status=ongoing")
|
|
|
|
SortOrder.NEWEST -> append("&sort=newest")
|
|
|
|
MangaState.FINISHED -> append("&status=completed")
|
|
|
|
SortOrder.ALPHABETICAL_DESC -> append("&sort=z-a")
|
|
|
|
MangaState.ABANDONED -> append("&status=dropped")
|
|
|
|
else -> append("&sort=a-z")
|
|
|
|
MangaState.PAUSED -> append("&status=paused")
|
|
|
|
}
|
|
|
|
else -> append("")
|
|
|
|
when (filter.states.oneOrThrowIfMany()) {
|
|
|
|
|
|
|
|
MangaState.ONGOING -> append("&status=ongoing")
|
|
|
|
|
|
|
|
MangaState.FINISHED -> append("&status=completed")
|
|
|
|
|
|
|
|
MangaState.ABANDONED -> append("&status=dropped")
|
|
|
|
|
|
|
|
MangaState.PAUSED -> append("&status=paused")
|
|
|
|
|
|
|
|
else -> Unit
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
null -> Unit
|
|
|
|
}
|
|
|
|
}
|
|
|
|
null -> append("")
|
|
|
|
append("&page=$page")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
append("&page=$page")
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
println("URL: $url")
|
|
|
|
|
|
|
|
val doc = webClient.httpGet(url).parseHtml()
|
|
|
|
val doc = webClient.httpGet(url).parseHtml()
|
|
|
|
return doc.select(".comics-grid .entry").map { div ->
|
|
|
|
return doc.select(".comics-grid .entry").map { div ->
|
|
|
|
val href = div.selectFirstOrThrow("a.thumb").attrAsRelativeUrl("href")
|
|
|
|
val href = div.selectFirstOrThrow("a.thumb").attrAsRelativeUrl("href")
|
|
|
|
val tags = div.select(".genres a[href*=/archive?genre=]").mapNotNullToSet { MangaTag(it.ownText(), it.attr("href"), source) }
|
|
|
|
val tags = div.select(".genres a[href*=/archive?genre=]")
|
|
|
|
|
|
|
|
.mapNotNullToSet { MangaTag(it.ownText().toTitleCase(sourceLocale), it.attr("href"), source) }
|
|
|
|
Manga(
|
|
|
|
Manga(
|
|
|
|
id = generateUid(href),
|
|
|
|
id = generateUid(href),
|
|
|
|
url = href,
|
|
|
|
url = href,
|
|
|
|
@ -72,7 +75,8 @@ internal class MangaWorld(context: MangaLoaderContext) :
|
|
|
|
rating = RATING_UNKNOWN,
|
|
|
|
rating = RATING_UNKNOWN,
|
|
|
|
tags = tags,
|
|
|
|
tags = tags,
|
|
|
|
author = div.selectFirst(".author a")?.text(),
|
|
|
|
author = div.selectFirst(".author a")?.text(),
|
|
|
|
state = when (div.selectFirst(".status a")?.text()) {
|
|
|
|
state =
|
|
|
|
|
|
|
|
when (div.selectFirst(".status a")?.text()) {
|
|
|
|
"In corso" -> MangaState.ONGOING
|
|
|
|
"In corso" -> MangaState.ONGOING
|
|
|
|
"Finito" -> MangaState.FINISHED
|
|
|
|
"Finito" -> MangaState.FINISHED
|
|
|
|
"Droppato" -> MangaState.ABANDONED
|
|
|
|
"Droppato" -> MangaState.ABANDONED
|
|
|
|
@ -90,7 +94,7 @@ internal class MangaWorld(context: MangaLoaderContext) :
|
|
|
|
return doc.select("div[aria-labelledby=genresDropdown] a").mapNotNullToSet {
|
|
|
|
return doc.select("div[aria-labelledby=genresDropdown] a").mapNotNullToSet {
|
|
|
|
MangaTag(
|
|
|
|
MangaTag(
|
|
|
|
key = it.attr("href"),
|
|
|
|
key = it.attr("href"),
|
|
|
|
title = it.text().trim(),
|
|
|
|
title = it.text().toTitleCase(sourceLocale),
|
|
|
|
source = source,
|
|
|
|
source = source,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@ -99,9 +103,15 @@ internal class MangaWorld(context: MangaLoaderContext) :
|
|
|
|
override suspend fun getDetails(manga: Manga): Manga {
|
|
|
|
override suspend fun getDetails(manga: Manga): Manga {
|
|
|
|
val doc = webClient.httpGet(manga.url.toAbsoluteUrl(domain)).parseHtml()
|
|
|
|
val doc = webClient.httpGet(manga.url.toAbsoluteUrl(domain)).parseHtml()
|
|
|
|
return manga.copy(
|
|
|
|
return manga.copy(
|
|
|
|
altTitle = doc.selectFirst(".meta-data .font-weight-bold:contains(Titoli alternativi:)")?.parent()?.ownText()?.substringAfter(": ")?.trim(),
|
|
|
|
altTitle =
|
|
|
|
|
|
|
|
doc.selectFirst(".meta-data .font-weight-bold:contains(Titoli alternativi:)")
|
|
|
|
|
|
|
|
?.parent()
|
|
|
|
|
|
|
|
?.ownText()
|
|
|
|
|
|
|
|
?.substringAfter(": ")
|
|
|
|
|
|
|
|
?.trim(),
|
|
|
|
description = doc.getElementById("noidungm")?.text().orEmpty(),
|
|
|
|
description = doc.getElementById("noidungm")?.text().orEmpty(),
|
|
|
|
chapters = doc.select(".chapters-wrapper .chapter a").mapChapters(reversed = true) { i, a ->
|
|
|
|
chapters =
|
|
|
|
|
|
|
|
doc.select(".chapters-wrapper .chapter a").mapChapters(reversed = true) { i, a ->
|
|
|
|
val url = a.attrAsRelativeUrl("href").toAbsoluteUrl(domain)
|
|
|
|
val url = a.attrAsRelativeUrl("href").toAbsoluteUrl(domain)
|
|
|
|
MangaChapter(
|
|
|
|
MangaChapter(
|
|
|
|
id = generateUid(url),
|
|
|
|
id = generateUid(url),
|
|
|
|
@ -109,7 +119,10 @@ internal class MangaWorld(context: MangaLoaderContext) :
|
|
|
|
number = i + 1,
|
|
|
|
number = i + 1,
|
|
|
|
url = url,
|
|
|
|
url = url,
|
|
|
|
scanlator = null,
|
|
|
|
scanlator = null,
|
|
|
|
uploadDate = SimpleDateFormat("dd MMMM yyyy", Locale.ITALIAN).tryParse(a.selectFirst(".chap-date")?.text()),
|
|
|
|
uploadDate =
|
|
|
|
|
|
|
|
SimpleDateFormat("dd MMMM yyyy", Locale.ITALIAN).tryParse(
|
|
|
|
|
|
|
|
a.selectFirst(".chap-date")?.text(),
|
|
|
|
|
|
|
|
),
|
|
|
|
branch = null,
|
|
|
|
branch = null,
|
|
|
|
source = source,
|
|
|
|
source = source,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
@ -130,4 +143,3 @@ internal class MangaWorld(context: MangaLoaderContext) :
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|