[viHentai] Add source

master
Draken 1 year ago
parent 7d5f0d3187
commit 2da8a4af84

@ -1 +1 @@
total: 1186 total: 1187

@ -208,79 +208,19 @@ internal class LxManga(context: MangaLoaderContext) : PagedMangaParser(context,
} }
} }
private fun availableTags() = arraySetOf( private suspend fun availableTags(): Set<MangaTag> {
MangaTag("3D", "3d", source), val url = "$domain/the-loai"
MangaTag("Adult", "adult", source), val doc = webClient.httpGet(url).parseHtml()
MangaTag("Animal girl", "animal-girl", source),
MangaTag("Artist", "artist", source), return doc.select("nav.grid.grid-cols-3.md\\:grid-cols-8 button").map { button ->
MangaTag("Bạo Dâm", "bao-dam", source), val key = button.attr("wire:click").substringAfterLast("'").substringBeforeLast("'")
MangaTag("CG", "cg", source), MangaTag(
MangaTag("Chơi Hai Lỗ", "choi-hai-lo", source), key = key,
MangaTag("Côn Trùng", "con-trung", source), title = button.select("span.text-ellipsis").text(),
MangaTag("Cosplay", "cosplay", source), source = source
MangaTag("Creampie", "creampie", source), )
MangaTag("Doujinshi", "doujinshi", source), }.toSet()
MangaTag("Drama", "drama", source), }
MangaTag("Đam Mỹ", "dam-my", source),
MangaTag("Đồ Bơi", "do-boi", source),
MangaTag("Elf", "elf", source),
MangaTag("Fantasy", "fantasy", source),
MangaTag("Furry", "furry", source),
MangaTag("Gangbang", "gangbang", source),
MangaTag("Gender Bender", "gender-bender", source),
MangaTag("Giáo Viên", "giao-vien", source),
MangaTag("Girl love", "girl-love", source),
MangaTag("Group", "group", source),
MangaTag("Guro", "guro", source),
MangaTag("Hài Hước", "hai-huoc", source),
MangaTag("Hãm Hiếp", "ham-hiep", source),
MangaTag("Harem", "harem", source),
MangaTag("Hầu Gái", "hau-gai", source),
MangaTag("Hậu Môn", "hau-mon", source),
MangaTag("Housewife", "housewife", source),
MangaTag("Kinh Dị", "kinh-di", source),
MangaTag("Kogal", "kogal", source),
MangaTag("Lão Già Dâm", "lao-gia-dam", source),
MangaTag("Lãng Mạn", "lang-man", source),
MangaTag("Loạn Luân", "loan-luan", source),
MangaTag("Loli", "loli", source),
MangaTag("LXHENTAI", "lxhentai", source),
MangaTag("Mang Thai", "mang-thai", source),
MangaTag("Manhwa", "manhwa", source),
MangaTag("Mắt Kính", "mat-kinh", source),
MangaTag("Mature", "mature", source),
MangaTag("Milf", "milf", source),
MangaTag("Mind Break", "mind-break", source),
MangaTag("Mind Control", "mind-control", source),
MangaTag("Monster Girl", "monster-girl", source),
MangaTag("Nàng Dâu", "nang-dau", source),
MangaTag("Netorare", "netorare", source),
MangaTag("Ngực Lớn", "nguc-lon", source),
MangaTag("Ngực Nhỏ", "nguc-nho", source),
MangaTag("Nô Lệ", "no-le", source),
MangaTag("NTR", "ntr", source),
MangaTag("Nữ Sinh", "nu-sinh", source),
MangaTag("Office lady", "office-lady", source),
MangaTag("OneShot", "oneshot", source),
MangaTag("Quái Vật", "quai-vat", source),
MangaTag("Series", "series", source),
MangaTag("Shota", "shota", source),
MangaTag("Supernatural", "supernatural", source),
MangaTag("Swinging", "swinging", source),
MangaTag("Thể Thao", "the-thao", source),
MangaTag("Three some", "three-some", source),
MangaTag("Thú Vật", "thu-vat", source),
MangaTag("Trap", "trap", source),
MangaTag("Truyện Comic", "truyen-comic", source),
MangaTag("Truyện Không Che", "truyen-khong-che", source),
MangaTag("Truyện Màu", "truyen-mau", source),
MangaTag("Truyện Ngắn", "truyen-ngan", source),
MangaTag("Virgin", "virgin", source),
MangaTag("Xúc Tua", "xuc-tua", source),
MangaTag("Y Tá", "y-ta", source),
MangaTag("Yaoi", "yaoi", source),
MangaTag("Yuri", "yuri", source),
)
private fun parseDateTime(dateStr: String): Long = runCatching { private fun parseDateTime(dateStr: String): Long = runCatching {
val parts = dateStr.split(' ') val parts = dateStr.split(' ')

@ -1,18 +1,21 @@
package org.koitharu.kotatsu.parsers.site.vi package org.koitharu.kotatsu.parsers.site.vi
import androidx.collection.arraySetOf import androidx.collection.arraySetOf
import androidx.collection.ArrayMap
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.PagedMangaParser import org.koitharu.kotatsu.parsers.PagedMangaParser
import org.koitharu.kotatsu.parsers.config.ConfigKey import org.koitharu.kotatsu.parsers.config.ConfigKey
import org.koitharu.kotatsu.parsers.model.* import org.koitharu.kotatsu.parsers.model.*
import org.koitharu.kotatsu.parsers.util.* import org.koitharu.kotatsu.parsers.util.*
import org.koitharu.kotatsu.parsers.util.suspendlazy.suspendLazy
import java.util.* import java.util.*
import java.text.SimpleDateFormat import java.text.SimpleDateFormat
@MangaSourceParser("TRUYENHENTAIVN", "TruyenHentaiVN", "vi", type = ContentType.HENTAI) @MangaSourceParser("TRUYENHENTAIVN", "TruyenHentaiVN", "vi", type = ContentType.HENTAI)
internal class TruyenHentaiVN(context: MangaLoaderContext) : PagedMangaParser(context, MangaParserSource.TRUYENHENTAIVN, 30) { internal class TruyenHentaiVN(context: MangaLoaderContext) : PagedMangaParser(context, MangaParserSource.TRUYENHENTAIVN, 30) {
private var cacheTags = suspendLazy(initializer = ::fetchTags)
override val configKeyDomain = ConfigKey.Domain("truyenhentaivn.live") override val configKeyDomain = ConfigKey.Domain("truyenhentaivn.live")
override fun onCreateConfig(keys: MutableCollection<ConfigKey<*>>) { override fun onCreateConfig(keys: MutableCollection<ConfigKey<*>>) {
@ -26,7 +29,7 @@ internal class TruyenHentaiVN(context: MangaLoaderContext) : PagedMangaParser(co
isSearchWithFiltersSupported = false isSearchWithFiltersSupported = false
) )
override suspend fun getFilterOptions() = MangaListFilterOptions(availableTags = getAvailableTags()) override suspend fun getFilterOptions() = MangaListFilterOptions(availableTags = cacheTags.get().values.toSet())
override val availableSortOrders: Set<SortOrder> = EnumSet.of(SortOrder.UPDATED) override val availableSortOrders: Set<SortOrder> = EnumSet.of(SortOrder.UPDATED)
@ -38,7 +41,7 @@ internal class TruyenHentaiVN(context: MangaLoaderContext) : PagedMangaParser(co
when { when {
!filter.tags.isNullOrEmpty() -> { !filter.tags.isNullOrEmpty() -> {
val tag = filter.tags.first() val tag = filter.tags.first()
append(tag.key) // Testing... append(tag.key)
if (page > 1) { if (page > 1) {
append("?page=") append("?page=")
append(page) append(page)
@ -156,12 +159,17 @@ internal class TruyenHentaiVN(context: MangaLoaderContext) : PagedMangaParser(co
} }
} }
private suspend fun getAvailableTags(): Set<MangaTag> { private suspend fun fetchTags(): Map<String, MangaTag> {
// Remove cached tags, avoid "Connection errors"
// Remake this function from site/vi/BlogTruyenVN.kt
val doc = webClient.httpGet("https://$domain").parseHtml() val doc = webClient.httpGet("https://$domain").parseHtml()
return doc.select("a.py-2[href^=/the-loai-]").mapNotNull { a -> val tagItems = doc.select("a.py-2[href^=/the-loai-]")
val key = a.attr("href") val tagMap = ArrayMap<String, MangaTag>(tagItems.size)
val title = a.text() for (tag in tagItems) {
MangaTag( key = key, title = title, source = source ) val key = tag.attr("href")
}.toSet() val title = tag.text()
tagMap[key] = MangaTag(key = key, title = title, source = source)
}
return tagMap
} }
} }

@ -0,0 +1,238 @@
package org.koitharu.kotatsu.parsers.site.vi
import androidx.collection.arraySetOf
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 java.util.*
@MangaSourceParser("VIHENTAI", "viHentai", "vi", type = ContentType.HENTAI)
internal class viHentai(context: MangaLoaderContext) : PagedMangaParser(context, MangaParserSource.VIHENTAI, 60) {
override val configKeyDomain = ConfigKey.Domain("vi-hentai.com")
override fun onCreateConfig(keys: MutableCollection<ConfigKey<*>>) {
super.onCreateConfig(keys)
keys.add(userAgentKey)
}
override val availableSortOrders: Set<SortOrder> = EnumSet.of(
SortOrder.ALPHABETICAL,
SortOrder.ALPHABETICAL_DESC,
SortOrder.UPDATED,
SortOrder.NEWEST,
SortOrder.POPULARITY,
)
override val filterCapabilities: MangaListFilterCapabilities
get() = MangaListFilterCapabilities(
isSearchSupported = true,
)
override suspend fun getFilterOptions() = MangaListFilterOptions(
availableTags = availableTags(),
availableStates = EnumSet.of(MangaState.ONGOING, MangaState.FINISHED),
)
override suspend fun getListPage(page: Int, order: SortOrder, filter: MangaListFilter): List<Manga> {
val url = buildString {
append("https://")
append(domain)
when {
!filter.query.isNullOrEmpty() -> {
append("/tim-kiem")
append("?filter[name]=")
append(filter.query.urlEncoded())
if (page > 1) {
append("&page=")
append(page)
}
append("&sort=")
append(
when (order) {
SortOrder.POPULARITY -> "-views"
SortOrder.UPDATED -> "-updated_at"
SortOrder.NEWEST -> "-created_at"
SortOrder.ALPHABETICAL -> "name"
SortOrder.ALPHABETICAL_DESC -> "-name"
else -> "-updated_at"
},
)
}
filter.tags.isNotEmpty() -> {
val tag = filter.tags.first()
append("/the-loai/")
append(tag.key)
append("?page=")
append(page)
}
else -> {
append("/danh-sach")
append("?sort=")
append(
when (order) {
SortOrder.POPULARITY -> "-views"
SortOrder.UPDATED -> "-updated_at"
SortOrder.NEWEST -> "-created_at"
SortOrder.ALPHABETICAL -> "name"
SortOrder.ALPHABETICAL_DESC -> "-name"
else -> "-updated_at"
},
)
append("&page=")
append(page)
}
}
if (filter.query.isNullOrEmpty()) {
append("&sort=")
when (order) {
SortOrder.POPULARITY -> append("-views")
SortOrder.UPDATED -> append("-updated_at")
SortOrder.NEWEST -> append("-created_at")
SortOrder.ALPHABETICAL -> append("name")
SortOrder.ALPHABETICAL_DESC -> append("-name")
else -> append("-updated_at")
}
}
if (filter.states.isNotEmpty()) {
append("&filter[status]=")
filter.states.forEach {
append(
when (it) {
MangaState.ONGOING -> "2,"
MangaState.FINISHED -> "1,"
else -> "1,2"
},
)
}
}
}
val doc = webClient.httpGet(url).parseHtml()
return doc.select("div.grid div.relative").map { div ->
val href = div.selectFirst("a[href^=/truyen/]")?.attrOrNull("href")
?: div.parseFailed("Không thể tìm thấy nguồn ảnh của Manga này!")
val coverUrl = div.selectFirstOrThrow("div.cover").let {
it.attrOrNull("data-bg") ?: it.attr("style").cssUrl()?.replace("s3.lxmanga.top", domain)
}.orEmpty()
Manga(
id = generateUid(href),
title = div.select("div.p-2 a.text-ellipsis").text(),
altTitle = null,
url = href,
publicUrl = href.toAbsoluteUrl(domain),
rating = RATING_UNKNOWN,
isNsfw = true,
coverUrl = coverUrl,
tags = setOf(),
state = null,
author = null,
source = source,
)
}
}
override suspend fun getDetails(manga: Manga): Manga {
val root = webClient.httpGet(manga.url.toAbsoluteUrl(domain)).parseHtml()
return manga.copy(
altTitle = root.selectLast("div.grow div:contains(Tên khác) span")?.textOrNull(),
state = when (root.selectFirst("div.mt-2:contains(Tình trạng) span.text-blue-500")?.text()) {
"Đang tiến hành" -> MangaState.ONGOING
"Đã hoàn thành" -> MangaState.FINISHED
else -> null
},
tags = root.select("div.mt-2:contains(Thể loại) a.bg-gray-500").mapToSet { a ->
MangaTag(
key = a.attr("href").removeSuffix('/').substringAfterLast('/'),
title = a.text(),
source = source,
)
},
author = root.selectFirst("div.mt-2:contains(Tác giả) span a")?.textOrNull(),
description = root.selectFirst("meta[name=description]")?.attrOrNull("content"),
chapters = root.select("div.justify-between ul.overflow-y-auto.overflow-x-hidden a")
.mapChapters(reversed = true) { i, a ->
val href = a.attrAsRelativeUrl("href")
val name = a.selectFirst("span.text-ellipsis")?.text().orEmpty()
val dateText = a.parent()?.selectFirst("span.timeago")?.attr("datetime").orEmpty()
val scanlator = root.selectFirst("div.mt-2:contains(Nhóm dịch) span a")?.textOrNull()
MangaChapter(
id = generateUid(href),
name = name,
number = i.toFloat(),
volume = 0,
url = href,
scanlator = scanlator,
uploadDate = parseDateTime(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()
return doc.select("div.text-center div.lazy")
.mapNotNull { div ->
val url = div.attr("data-src")
if (url.endsWith(".jpg", ignoreCase = true) ||
url.endsWith(".png", ignoreCase = true)
) {
MangaPage(
id = generateUid(url),
url = url,
preview = null,
source = source,
)
} else {
null
}
}
}
private fun parseDateTime(dateStr: String): Long = runCatching {
val parts = dateStr.split(' ')
val dateParts = parts[0].split('-')
val timeParts = parts[1].split(':')
val calendar = Calendar.getInstance()
calendar.set(
dateParts[0].toInt(),
dateParts[1].toInt() - 1,
dateParts[2].toInt(),
timeParts[0].toInt(),
timeParts[1].toInt(),
timeParts[2].toInt(),
)
calendar.timeInMillis
}.getOrDefault(0L)
private suspend fun availableTags(): Set<MangaTag> {
val doc = webClient.httpGet("https://$domain").parseHtml()
return doc.select("ul.grid.grid-cols-2 a").mapToSet { a ->
MangaTag(
key = a.attr("href").removeSuffix('/').substringAfterLast('/'),
title = a.text(),
source = source,
)
}
}
}
Loading…
Cancel
Save