Add a new Brazilian source (#17996)

Add a new Brazilian source.
This commit is contained in:
Alessandro Jean 2023-09-16 19:55:06 -03:00 committed by GitHub
parent 0f0396d16d
commit a7abd1a904
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 150 additions and 0 deletions

View File

@ -0,0 +1,2 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest />

View File

@ -0,0 +1,12 @@
apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
ext {
extName = 'Blackout Comics'
pkgNameSuffix = 'pt.blackoutcomics'
extClass = '.BlackoutComics'
extVersionCode = 1
isNsfw = true
}
apply from: "$rootDir/common.gradle"

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 19 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 68 KiB

View File

@ -0,0 +1,136 @@
package eu.kanade.tachiyomi.extension.pt.blackoutcomics
import eu.kanade.tachiyomi.network.GET
import eu.kanade.tachiyomi.network.interceptor.rateLimit
import eu.kanade.tachiyomi.source.model.FilterList
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 okhttp3.Headers
import okhttp3.HttpUrl.Companion.toHttpUrl
import okhttp3.OkHttpClient
import okhttp3.Request
import org.jsoup.nodes.Document
import org.jsoup.nodes.Element
import java.text.SimpleDateFormat
import java.util.Locale
import java.util.concurrent.TimeUnit
class BlackoutComics : ParsedHttpSource() {
override val name = "Blackout Comics"
override val baseUrl = "https://blackoutcomics.com"
override val lang = "pt-BR"
override val supportsLatest = true
override val client: OkHttpClient = network.cloudflareClient.newBuilder()
.rateLimit(1, 2, TimeUnit.SECONDS)
.build()
override fun headersBuilder(): Headers.Builder = Headers.Builder()
.add("Referer", "$baseUrl/")
override fun popularMangaRequest(page: Int): Request = GET(baseUrl, headers)
override fun popularMangaSelector(): String = "h3:contains(Mais Lidos) ~ div.anime-box a"
override fun popularMangaFromElement(element: Element): SManga = SManga.create().apply {
title = element.selectFirst("div.anime-blog p")!!.text()
thumbnail_url = element.selectFirst("img")!!.absUrl("src")
url = element.attr("href")
}
override fun popularMangaNextPageSelector(): String? = null
override fun latestUpdatesRequest(page: Int) = popularMangaRequest(page)
override fun latestUpdatesSelector() = "div:contains(Atualizados Recentemente) + div.row div.anime-blog"
override fun latestUpdatesFromElement(element: Element): SManga = SManga.create().apply {
title = element.selectFirst("a p")!!.text()
thumbnail_url = element.selectFirst("img")!!.absUrl("src")
url = element.selectFirst("a")!!.attr("href")
}
override fun latestUpdatesNextPageSelector(): String? = null
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
val url = "$baseUrl/comics".toHttpUrl().newBuilder()
.addQueryParameter("search", query)
.build()
return GET(url, headers)
}
override fun searchMangaSelector() = "section.anime div.anime-box"
override fun searchMangaFromElement(element: Element): SManga = SManga.create().apply {
title = element.selectFirst("p")!!.text()
thumbnail_url = element.selectFirst("img")!!.absUrl("src")
url = element.selectFirst("[onclick]")!!.attr("onclick")
.substringAfter("'")
.substringBeforeLast("'")
}
override fun searchMangaNextPageSelector(): String? = null
override fun mangaDetailsParse(document: Document): SManga = SManga.create().apply {
val infoElement = document.selectFirst("section.video")!!
title = infoElement.selectFirst("h2")!!.text()
author = infoElement.selectFirst("div.trailer-content p:contains(Autor:) b")!!.text()
artist = infoElement.selectFirst("div.trailer-content p:contains(Artista:) b")!!.text()
genre = infoElement.selectFirst("div.trailer-content p:contains(Genêros:)")!!.ownText()
status = infoElement.selectFirst("div.trailer-content p:contains(Status:) b")!!.text().toStatus()
description = infoElement.selectFirst("h3:contains(Descrição) + p")!!.text()
thumbnail_url = infoElement.selectFirst("img")!!.absUrl("src")
}
override fun chapterListSelector() = "section.relese h5"
override fun chapterFromElement(element: Element): SChapter = SChapter.create().apply {
name = element.selectFirst("a")!!.ownText()
date_upload = element.select("span:last-of-type").text().toDate()
url = element.selectFirst("a")!!.attr("href")
}
override fun pageListParse(document: Document): List<Page> {
return document.select("div.chapter-image canvas")
.mapIndexed { index, element ->
Page(index, document.location(), element.absUrl("data-src"))
}
}
override fun imageUrlParse(document: Document) = ""
override fun imageRequest(page: Page): Request {
val newHeaders = headersBuilder()
.set("Referer", page.url)
.build()
return GET(page.imageUrl!!, newHeaders)
}
private fun String.toDate(): Long {
return runCatching { DATE_FORMATTER.parse(trim())?.time }
.getOrNull() ?: 0L
}
private fun String.toStatus() = when (this) {
"Em Lançamento" -> SManga.ONGOING
"Completo" -> SManga.COMPLETED
"Cancelado" -> SManga.CANCELLED
"Em Espera" -> SManga.ON_HIATUS
else -> SManga.UNKNOWN
}
companion object {
private val DATE_FORMATTER by lazy {
SimpleDateFormat("dd/MM/yyyy", Locale("pt", "BR"))
}
}
}