NHentai | Fixed some images not showing for some titles (#6070)

* NHentai | Fixed some images not showing for some titles

* little

* Apply AwkwardPeak's suggestions

* comma
This commit is contained in:
KenjieDec 2024-11-17 20:07:49 +07:00 committed by Draff
parent 9419e9b07a
commit d4fcb880c4
No known key found for this signature in database
GPG Key ID: E8A89F3211677653
4 changed files with 111 additions and 70 deletions

View File

@ -1,7 +1,7 @@
ext { ext {
extName = 'NHentai' extName = 'NHentai'
extClass = '.NHFactory' extClass = '.NHFactory'
extVersionCode = 46 extVersionCode = 47
isNsfw = true isNsfw = true
} }

View File

@ -0,0 +1,37 @@
package eu.kanade.tachiyomi.extension.all.nhentai
import kotlinx.serialization.Serializable
@Serializable
class Hentai(
var id: Int,
val images: Images,
val media_id: String,
val tags: List<Tag>,
val title: Title,
val upload_date: Long,
val num_favorites: Long,
)
@Serializable
class Title(
var english: String? = null,
val japanese: String? = null,
val pretty: String? = null,
)
@Serializable
class Images(
val pages: List<Image>,
)
@Serializable
class Image(
val t: String,
)
@Serializable
class Tag(
val name: String,
val type: String,
)

View File

@ -1,63 +1,36 @@
package eu.kanade.tachiyomi.extension.all.nhentai package eu.kanade.tachiyomi.extension.all.nhentai
import org.jsoup.nodes.Document
import org.jsoup.nodes.Element import org.jsoup.nodes.Element
import java.text.SimpleDateFormat
object NHUtils { object NHUtils {
fun getArtists(document: Document): String { fun getArtists(data: Hentai): String {
val artists = document.select("#tags > div:nth-child(4) > span > a .name") val artists = data.tags.filter { it.type == "artist" }
return artists.joinToString(", ") { it.cleanTag() } return artists.joinToString(", ") { it.name }
} }
fun getGroups(document: Document): String? { fun getGroups(data: Hentai): String? {
val groups = document.select("#tags > div:nth-child(5) > span > a .name") val groups = data.tags.filter { it.type == "group" }
return if (groups.isNotEmpty()) { return groups.joinToString(", ") { it.name }.takeIf { it.isBlank() }
groups.joinToString(", ") { it.cleanTag() } }
} else {
null fun getTagDescription(data: Hentai): String {
val tags = data.tags.groupBy { it.type }
return buildString {
tags["category"]?.joinToString { it.name }?.let {
append("Categories: ", it, "\n")
}
tags["parody"]?.joinToString { it.name }?.let {
append("Parodies: ", it, "\n")
}
tags["character"]?.joinToString { it.name }?.let {
append("Characters: ", it, "\n\n")
}
} }
} }
fun getTagDescription(document: Document): String { fun getTags(data: Hentai): String {
val stringBuilder = StringBuilder() val artists = data.tags.filter { it.type == "tag" }
return artists.joinToString(", ") { it.name }
val categories = document.select("#tags > div:nth-child(7) > span > a .name")
if (categories.isNotEmpty()) {
stringBuilder.append("Categories: ")
stringBuilder.append(categories.joinToString(", ") { it.cleanTag() })
stringBuilder.append("\n\n")
}
val parodies = document.select("#tags > div:nth-child(1) > span > a .name")
if (parodies.isNotEmpty()) {
stringBuilder.append("Parodies: ")
stringBuilder.append(parodies.joinToString(", ") { it.cleanTag() })
stringBuilder.append("\n\n")
}
val characters = document.select("#tags > div:nth-child(2) > span > a .name")
if (characters.isNotEmpty()) {
stringBuilder.append("Characters: ")
stringBuilder.append(characters.joinToString(", ") { it.cleanTag() })
}
return stringBuilder.toString()
}
fun getTags(document: Document): String {
val tags = document.select("#tags > div:nth-child(3) > span > a .name")
return tags.map { it.cleanTag() }.sorted().joinToString(", ")
}
fun getNumPages(document: Document): String {
return document.select("#tags > div:nth-child(8) > span > a .name").first()!!.cleanTag()
}
fun getTime(document: Document): Long {
val timeString = document.toString().substringAfter("datetime=\"").substringBefore("\">").replace("T", " ")
return SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSSSSSZ").parse(timeString)?.time ?: 0L
} }
private fun Element.cleanTag(): String = text().replace(Regex("\\(.*\\)"), "").trim() private fun Element.cleanTag(): String = text().replace(Regex("\\(.*\\)"), "").trim()

View File

@ -6,10 +6,8 @@ import androidx.preference.ListPreference
import androidx.preference.PreferenceScreen import androidx.preference.PreferenceScreen
import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getArtists import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getArtists
import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getGroups import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getGroups
import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getNumPages
import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getTagDescription import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getTagDescription
import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getTags import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getTags
import eu.kanade.tachiyomi.extension.all.nhentai.NHUtils.getTime
import eu.kanade.tachiyomi.lib.randomua.addRandomUAPreferenceToScreen import eu.kanade.tachiyomi.lib.randomua.addRandomUAPreferenceToScreen
import eu.kanade.tachiyomi.lib.randomua.getPrefCustomUA import eu.kanade.tachiyomi.lib.randomua.getPrefCustomUA
import eu.kanade.tachiyomi.lib.randomua.getPrefUAType import eu.kanade.tachiyomi.lib.randomua.getPrefUAType
@ -27,6 +25,8 @@ import eu.kanade.tachiyomi.source.model.SManga
import eu.kanade.tachiyomi.source.model.UpdateStrategy import eu.kanade.tachiyomi.source.model.UpdateStrategy
import eu.kanade.tachiyomi.source.online.ParsedHttpSource import eu.kanade.tachiyomi.source.online.ParsedHttpSource
import eu.kanade.tachiyomi.util.asJsoup import eu.kanade.tachiyomi.util.asJsoup
import kotlinx.serialization.decodeFromString
import kotlinx.serialization.json.Json
import okhttp3.HttpUrl.Companion.toHttpUrl import okhttp3.HttpUrl.Companion.toHttpUrl
import okhttp3.OkHttpClient import okhttp3.OkHttpClient
import okhttp3.Request import okhttp3.Request
@ -36,6 +36,7 @@ import org.jsoup.nodes.Element
import rx.Observable import rx.Observable
import uy.kohesive.injekt.Injekt import uy.kohesive.injekt.Injekt
import uy.kohesive.injekt.api.get import uy.kohesive.injekt.api.get
import uy.kohesive.injekt.injectLazy
open class NHentai( open class NHentai(
override val lang: String, override val lang: String,
@ -50,6 +51,8 @@ open class NHentai(
override val supportsLatest = true override val supportsLatest = true
private val json: Json by injectLazy()
private val preferences: SharedPreferences by lazy { private val preferences: SharedPreferences by lazy {
Injekt.get<Application>().getSharedPreferences("source_$id", 0x0000) Injekt.get<Application>().getSharedPreferences("source_$id", 0x0000)
} }
@ -71,6 +74,7 @@ open class NHentai(
} }
private val shortenTitleRegex = Regex("""(\[[^]]*]|[({][^)}]*[)}])""") private val shortenTitleRegex = Regex("""(\[[^]]*]|[({][^)}]*[)}])""")
private val dataRegex = Regex("""JSON.parse\("([^*]*)"\)""")
private fun String.shortenTitle() = this.replace(shortenTitleRegex, "").trim() private fun String.shortenTitle() = this.replace(shortenTitleRegex, "").trim()
override fun setupPreferenceScreen(screen: PreferenceScreen) { override fun setupPreferenceScreen(screen: PreferenceScreen) {
@ -103,7 +107,7 @@ open class NHentai(
title = element.select("a > div").text().replace("\"", "").let { title = element.select("a > div").text().replace("\"", "").let {
if (displayFullTitle) it.trim() else it.shortenTitle() if (displayFullTitle) it.trim() else it.shortenTitle()
} }
thumbnail_url = element.select(".cover img").first()!!.let { img -> thumbnail_url = element.selectFirst(".cover img")!!.let { img ->
if (img.hasAttr("data-src")) img.attr("abs:data-src") else img.attr("abs:src") if (img.hasAttr("data-src")) img.attr("abs:data-src") else img.attr("abs:src")
} }
} }
@ -207,22 +211,25 @@ open class NHentai(
override fun searchMangaNextPageSelector() = latestUpdatesNextPageSelector() override fun searchMangaNextPageSelector() = latestUpdatesNextPageSelector()
override fun mangaDetailsParse(document: Document): SManga { override fun mangaDetailsParse(document: Document): SManga {
val fullTitle = document.select("#info > h1").text().replace("\"", "").trim() val script = document.selectFirst("script:containsData(JSON.parse)")!!.data()
val json = dataRegex.find(script)?.groupValues!![1]
val data = json.parseAs<Hentai>()
return SManga.create().apply { return SManga.create().apply {
title = if (displayFullTitle) fullTitle else fullTitle.shortenTitle() title = if (displayFullTitle) data.title.english ?: data.title.japanese ?: data.title.pretty!! else data.title.pretty ?: (data.title.english ?: data.title.japanese)!!.shortenTitle()
thumbnail_url = document.select("#cover > a > img").attr("data-src") thumbnail_url = document.select("#cover > a > img").attr("data-src")
status = SManga.COMPLETED status = SManga.COMPLETED
artist = getArtists(document) artist = getArtists(data)
author = getGroups(document) author = getGroups(data)
// Some people want these additional details in description // Some people want these additional details in description
description = "Full English and Japanese titles:\n" description = "Full English and Japanese titles:\n"
.plus("$fullTitle\n") .plus("${data.title.english}\n")
.plus("${document.select("div#info h2").text()}\n\n") .plus("${data.title.japanese}\n\n")
.plus("Pages: ${getNumPages(document)}\n") .plus("Pages: ${data.images.pages.size}\n")
.plus("Favorited by: ${document.select("div#info i.fa-heart ~ span span").text().removeSurrounding("(", ")")}\n") .plus("Favorited by: ${data.num_favorites}\n")
.plus(getTagDescription(document)) .plus(getTagDescription(data))
genre = getTags(document) genre = getTags(data)
update_strategy = UpdateStrategy.ONLY_FETCH_ONCE update_strategy = UpdateStrategy.ONLY_FETCH_ONCE
} }
} }
@ -231,11 +238,16 @@ open class NHentai(
override fun chapterListParse(response: Response): List<SChapter> { override fun chapterListParse(response: Response): List<SChapter> {
val document = response.asJsoup() val document = response.asJsoup()
val script = document.selectFirst("script:containsData(JSON.parse)")!!.data()
val json = dataRegex.find(script)?.groupValues!![1]
val data = json.parseAs<Hentai>()
return listOf( return listOf(
SChapter.create().apply { SChapter.create().apply {
name = "Chapter" name = "Chapter"
scanlator = getGroups(document) scanlator = getGroups(data)
date_upload = getTime(document) date_upload = data.upload_date * 1000
setUrlWithoutDomain(response.request.url.encodedPath) setUrlWithoutDomain(response.request.url.encodedPath)
}, },
) )
@ -246,11 +258,23 @@ open class NHentai(
override fun chapterListSelector() = throw UnsupportedOperationException() override fun chapterListSelector() = throw UnsupportedOperationException()
override fun pageListParse(document: Document): List<Page> { override fun pageListParse(document: Document): List<Page> {
val script = document.select("script:containsData(media_server)").first()!!.data() val script = document.selectFirst("script:containsData(media_server)")!!.data()
val mediaServer = Regex("""media_server\s*:\s*(\d+)""").find(script)?.groupValues!![1] val script2 = document.selectFirst("script:containsData(JSON.parse)")!!.data()
return document.select("div.thumbs a > img").mapIndexed { i, img -> val mediaServer = Regex("""media_server\s*:\s*(\d+)""").find(script)?.groupValues!![1]
Page(i, "", img.attr("abs:data-src").replace("t.nh", "i.nh").replace("t\\d+.nh".toRegex(), "i$mediaServer.nh").replace("t.", ".")) val json = dataRegex.find(script2)?.groupValues!![1]
val data = json.parseAs<Hentai>()
return data.images.pages.mapIndexed { i, image ->
Page(
i,
imageUrl = "${baseUrl.replace("https://", "https://i$mediaServer.")}/galleries/${data.media_id}/${i + 1}" +
when (image.t) {
"w" -> ".webp"
"p" -> ".png"
else -> ".jpg"
},
)
} }
} }
@ -303,6 +327,13 @@ open class NHentai(
), ),
) )
private inline fun <reified T> String.parseAs(): T {
return json.decodeFromString(
Regex("""\\u([0-9A-Fa-f]{4})""").replace(this) {
it.groupValues[1].toInt(16).toChar().toString()
},
)
}
private open class UriPartFilter(displayName: String, val vals: Array<Pair<String, String>>) : private open class UriPartFilter(displayName: String, val vals: Array<Pair<String, String>>) :
Filter.Select<String>(displayName, vals.map { it.first }.toTypedArray()) { Filter.Select<String>(displayName, vals.map { it.first }.toTypedArray()) {
fun toUriPart() = vals[state].second fun toUriPart() = vals[state].second