Update CentralDeMangas url and refactor the code. (#1644)
Update CentralDeMangas url and refactor the code
This commit is contained in:
parent
657aeed536
commit
c87be4188f
|
@ -5,7 +5,7 @@ ext {
|
||||||
appName = 'Tachiyomi: Central de Mangás'
|
appName = 'Tachiyomi: Central de Mangás'
|
||||||
pkgNameSuffix = 'pt.centraldemangas'
|
pkgNameSuffix = 'pt.centraldemangas'
|
||||||
extClass = '.CentralDeMangas'
|
extClass = '.CentralDeMangas'
|
||||||
extVersionCode = 1
|
extVersionCode = 2
|
||||||
libVersion = '1.2'
|
libVersion = '1.2'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,12 +2,15 @@ package eu.kanade.tachiyomi.extension.pt.centraldemangas
|
||||||
|
|
||||||
import com.github.salomonbrys.kotson.array
|
import com.github.salomonbrys.kotson.array
|
||||||
import com.github.salomonbrys.kotson.get
|
import com.github.salomonbrys.kotson.get
|
||||||
|
import com.github.salomonbrys.kotson.string
|
||||||
|
import com.google.gson.JsonArray
|
||||||
import com.google.gson.JsonParser
|
import com.google.gson.JsonParser
|
||||||
import eu.kanade.tachiyomi.network.GET
|
import eu.kanade.tachiyomi.network.GET
|
||||||
import eu.kanade.tachiyomi.network.asObservableSuccess
|
import eu.kanade.tachiyomi.network.asObservableSuccess
|
||||||
import eu.kanade.tachiyomi.source.model.*
|
import eu.kanade.tachiyomi.source.model.*
|
||||||
import eu.kanade.tachiyomi.source.online.ParsedHttpSource
|
import eu.kanade.tachiyomi.source.online.ParsedHttpSource
|
||||||
import okhttp3.Headers
|
import okhttp3.Headers
|
||||||
|
import okhttp3.OkHttpClient
|
||||||
import okhttp3.Request
|
import okhttp3.Request
|
||||||
import okhttp3.Response
|
import okhttp3.Response
|
||||||
import org.jsoup.nodes.Document
|
import org.jsoup.nodes.Document
|
||||||
|
@ -22,63 +25,50 @@ class CentralDeMangas : ParsedHttpSource() {
|
||||||
|
|
||||||
override val name = "Central de Mangás"
|
override val name = "Central de Mangás"
|
||||||
|
|
||||||
override val baseUrl = "http://cdmnet.com.br"
|
override val baseUrl = "http://centraldemangas.online"
|
||||||
|
|
||||||
override val lang = "pt"
|
override val lang = "pt"
|
||||||
|
|
||||||
override val supportsLatest = true
|
override val supportsLatest = true
|
||||||
|
|
||||||
// Sometimes the site is very slow.
|
// Sometimes the site is very slow.
|
||||||
override val client =
|
override val client: OkHttpClient =
|
||||||
network.client.newBuilder()
|
network.client.newBuilder()
|
||||||
.connectTimeout(3, TimeUnit.MINUTES)
|
.connectTimeout(3, TimeUnit.MINUTES)
|
||||||
.readTimeout(3, TimeUnit.MINUTES)
|
.readTimeout(3, TimeUnit.MINUTES)
|
||||||
.writeTimeout(3, TimeUnit.MINUTES)
|
.writeTimeout(3, TimeUnit.MINUTES)
|
||||||
.build()
|
.build()
|
||||||
|
|
||||||
private val catalogHeaders = Headers.Builder().apply {
|
override fun headersBuilder(): Headers.Builder = Headers.Builder()
|
||||||
add("User-Agent", "Mozilla/5.0 (Windows NT 6.3; WOW64)")
|
.add("User-Agent", USER_AGENT)
|
||||||
add("Host", "cdmnet.com.br")
|
.add("Origin", baseUrl)
|
||||||
add("Referer", baseUrl)
|
.add("Referer", baseUrl)
|
||||||
}.build()
|
|
||||||
|
|
||||||
override fun popularMangaRequest(page: Int): Request = GET(baseUrl, catalogHeaders)
|
override fun popularMangaRequest(page: Int): Request = GET(baseUrl, headers)
|
||||||
|
|
||||||
override fun popularMangaSelector(): String = "div.ui.eight.doubling.stackable.cards div.card"
|
override fun popularMangaSelector(): String = "div.ui.eight.doubling.stackable.cards div.card"
|
||||||
|
|
||||||
override fun popularMangaFromElement(element: Element): SManga {
|
override fun popularMangaFromElement(element: Element): SManga = SManga.create().apply {
|
||||||
val manga = SManga.create()
|
title = element.select("div.content a").last().text()
|
||||||
|
url = element.select("div.content a").last().attr("href")
|
||||||
manga.thumbnail_url = element.select("div.ui.image a img")
|
thumbnail_url = element.select("div.ui.image a img").first()?.attr("src")
|
||||||
.first()?.attr("src")?.replace("60x80", "150x200")
|
|
||||||
element.select("div.content a").last().let {
|
|
||||||
manga.url = it.attr("href")
|
|
||||||
manga.title = it.text()
|
|
||||||
}
|
|
||||||
|
|
||||||
return manga
|
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun popularMangaNextPageSelector() = null
|
override fun popularMangaNextPageSelector(): String? = null
|
||||||
|
|
||||||
override fun latestUpdatesRequest(page: Int): Request = GET(baseUrl, catalogHeaders)
|
override fun latestUpdatesRequest(page: Int): Request = GET(baseUrl, headers)
|
||||||
|
|
||||||
override fun latestUpdatesSelector() = "div.ui.black.segment div.ui.divided.celled.list div.item"
|
override fun latestUpdatesSelector() = "div.ui.black.segment div.ui.divided.celled.list div.item"
|
||||||
|
|
||||||
override fun latestUpdatesFromElement(element: Element): SManga {
|
override fun latestUpdatesFromElement(element: Element): SManga = SManga.create().apply {
|
||||||
val manga = SManga.create()
|
title = element.select("div.content div.header a.popar").last().text()
|
||||||
|
url = element.select("div.content div.header a.popar").last().attr("href")
|
||||||
manga.thumbnail_url = element.select("div.ui.tiny.bordered.image a img")
|
thumbnail_url = element.select("div.ui.tiny.bordered.image a img").first()
|
||||||
.first()?.attr("src")?.replace("60x80", "150x200")
|
?.attr("src")
|
||||||
element.select("div.content div.header a.popar").last().let {
|
?.replace("60x80", "150x200")
|
||||||
manga.url = it.attr("href")
|
|
||||||
manga.title = it.text()
|
|
||||||
}
|
|
||||||
|
|
||||||
return manga
|
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun latestUpdatesNextPageSelector() = null
|
override fun latestUpdatesNextPageSelector(): String? = null
|
||||||
|
|
||||||
override fun fetchSearchManga(page: Int, query: String, filters: FilterList): Observable<MangasPage> {
|
override fun fetchSearchManga(page: Int, query: String, filters: FilterList): Observable<MangasPage> {
|
||||||
return client.newCall(searchMangaRequest(page, query, filters))
|
return client.newCall(searchMangaRequest(page, query, filters))
|
||||||
|
@ -88,48 +78,46 @@ class CentralDeMangas : ParsedHttpSource() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun searchMangaParse(response: Response): MangasPage = searchMangaParse(response, "")
|
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
|
||||||
|
val newHeaders = headers.newBuilder()
|
||||||
|
.set("X-Requested-With", "XMLHttpRequest")
|
||||||
|
.build()
|
||||||
|
|
||||||
private fun searchMangaParse(response: Response, query: String?): MangasPage {
|
return GET("$baseUrl/api/titulos", newHeaders)
|
||||||
val result = jsonParser.parse(response.body()!!.string()).array
|
}
|
||||||
|
|
||||||
|
private fun searchMangaParse(response: Response, query: String): MangasPage {
|
||||||
|
val result = response.asJsonArray()
|
||||||
|
|
||||||
val resultFiltered = result
|
val resultFiltered = result
|
||||||
.filter { it["title"].asString.contains(query ?: "", true) }
|
.filter { it["title"].string.contains(query, true) }
|
||||||
.map {
|
.map {
|
||||||
SManga.create().apply {
|
SManga.create().apply {
|
||||||
title = it["title"].asString
|
title = it["title"].string
|
||||||
url = it["url"].asString
|
url = it["url"].string
|
||||||
}
|
thumbnail_url = getCoverUrl(it["url"].string.substringAfterLast("/"))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return MangasPage(resultFiltered, false)
|
return MangasPage(resultFiltered, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
|
override fun searchMangaNextPageSelector(): String? = null
|
||||||
return GET("$baseUrl/api/titulos", headers)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun searchMangaSelector() = throw Exception("This method should not be called!")
|
override fun mangaDetailsParse(document: Document): SManga = SManga.create().apply {
|
||||||
|
|
||||||
override fun searchMangaFromElement(element: Element): SManga = throw Exception("This method should not be called!")
|
|
||||||
|
|
||||||
override fun searchMangaNextPageSelector() = ""
|
|
||||||
|
|
||||||
override fun mangaDetailsParse(document: Document): SManga {
|
|
||||||
val elementList = document.select("div.ui.black.segment div.ui.relaxed.list").first()
|
val elementList = document.select("div.ui.black.segment div.ui.relaxed.list").first()
|
||||||
|
|
||||||
return SManga.create().apply {
|
author = elementList.select("div.item:eq(3) div.content div.description").text()
|
||||||
author = elementList.select("div.item:eq(3) div.content div.description").text()
|
artist = elementList.select("div.item:eq(2) div.content div.description").text()
|
||||||
artist = elementList.select("div.item:eq(2) div.content div.description").text()
|
genre = elementList.select("div.item:eq(4) div.content div.description a")
|
||||||
genre = elementList.select("div.item:eq(4) div.content div.description a")
|
.joinToString { it.text() }
|
||||||
.joinToString { it.text() }
|
|
||||||
|
|
||||||
status = elementList.select("div.item:eq(6) div.content div.description")
|
status = elementList.select("div.item:eq(6) div.content div.description")
|
||||||
.text().orEmpty().let { parseStatus(it) }
|
.text().orEmpty().let { parseStatus(it) }
|
||||||
|
|
||||||
description = elementList.select("div.item:eq(0) div.content div.description").text()
|
description = elementList.select("div.item:eq(0) div.content div.description").text()
|
||||||
thumbnail_url = elementList.select("div.item:eq(0) div.content div.description img").attr("src")
|
thumbnail_url = elementList.select("div.item:eq(0) div.content div.description img")
|
||||||
}
|
.attr("src")
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun parseStatus(status: String) = when {
|
private fun parseStatus(status: String) = when {
|
||||||
|
@ -140,20 +128,20 @@ class CentralDeMangas : ParsedHttpSource() {
|
||||||
|
|
||||||
override fun chapterListParse(response: Response): List<SChapter> {
|
override fun chapterListParse(response: Response): List<SChapter> {
|
||||||
// Filter only manga chapters.
|
// Filter only manga chapters.
|
||||||
return super.chapterListParse(response).filter { !it.url.contains("/novel/") }
|
return super.chapterListParse(response)
|
||||||
|
.filter { !it.url.contains("/novel/") }
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun chapterListSelector() = "table.ui.small.compact.very.basic.table tbody tr:not(.active)"
|
override fun chapterListSelector() = "table.ui.small.compact.very.basic.table tbody tr:not(.active)"
|
||||||
|
|
||||||
override fun chapterFromElement(element: Element): SChapter {
|
override fun chapterFromElement(element: Element): SChapter = SChapter.create().apply {
|
||||||
val firstColumn = element.select("td:eq(0)")
|
val firstColumn = element.select("td:eq(0)")
|
||||||
val secondColumn = element.select("td:eq(1)")
|
val secondColumn = element.select("td:eq(1)")
|
||||||
|
|
||||||
return SChapter.create().apply {
|
url = firstColumn.select("a").first().attr("href")
|
||||||
url = firstColumn.select("a").first().attr("href")
|
name = firstColumn.select("a").first().text()
|
||||||
name = firstColumn.select("a").first().text()
|
date_upload = secondColumn.select("small").first()?.text()
|
||||||
date_upload = secondColumn.select("small").first()?.text()?.let { parseChapterDate(it) } ?: 0
|
?.let { parseChapterDate(it) } ?: 0
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun parseChapterDate(date: String) : Long {
|
private fun parseChapterDate(date: String) : Long {
|
||||||
|
@ -166,32 +154,48 @@ class CentralDeMangas : ParsedHttpSource() {
|
||||||
|
|
||||||
override fun pageListParse(document: Document): List<Page> {
|
override fun pageListParse(document: Document): List<Page> {
|
||||||
val script = document.select("script").last().data()
|
val script = document.select("script").last().data()
|
||||||
val urlSuffix = script.substringAfter(SCRIPT_URL_BEGIN).substringBefore(SCRIPT_URL_END)
|
val urlSuffix = script
|
||||||
val pages = script.substringAfter(SCRIPT_PAGES_BEGIN).substringBefore(SCRIPT_PAGES_END)
|
.substringAfter(SCRIPT_URL_BEGIN)
|
||||||
.replace("'", "").split(",")
|
.substringBefore(SCRIPT_URL_END)
|
||||||
|
val pages = script
|
||||||
|
.substringAfter(SCRIPT_PAGES_BEGIN)
|
||||||
|
.substringBefore(SCRIPT_PAGES_END)
|
||||||
|
.replace("'", "")
|
||||||
|
.split(",")
|
||||||
|
|
||||||
|
val chapterUrl = document.select("meta[property='og:url']").attr("content")
|
||||||
|
|
||||||
return pages
|
return pages
|
||||||
.mapIndexed { i, page -> Page(i, "", "$urlSuffix$page.jpg")}
|
.mapIndexed { i, page -> Page(i, chapterUrl, "$urlSuffix$page.jpg") }
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun imageRequest(page: Page): Request {
|
override fun imageRequest(page: Page): Request {
|
||||||
var imageHeaders = Headers.Builder().apply {
|
val imageHeaders = Headers.Builder()
|
||||||
add("Referer", baseUrl)
|
.add("Referer", page.url)
|
||||||
}
|
.build()
|
||||||
|
|
||||||
return GET(page.imageUrl!!, imageHeaders.build())
|
return GET(page.imageUrl!!, imageHeaders)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun imageUrlParse(document: Document) = ""
|
override fun imageUrlParse(document: Document) = ""
|
||||||
|
|
||||||
|
override fun searchMangaSelector() = throw Exception("This method should not be called!")
|
||||||
|
|
||||||
|
override fun searchMangaFromElement(element: Element): SManga = throw Exception("This method should not be called!")
|
||||||
|
|
||||||
|
private fun getCoverUrl(slug: String): String = "$COVER_CDN/150x200/$slug.jpg"
|
||||||
|
|
||||||
|
private fun Response.asJsonArray(): JsonArray = JSON_PARSER.parse(body()!!.string()).array
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
private const val USER_AGENT = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3497.92 Safari/537.36"
|
||||||
|
private const val COVER_CDN = "http://capas.centraldemangas.com.br"
|
||||||
|
|
||||||
private const val SCRIPT_URL_BEGIN = "var urlSulfix = '"
|
private const val SCRIPT_URL_BEGIN = "var urlSulfix = '"
|
||||||
private const val SCRIPT_URL_END = "';"
|
private const val SCRIPT_URL_END = "';"
|
||||||
private const val SCRIPT_PAGES_BEGIN = "var pages = ["
|
private const val SCRIPT_PAGES_BEGIN = "var pages = ["
|
||||||
private const val SCRIPT_PAGES_END = ",];"
|
private const val SCRIPT_PAGES_END = ",];"
|
||||||
|
|
||||||
val jsonParser by lazy {
|
val JSON_PARSER by lazy { JsonParser() }
|
||||||
JsonParser()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue