|
|
|
@ -1,157 +1,113 @@
|
|
|
|
package org.koitharu.kotatsu.parsers.site.vi
|
|
|
|
package org.koitharu.kotatsu.parsers.site.vi
|
|
|
|
|
|
|
|
|
|
|
|
import org.jsoup.nodes.Document
|
|
|
|
|
|
|
|
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
|
|
|
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
|
|
|
import org.koitharu.kotatsu.parsers.MangaSourceParser
|
|
|
|
import org.koitharu.kotatsu.parsers.MangaSourceParser
|
|
|
|
import org.koitharu.kotatsu.parsers.config.ConfigKey
|
|
|
|
import org.koitharu.kotatsu.parsers.config.ConfigKey
|
|
|
|
import org.koitharu.kotatsu.parsers.core.LegacyPagedMangaParser
|
|
|
|
import org.koitharu.kotatsu.parsers.core.LegacyPagedMangaParser
|
|
|
|
import org.koitharu.kotatsu.parsers.model.*
|
|
|
|
import org.koitharu.kotatsu.parsers.model.*
|
|
|
|
import org.koitharu.kotatsu.parsers.network.UserAgents
|
|
|
|
|
|
|
|
import org.koitharu.kotatsu.parsers.util.*
|
|
|
|
import org.koitharu.kotatsu.parsers.util.*
|
|
|
|
import java.util.*
|
|
|
|
import java.util.*
|
|
|
|
|
|
|
|
|
|
|
|
@MangaSourceParser("HENTAIVNBUZZ", "HentaiVn.buzz", "vi", type = ContentType.HENTAI)
|
|
|
|
@MangaSourceParser("HENTAIVNBUZZ", "HentaiVn.buzz", "vi", type = ContentType.HENTAI)
|
|
|
|
internal class HentaiVnBuzz(context: MangaLoaderContext) :
|
|
|
|
internal class HentaiVnBuzz(context: MangaLoaderContext) :
|
|
|
|
LegacyPagedMangaParser(context, MangaParserSource.HENTAIVNBUZZ, 24) {
|
|
|
|
LegacyPagedMangaParser(context, MangaParserSource.HENTAIVNBUZZ, 60) {
|
|
|
|
|
|
|
|
|
|
|
|
override val configKeyDomain = ConfigKey.Domain("hentaivn.email")
|
|
|
|
override val configKeyDomain = ConfigKey.Domain("hentaivn.beer")
|
|
|
|
|
|
|
|
|
|
|
|
override fun onCreateConfig(keys: MutableCollection<ConfigKey<*>>) {
|
|
|
|
override val availableSortOrders: Set<SortOrder> =
|
|
|
|
super.onCreateConfig(keys)
|
|
|
|
EnumSet.of(
|
|
|
|
keys.add(userAgentKey)
|
|
|
|
SortOrder.UPDATED,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
override val userAgentKey = ConfigKey.UserAgent(UserAgents.CHROME_DESKTOP)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
override val availableSortOrders: Set<SortOrder> = EnumSet.of(
|
|
|
|
|
|
|
|
SortOrder.NEWEST,
|
|
|
|
SortOrder.NEWEST,
|
|
|
|
SortOrder.POPULARITY,
|
|
|
|
SortOrder.POPULARITY,
|
|
|
|
SortOrder.UPDATED,
|
|
|
|
SortOrder.RATING,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
override val filterCapabilities: MangaListFilterCapabilities
|
|
|
|
override val filterCapabilities: MangaListFilterCapabilities
|
|
|
|
get() = MangaListFilterCapabilities(
|
|
|
|
get() = MangaListFilterCapabilities(
|
|
|
|
|
|
|
|
isMultipleTagsSupported = true,
|
|
|
|
|
|
|
|
isTagsExclusionSupported = true,
|
|
|
|
isSearchSupported = true,
|
|
|
|
isSearchSupported = true,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
override suspend fun getFilterOptions() = MangaListFilterOptions(
|
|
|
|
override suspend fun getFilterOptions() = MangaListFilterOptions(
|
|
|
|
availableTags = fetchTags(),
|
|
|
|
availableTags = fetchAvailableTags(),
|
|
|
|
availableStates = EnumSet.of(MangaState.ONGOING, MangaState.FINISHED),
|
|
|
|
availableStates = EnumSet.of(MangaState.ONGOING, MangaState.FINISHED),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
override suspend fun getListPage(page: Int, order: SortOrder, filter: MangaListFilter): List<Manga> {
|
|
|
|
override suspend fun getListPage(page: Int, order: SortOrder, filter: MangaListFilter): List<Manga> {
|
|
|
|
val url = when {
|
|
|
|
val url = buildString {
|
|
|
|
!filter.query.isNullOrEmpty() -> {
|
|
|
|
append("https://")
|
|
|
|
buildString {
|
|
|
|
append(domain)
|
|
|
|
append("/tim-kiem?key_word=")
|
|
|
|
append("/tim-kiem-nang-cao")
|
|
|
|
append(filter.query.urlEncoded())
|
|
|
|
|
|
|
|
if (page > 1) {
|
|
|
|
|
|
|
|
append("&page=")
|
|
|
|
|
|
|
|
append(page)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
filter.tags.isNotEmpty() -> {
|
|
|
|
append("?page=")
|
|
|
|
val tag = filter.tags.first()
|
|
|
|
|
|
|
|
buildString {
|
|
|
|
|
|
|
|
append("/the-loai/")
|
|
|
|
|
|
|
|
append(tag.key)
|
|
|
|
|
|
|
|
append("?")
|
|
|
|
|
|
|
|
when (order) {
|
|
|
|
|
|
|
|
SortOrder.NEWEST -> append("sort=0")
|
|
|
|
|
|
|
|
SortOrder.UPDATED -> append("sort=1")
|
|
|
|
|
|
|
|
SortOrder.POPULARITY -> append("sort=2")
|
|
|
|
|
|
|
|
else -> append("sort=0")
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (filter.states.isNotEmpty()) {
|
|
|
|
|
|
|
|
filter.states.forEach {
|
|
|
|
|
|
|
|
when (it) {
|
|
|
|
|
|
|
|
MangaState.ONGOING -> append("&is_full=0")
|
|
|
|
|
|
|
|
MangaState.FINISHED -> append("&is_full=1")
|
|
|
|
|
|
|
|
else -> append("")
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (page > 1) {
|
|
|
|
|
|
|
|
append("&page=")
|
|
|
|
|
|
|
|
append(page)
|
|
|
|
append(page)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
else -> {
|
|
|
|
append("&sort=")
|
|
|
|
buildString {
|
|
|
|
append(
|
|
|
|
append("/danh-sach/truyen-moi?")
|
|
|
|
|
|
|
|
when (order) {
|
|
|
|
when (order) {
|
|
|
|
SortOrder.NEWEST -> append("sort=0")
|
|
|
|
SortOrder.UPDATED -> "0"
|
|
|
|
SortOrder.UPDATED -> append("sort=1")
|
|
|
|
SortOrder.NEWEST -> "1"
|
|
|
|
SortOrder.POPULARITY -> append("sort=2")
|
|
|
|
SortOrder.POPULARITY -> "2"
|
|
|
|
else -> append("sort=0")
|
|
|
|
SortOrder.RATING -> "6"
|
|
|
|
}
|
|
|
|
else -> "0"
|
|
|
|
if (filter.states.isNotEmpty()) {
|
|
|
|
},
|
|
|
|
filter.states.forEach {
|
|
|
|
)
|
|
|
|
when (it) {
|
|
|
|
|
|
|
|
MangaState.ONGOING -> append("&is_full=0")
|
|
|
|
append("&status=")
|
|
|
|
MangaState.FINISHED -> append("&is_full=1")
|
|
|
|
append(
|
|
|
|
else -> append("")
|
|
|
|
when (filter.states.oneOrThrowIfMany()) {
|
|
|
|
}
|
|
|
|
MangaState.ONGOING -> "1"
|
|
|
|
}
|
|
|
|
MangaState.FINISHED -> "2"
|
|
|
|
}
|
|
|
|
else -> "0"
|
|
|
|
if (page > 1) {
|
|
|
|
},
|
|
|
|
append("&page=")
|
|
|
|
)
|
|
|
|
append(page)
|
|
|
|
|
|
|
|
}
|
|
|
|
if (filter.tags.isNotEmpty()) {
|
|
|
|
}
|
|
|
|
append("&category=")
|
|
|
|
|
|
|
|
append(filter.tags.joinToString(",") { it.key })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (filter.tagsExclude.isNotEmpty()) {
|
|
|
|
|
|
|
|
append("¬category=")
|
|
|
|
|
|
|
|
append(filter.tagsExclude.joinToString(",") { it.key })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
val fullUrl = url.toAbsoluteUrl(domain)
|
|
|
|
if (!filter.query.isNullOrEmpty()) {
|
|
|
|
val doc = webClient.httpGet(fullUrl).parseHtml()
|
|
|
|
clear()
|
|
|
|
return when {
|
|
|
|
|
|
|
|
!filter.query.isNullOrEmpty() -> parseSearchManga(doc)
|
|
|
|
append("https://")
|
|
|
|
filter.tags.isNotEmpty() -> parseSearchManga(doc)
|
|
|
|
append(domain)
|
|
|
|
else -> parseListManga(doc)
|
|
|
|
append("/tim-kiem?q=")
|
|
|
|
|
|
|
|
append(filter.query.urlEncoded())
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return@buildString // end of buildString
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private fun parseSearchManga(doc: Document): List<Manga> {
|
|
|
|
val doc = webClient.httpGet(url).parseHtml()
|
|
|
|
return doc.select(".story-item-list.d-flex.align-items-center.position-relative.mb-1").map { div ->
|
|
|
|
return doc.select("ul.list_grid.grid > li").map { element ->
|
|
|
|
val href = div.selectFirstOrThrow("a.story-item-list__image").attrAsRelativeUrl("href")
|
|
|
|
val aTag = element.selectFirstOrThrow("h3 a")
|
|
|
|
val coverUrl = div.selectFirst("img")?.attr("data-src")
|
|
|
|
val tags = element.select(".genre-item").mapToSet {
|
|
|
|
val title = div.selectFirst("img")?.attr("alt").orEmpty()
|
|
|
|
MangaTag(
|
|
|
|
Manga(
|
|
|
|
key = it.attr("href").substringAfterLast('-').substringBeforeLast('.'),
|
|
|
|
id = generateUid(href),
|
|
|
|
title = it.text().toTitleCase(sourceLocale),
|
|
|
|
title = title,
|
|
|
|
|
|
|
|
altTitles = emptySet(),
|
|
|
|
|
|
|
|
url = href,
|
|
|
|
|
|
|
|
publicUrl = href.toAbsoluteUrl(domain),
|
|
|
|
|
|
|
|
rating = RATING_UNKNOWN,
|
|
|
|
|
|
|
|
contentRating = if (isNsfwSource) ContentRating.ADULT else null,
|
|
|
|
|
|
|
|
coverUrl = coverUrl,
|
|
|
|
|
|
|
|
tags = emptySet(),
|
|
|
|
|
|
|
|
state = null,
|
|
|
|
|
|
|
|
authors = emptySet(),
|
|
|
|
|
|
|
|
source = source,
|
|
|
|
source = source,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private fun parseListManga(doc: Document): List<Manga> {
|
|
|
|
val href = aTag.attrAsRelativeUrl("href")
|
|
|
|
return doc.select(".story-item-list.d-flex.align-items-center.position-relative.mb-1").map { div ->
|
|
|
|
|
|
|
|
val href = div.selectFirstOrThrow("a.story-item-list__image").attrAsRelativeUrl("href")
|
|
|
|
|
|
|
|
val coverUrl = div.selectFirst("img")?.attr("data-src").orEmpty()
|
|
|
|
|
|
|
|
val title = div.selectFirst("img")?.attr("alt").orEmpty()
|
|
|
|
|
|
|
|
Manga(
|
|
|
|
Manga(
|
|
|
|
id = generateUid(href),
|
|
|
|
id = generateUid(href),
|
|
|
|
title = title,
|
|
|
|
title = aTag.text(),
|
|
|
|
altTitles = emptySet(),
|
|
|
|
altTitles = emptySet(),
|
|
|
|
url = href,
|
|
|
|
url = href,
|
|
|
|
publicUrl = href.toAbsoluteUrl(domain),
|
|
|
|
publicUrl = aTag.attrAsAbsoluteUrl("href"),
|
|
|
|
rating = RATING_UNKNOWN,
|
|
|
|
rating = RATING_UNKNOWN,
|
|
|
|
contentRating = if (isNsfwSource) ContentRating.ADULT else null,
|
|
|
|
contentRating = ContentRating.ADULT,
|
|
|
|
coverUrl = coverUrl,
|
|
|
|
coverUrl = element.selectFirst(".book_avatar a img")?.src(),
|
|
|
|
tags = emptySet(),
|
|
|
|
tags = tags,
|
|
|
|
state = null,
|
|
|
|
state = null,
|
|
|
|
authors = emptySet(),
|
|
|
|
authors = emptySet(),
|
|
|
|
source = source,
|
|
|
|
source = source,
|
|
|
|
@ -161,26 +117,29 @@ internal class HentaiVnBuzz(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()
|
|
|
|
val author = doc.select("p:contains(Tác giả:) a").text().nullIfEmpty()
|
|
|
|
val tags = doc.select("ul.list01 li").mapToSet {
|
|
|
|
return manga.copy(
|
|
|
|
|
|
|
|
authors = setOfNotNull(author),
|
|
|
|
|
|
|
|
tags = doc.select("div.mb-1 span a").mapToSet { element ->
|
|
|
|
|
|
|
|
MangaTag(
|
|
|
|
MangaTag(
|
|
|
|
key = element.attr("href").substringAfter("/the-loai/"),
|
|
|
|
key = it.attr("href").substringAfterLast('-').substringBeforeLast('.'),
|
|
|
|
title = element.text().substringBefore(',').trim(), // force trim before , symbol and space
|
|
|
|
title = it.text().toTitleCase(sourceLocale),
|
|
|
|
source = source,
|
|
|
|
source = source,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
description = null,
|
|
|
|
val author = doc.selectFirst("li.author a")?.textOrNull()
|
|
|
|
state = when (doc.select("p:contains(Trạng thái:) span").text()) {
|
|
|
|
|
|
|
|
"Đang ra" -> MangaState.ONGOING
|
|
|
|
return manga.copy(
|
|
|
|
|
|
|
|
altTitles = setOfNotNull(doc.selectFirst("h2.other-name")?.textOrNull()),
|
|
|
|
|
|
|
|
authors = setOfNotNull(author),
|
|
|
|
|
|
|
|
tags = tags,
|
|
|
|
|
|
|
|
description = doc.selectFirst("div.story-detail-info")?.html(),
|
|
|
|
|
|
|
|
state = when (doc.selectFirst(".status p.col-xs-9")?.text()) {
|
|
|
|
|
|
|
|
"Đang tiến hành" -> MangaState.ONGOING
|
|
|
|
"Hoàn thành" -> MangaState.FINISHED
|
|
|
|
"Hoàn thành" -> MangaState.FINISHED
|
|
|
|
else -> null
|
|
|
|
else -> null
|
|
|
|
},
|
|
|
|
},
|
|
|
|
chapters = doc.select("div.story-detail__list-chapter--list ul.list-unstyled li a")
|
|
|
|
chapters = doc.select("div.list_chapter div.works-chapter-item").mapChapters(reversed = true) { i, div ->
|
|
|
|
.mapIndexed { i, element ->
|
|
|
|
val a = div.selectFirstOrThrow("a")
|
|
|
|
val href = element.attrAsRelativeUrl("href")
|
|
|
|
val href = a.attrAsRelativeUrl("href")
|
|
|
|
val name = element.text().removePrefix("- ")
|
|
|
|
val name = a.text()
|
|
|
|
MangaChapter(
|
|
|
|
MangaChapter(
|
|
|
|
id = generateUid(href),
|
|
|
|
id = generateUid(href),
|
|
|
|
title = name,
|
|
|
|
title = name,
|
|
|
|
@ -188,7 +147,7 @@ internal class HentaiVnBuzz(context: MangaLoaderContext) :
|
|
|
|
volume = 0,
|
|
|
|
volume = 0,
|
|
|
|
url = href,
|
|
|
|
url = href,
|
|
|
|
scanlator = null,
|
|
|
|
scanlator = null,
|
|
|
|
uploadDate = 0,
|
|
|
|
uploadDate = 0L,
|
|
|
|
branch = null,
|
|
|
|
branch = null,
|
|
|
|
source = source,
|
|
|
|
source = source,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
@ -199,9 +158,8 @@ internal class HentaiVnBuzz(context: MangaLoaderContext) :
|
|
|
|
override suspend fun getPages(chapter: MangaChapter): List<MangaPage> {
|
|
|
|
override suspend fun getPages(chapter: MangaChapter): List<MangaPage> {
|
|
|
|
val fullUrl = chapter.url.toAbsoluteUrl(domain)
|
|
|
|
val fullUrl = chapter.url.toAbsoluteUrl(domain)
|
|
|
|
val doc = webClient.httpGet(fullUrl).parseHtml()
|
|
|
|
val doc = webClient.httpGet(fullUrl).parseHtml()
|
|
|
|
val imageUrls = doc.select("meta[property='og:image']").map { it.attr("content") }
|
|
|
|
return doc.select(".chapter_content img").map { img ->
|
|
|
|
val finalUrls = imageUrls.drop(1)
|
|
|
|
val url = img.requireSrc()
|
|
|
|
return finalUrls.map { url ->
|
|
|
|
|
|
|
|
MangaPage(
|
|
|
|
MangaPage(
|
|
|
|
id = generateUid(url),
|
|
|
|
id = generateUid(url),
|
|
|
|
url = url,
|
|
|
|
url = url,
|
|
|
|
@ -211,18 +169,15 @@ internal class HentaiVnBuzz(context: MangaLoaderContext) :
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private suspend fun fetchTags(): Set<MangaTag> {
|
|
|
|
private suspend fun fetchAvailableTags(): Set<MangaTag> {
|
|
|
|
val doc = webClient.httpGet("https://$domain/").parseHtml()
|
|
|
|
val doc = webClient.httpGet("https://$domain/tim-kiem-nang-cao").parseHtml()
|
|
|
|
val list = doc.select("ul.dropdown-menu.dropdown-menu-custom li a")
|
|
|
|
val elements = doc.select(".genre-item")
|
|
|
|
return list.mapToSet { tags ->
|
|
|
|
return elements.mapIndexed { i, element ->
|
|
|
|
val href = tags.attr("href")
|
|
|
|
|
|
|
|
val key = href.substringAfter("/the-loai/").substringBefore("/")
|
|
|
|
|
|
|
|
val title = tags.text()
|
|
|
|
|
|
|
|
MangaTag(
|
|
|
|
MangaTag(
|
|
|
|
key = key,
|
|
|
|
key = (i + 1).toString(),
|
|
|
|
title = title,
|
|
|
|
title = element.text().toTitleCase(sourceLocale),
|
|
|
|
source = source,
|
|
|
|
source = source,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}.toSet()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|