Add new niceoppai extention (#13552)
* Update Niceoppai.kt * update overrideVersionCode * remove comment code * Update MadaraGenerator.kt * Update Niceoppai.kt * formating code * remaster code and change type extention * remove niceoppai form madaraGenerator * Update src/th/niceoppai/build.gradle Co-authored-by: Vetle Ledaal <vetle.ledaal@gmail.com> Co-authored-by: Vetle Ledaal <vetle.ledaal@gmail.com>
|
@ -1,11 +0,0 @@
|
||||||
package eu.kanade.tachiyomi.extension.th.niceoppai
|
|
||||||
|
|
||||||
import eu.kanade.tachiyomi.multisrc.madara.Madara
|
|
||||||
import eu.kanade.tachiyomi.network.GET
|
|
||||||
import okhttp3.Request
|
|
||||||
|
|
||||||
class Niceoppai : Madara("Niceoppai", "https://www.niceoppai.net", "th") {
|
|
||||||
private fun pagePath(page: Int) = if (page > 1) "$page/" else ""
|
|
||||||
override fun popularMangaRequest(page: Int): Request = GET("$baseUrl/manga_list/all/any/most-popular/${pagePath(page)}", headers)
|
|
||||||
override fun latestUpdatesRequest(page: Int): Request = GET("$baseUrl/manga_list/all/any/last-updated/${pagePath(page)}", headers)
|
|
||||||
}
|
|
|
@ -355,7 +355,6 @@ class MadaraGenerator : ThemeSourceGenerator {
|
||||||
SingleLang("NeatManga", "https://neatmanga.com", "en", overrideVersionCode = 2),
|
SingleLang("NeatManga", "https://neatmanga.com", "en", overrideVersionCode = 2),
|
||||||
SingleLang("NekoBreaker Scan", "https://nekobreakerscan.com", "pt-BR", overrideVersionCode = 1),
|
SingleLang("NekoBreaker Scan", "https://nekobreakerscan.com", "pt-BR", overrideVersionCode = 1),
|
||||||
SingleLang("NekoScan", "https://nekoscan.com", "en", overrideVersionCode = 2),
|
SingleLang("NekoScan", "https://nekoscan.com", "en", overrideVersionCode = 2),
|
||||||
SingleLang("Niceoppai", "https://www.niceoppai.net", "th"),
|
|
||||||
SingleLang("Night Comic", "https://www.nightcomic.com", "en", overrideVersionCode = 1),
|
SingleLang("Night Comic", "https://www.nightcomic.com", "en", overrideVersionCode = 1),
|
||||||
SingleLang("Niji Translations", "https://niji-translations.com", "ar", overrideVersionCode = 1),
|
SingleLang("Niji Translations", "https://niji-translations.com", "ar", overrideVersionCode = 1),
|
||||||
SingleLang("Ninja Scan", "https://ninjascan.xyz", "pt-BR", overrideVersionCode = 1),
|
SingleLang("Ninja Scan", "https://ninjascan.xyz", "pt-BR", overrideVersionCode = 1),
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<manifest package="eu.kanade.tachiyomi.extension" />
|
|
@ -0,0 +1,12 @@
|
||||||
|
apply plugin: 'com.android.application'
|
||||||
|
apply plugin: 'kotlin-android'
|
||||||
|
|
||||||
|
ext {
|
||||||
|
extName = 'Niceoppai'
|
||||||
|
pkgNameSuffix = 'th.niceoppai'
|
||||||
|
extClass = '.Niceoppai'
|
||||||
|
extVersionCode = 25
|
||||||
|
isNsfw = true
|
||||||
|
}
|
||||||
|
|
||||||
|
apply from: "$rootDir/common.gradle"
|
Before Width: | Height: | Size: 5.6 KiB After Width: | Height: | Size: 5.6 KiB |
Before Width: | Height: | Size: 3.1 KiB After Width: | Height: | Size: 3.1 KiB |
Before Width: | Height: | Size: 8.2 KiB After Width: | Height: | Size: 8.2 KiB |
Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 14 KiB |
Before Width: | Height: | Size: 22 KiB After Width: | Height: | Size: 22 KiB |
Before Width: | Height: | Size: 84 KiB After Width: | Height: | Size: 84 KiB |
|
@ -0,0 +1,295 @@
|
||||||
|
package eu.kanade.tachiyomi.extension.th.niceoppai
|
||||||
|
|
||||||
|
import eu.kanade.tachiyomi.network.GET
|
||||||
|
import eu.kanade.tachiyomi.network.asObservableSuccess
|
||||||
|
import eu.kanade.tachiyomi.source.model.FilterList
|
||||||
|
import eu.kanade.tachiyomi.source.model.MangasPage
|
||||||
|
import eu.kanade.tachiyomi.source.model.Page
|
||||||
|
import eu.kanade.tachiyomi.source.model.SChapter
|
||||||
|
import eu.kanade.tachiyomi.source.model.SManga
|
||||||
|
import eu.kanade.tachiyomi.source.online.ParsedHttpSource
|
||||||
|
import eu.kanade.tachiyomi.util.asJsoup
|
||||||
|
import okhttp3.OkHttpClient
|
||||||
|
import okhttp3.Request
|
||||||
|
import org.jsoup.nodes.Document
|
||||||
|
import org.jsoup.nodes.Element
|
||||||
|
import rx.Observable
|
||||||
|
import java.text.ParseException
|
||||||
|
import java.text.SimpleDateFormat
|
||||||
|
import java.util.Calendar
|
||||||
|
import java.util.Locale
|
||||||
|
import java.util.concurrent.TimeUnit
|
||||||
|
|
||||||
|
class Niceoppai : ParsedHttpSource() {
|
||||||
|
|
||||||
|
private val dateFormat: SimpleDateFormat = SimpleDateFormat("MM dd, yyyy", Locale.US)
|
||||||
|
override val baseUrl: String = "https://www.niceoppai.net"
|
||||||
|
|
||||||
|
override val lang: String = "th"
|
||||||
|
override val name: String = "Niceoppai"
|
||||||
|
|
||||||
|
override val supportsLatest: Boolean = true
|
||||||
|
|
||||||
|
override val client: OkHttpClient = network.cloudflareClient.newBuilder()
|
||||||
|
.connectTimeout(1, TimeUnit.MINUTES)
|
||||||
|
.readTimeout(1, TimeUnit.MINUTES)
|
||||||
|
.writeTimeout(1, TimeUnit.MINUTES)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
// Popular
|
||||||
|
|
||||||
|
override fun popularMangaRequest(page: Int): Request {
|
||||||
|
return GET("$baseUrl/manga_list/all/any/most-popular-weekly/$page", headers)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun popularMangaSelector() = "div.nde"
|
||||||
|
|
||||||
|
override fun popularMangaFromElement(element: Element): SManga {
|
||||||
|
val manga = SManga.create()
|
||||||
|
|
||||||
|
manga.title = element.select("div.det a").text()
|
||||||
|
|
||||||
|
element.select("div.cvr").let {
|
||||||
|
manga.setUrlWithoutDomain(it.select("div.img_wrp a").attr("href"))
|
||||||
|
manga.thumbnail_url = it.select("img").attr("abs:src")
|
||||||
|
manga.initialized = false
|
||||||
|
}
|
||||||
|
|
||||||
|
return manga
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun popularMangaNextPageSelector() = "ul.pgg li a"
|
||||||
|
|
||||||
|
// Latest
|
||||||
|
|
||||||
|
override fun latestUpdatesRequest(page: Int): Request {
|
||||||
|
return GET("$baseUrl/manga_list/all/any/last-updated/$page", headers)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun latestUpdatesSelector() = popularMangaSelector()
|
||||||
|
|
||||||
|
override fun latestUpdatesFromElement(element: Element): SManga =
|
||||||
|
popularMangaFromElement(element)
|
||||||
|
|
||||||
|
override fun latestUpdatesNextPageSelector() = popularMangaNextPageSelector()
|
||||||
|
|
||||||
|
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request =
|
||||||
|
throw Exception("Unused")
|
||||||
|
|
||||||
|
override fun searchMangaSelector(): String = throw Exception("Unused")
|
||||||
|
|
||||||
|
override fun searchMangaFromElement(element: Element): SManga = throw Exception("Unused")
|
||||||
|
|
||||||
|
override fun searchMangaNextPageSelector(): String = throw Exception("Unused")
|
||||||
|
|
||||||
|
override fun fetchSearchManga(
|
||||||
|
page: Int,
|
||||||
|
query: String,
|
||||||
|
filters: FilterList
|
||||||
|
): Observable<MangasPage> {
|
||||||
|
val searchMethod = query.startsWith("http")
|
||||||
|
return client.newCall(
|
||||||
|
GET("$baseUrl/manga_list/category/$query/name-az/$page")
|
||||||
|
)
|
||||||
|
.asObservableSuccess()
|
||||||
|
.map {
|
||||||
|
val document = it.asJsoup()
|
||||||
|
val mangas: List<SManga> = if (searchMethod) {
|
||||||
|
listOf(
|
||||||
|
SManga.create().apply {
|
||||||
|
url = query.substringAfter(baseUrl)
|
||||||
|
title = document.select("h1.ttl").text()
|
||||||
|
thumbnail_url =
|
||||||
|
document.select("div.cvr_ara imag.cvr").attr("abs:src")
|
||||||
|
initialized = false
|
||||||
|
}
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
document.select(popularMangaSelector()).map { element ->
|
||||||
|
popularMangaFromElement(element)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
MangasPage(mangas, !searchMethod)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Manga summary page
|
||||||
|
|
||||||
|
private fun getStatus(status: String) = when (status) {
|
||||||
|
"ยังไม่จบ" -> SManga.ONGOING
|
||||||
|
"จบแล้ว" -> SManga.COMPLETED
|
||||||
|
else -> SManga.UNKNOWN
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun mangaDetailsParse(document: Document): SManga {
|
||||||
|
val infoElement = document.select("div.det").first()
|
||||||
|
|
||||||
|
return SManga.create().apply {
|
||||||
|
title = document.title() + infoElement.select("p")[2].ownText()
|
||||||
|
author = infoElement.select("p")[3].ownText()
|
||||||
|
artist = author
|
||||||
|
status = getStatus(infoElement.select("p")[10].ownText())
|
||||||
|
genre = infoElement.select("p")[6].select("a").joinToString { it.ownText() }
|
||||||
|
description = infoElement.select("p").first().ownText()
|
||||||
|
thumbnail_url = document.select("div.mng_ifo div.cvr_ara img").first().attr("abs:src")
|
||||||
|
initialized = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Chapters
|
||||||
|
|
||||||
|
private fun parseChapterDate(date: String?): Long {
|
||||||
|
date ?: return 0
|
||||||
|
|
||||||
|
fun SimpleDateFormat.tryParse(string: String): Long {
|
||||||
|
return try {
|
||||||
|
parse(string)?.time ?: 0
|
||||||
|
} catch (_: ParseException) {
|
||||||
|
0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return when {
|
||||||
|
// Handle 'yesterday' and 'today', using midnight
|
||||||
|
WordSet("yesterday", "يوم واحد").startsWith(date) -> {
|
||||||
|
Calendar.getInstance().apply {
|
||||||
|
add(Calendar.DAY_OF_MONTH, -1) // yesterday
|
||||||
|
set(Calendar.HOUR_OF_DAY, 0)
|
||||||
|
set(Calendar.MINUTE, 0)
|
||||||
|
set(Calendar.SECOND, 0)
|
||||||
|
set(Calendar.MILLISECOND, 0)
|
||||||
|
}.timeInMillis
|
||||||
|
}
|
||||||
|
WordSet("today").startsWith(date) -> {
|
||||||
|
Calendar.getInstance().apply {
|
||||||
|
set(Calendar.HOUR_OF_DAY, 0)
|
||||||
|
set(Calendar.MINUTE, 0)
|
||||||
|
set(Calendar.SECOND, 0)
|
||||||
|
set(Calendar.MILLISECOND, 0)
|
||||||
|
}.timeInMillis
|
||||||
|
}
|
||||||
|
WordSet("يومين").startsWith(date) -> {
|
||||||
|
Calendar.getInstance().apply {
|
||||||
|
add(Calendar.DAY_OF_MONTH, -2) // day before yesterday
|
||||||
|
set(Calendar.HOUR_OF_DAY, 0)
|
||||||
|
set(Calendar.MINUTE, 0)
|
||||||
|
set(Calendar.SECOND, 0)
|
||||||
|
set(Calendar.MILLISECOND, 0)
|
||||||
|
}.timeInMillis
|
||||||
|
}
|
||||||
|
WordSet("ago", "atrás", "önce", "قبل").endsWith(date) -> {
|
||||||
|
parseRelativeDate(date)
|
||||||
|
}
|
||||||
|
|
||||||
|
date.contains(Regex("""\d(st|nd|rd|th)""")) -> {
|
||||||
|
// Clean date (e.g. 5th December 2019 to 5 December 2019) before parsing it
|
||||||
|
date.split(" ").map {
|
||||||
|
if (it.contains(Regex("""\d\D\D"""))) {
|
||||||
|
it.replace(Regex("""\D"""), "")
|
||||||
|
} else {
|
||||||
|
it
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.let { dateFormat.tryParse(it.joinToString(" ")) }
|
||||||
|
}
|
||||||
|
else -> dateFormat.tryParse(date)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parses dates in this form:
|
||||||
|
// 21 horas ago
|
||||||
|
private fun parseRelativeDate(date: String): Long {
|
||||||
|
val number = Regex("""(\d+)""").find(date)?.value?.toIntOrNull() ?: return 0
|
||||||
|
val cal = Calendar.getInstance()
|
||||||
|
|
||||||
|
return when {
|
||||||
|
WordSet("hari", "gün", "jour", "día", "dia", "day", "วัน", "ngày", "giorni", "أيام").anyWordIn(date) -> cal.apply { add(Calendar.DAY_OF_MONTH, -number) }.timeInMillis
|
||||||
|
WordSet("jam", "saat", "heure", "hora", "hour", "ชั่วโมง", "giờ", "ore", "ساعة").anyWordIn(date) -> cal.apply { add(Calendar.HOUR, -number) }.timeInMillis
|
||||||
|
WordSet("menit", "dakika", "min", "minute", "minuto", "นาที", "دقائق").anyWordIn(date) -> cal.apply { add(Calendar.MINUTE, -number) }.timeInMillis
|
||||||
|
WordSet("detik", "segundo", "second", "วินาที").anyWordIn(date) -> cal.apply { add(Calendar.SECOND, -number) }.timeInMillis
|
||||||
|
WordSet("week").anyWordIn(date) -> cal.apply { add(Calendar.DAY_OF_MONTH, -number * 7) }.timeInMillis
|
||||||
|
WordSet("month").anyWordIn(date) -> cal.apply { add(Calendar.MONTH, -number) }.timeInMillis
|
||||||
|
WordSet("year").anyWordIn(date) -> cal.apply { add(Calendar.YEAR, -number) }.timeInMillis
|
||||||
|
else -> 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun chapterListSelector() = "ul.lst li.lng_"
|
||||||
|
|
||||||
|
override fun chapterFromElement(element: Element): SChapter = throw Exception("Unused")
|
||||||
|
|
||||||
|
private fun chapterFromElementWithIndex(element: Element, idx: Int, manga: SManga): SChapter {
|
||||||
|
val chapter = SChapter.create()
|
||||||
|
|
||||||
|
with(element) {
|
||||||
|
val btn = element.select("a.lst")
|
||||||
|
chapter.setUrlWithoutDomain(btn.attr("href"))
|
||||||
|
chapter.name = btn.select("b.val").text()
|
||||||
|
val dateText = btn.select("b.dte").text()
|
||||||
|
chapter.date_upload = parseChapterDate(dateText)
|
||||||
|
|
||||||
|
if (chapter.name.isEmpty()) {
|
||||||
|
chapter.chapter_number = 0.0f
|
||||||
|
} else {
|
||||||
|
val wordsChapter = chapter.name.replace("ตอนที่. ", "").split(" - ")
|
||||||
|
try {
|
||||||
|
chapter.chapter_number = wordsChapter[0].toFloat()
|
||||||
|
} catch (ex: NumberFormatException) {
|
||||||
|
chapter.chapter_number = (idx + 1).toFloat()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return chapter
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun fetchChapterList(manga: SManga): Observable<List<SChapter>> {
|
||||||
|
|
||||||
|
return client.newCall(GET("$baseUrl/${manga.url}"))
|
||||||
|
.asObservableSuccess()
|
||||||
|
.map {
|
||||||
|
val chList: List<SChapter>
|
||||||
|
val mangaDocument = it.asJsoup()
|
||||||
|
|
||||||
|
if (mangaDocument.select(chapterListSelector()).isEmpty()) {
|
||||||
|
manga.status = SManga.COMPLETED
|
||||||
|
val createdChapter = SChapter.create().apply {
|
||||||
|
url = manga.url
|
||||||
|
name = "Chapter 1"
|
||||||
|
chapter_number = 1.0f
|
||||||
|
}
|
||||||
|
chList = listOf(createdChapter)
|
||||||
|
} else {
|
||||||
|
chList =
|
||||||
|
mangaDocument.select(chapterListSelector()).mapIndexed { idx, Chapter ->
|
||||||
|
chapterFromElementWithIndex(Chapter, idx, manga)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
chList
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pages
|
||||||
|
|
||||||
|
override fun pageListParse(document: Document): List<Page> {
|
||||||
|
return document.select("div.mng_rdr div#image-container img").mapIndexed { i, img ->
|
||||||
|
if (img.hasAttr("data-src")) {
|
||||||
|
Page(i, "", img.attr("abs:data-src"))
|
||||||
|
} else {
|
||||||
|
Page(i, "", img.attr("abs:src"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun imageUrlParse(document: Document): String =
|
||||||
|
throw UnsupportedOperationException("Not used")
|
||||||
|
|
||||||
|
override fun getFilterList() = FilterList()
|
||||||
|
}
|
||||||
|
|
||||||
|
class WordSet(private vararg val words: String) {
|
||||||
|
fun anyWordIn(dateString: String): Boolean = words.any { dateString.contains(it, ignoreCase = true) }
|
||||||
|
fun startsWith(dateString: String): Boolean = words.any { dateString.startsWith(it, ignoreCase = true) }
|
||||||
|
fun endsWith(dateString: String): Boolean = words.any { dateString.endsWith(it, ignoreCase = true) }
|
||||||
|
}
|