Added mangaeden(it) extension (#9)
* Added mangaeden(it) extension * Not overriding id anymore * Fixed search selector for the case the query has no results * Minor modifications: today/yesterday management, null safety and code enhancements * Type and status are now multiselect * Genres, types and statuses now are functions * Renamed into Manga Eden, fixed missing cover imgs and checkbox filter now are shown at the end
This commit is contained in:
parent
948447f5ec
commit
b18e6f9f32
|
@ -0,0 +1,13 @@
|
||||||
|
apply plugin: 'com.android.application'
|
||||||
|
apply plugin: 'kotlin-android'
|
||||||
|
|
||||||
|
ext {
|
||||||
|
appName = 'Tachiyomi: Mangaeden'
|
||||||
|
pkgNameSuffix = "it.mangaeden"
|
||||||
|
extClass = '.Mangaeden'
|
||||||
|
extVersionCode = 1
|
||||||
|
extVersionSuffix = 1
|
||||||
|
libVersion = '1.0'
|
||||||
|
}
|
||||||
|
|
||||||
|
apply from: '../common.gradle'
|
|
@ -0,0 +1,204 @@
|
||||||
|
package eu.kanade.tachiyomi.extension.it.mangaeden
|
||||||
|
|
||||||
|
import eu.kanade.tachiyomi.network.GET
|
||||||
|
import eu.kanade.tachiyomi.source.model.*
|
||||||
|
import eu.kanade.tachiyomi.source.online.ParsedHttpSource
|
||||||
|
import okhttp3.HttpUrl
|
||||||
|
import okhttp3.Request
|
||||||
|
import org.jsoup.nodes.Document
|
||||||
|
import org.jsoup.nodes.Element
|
||||||
|
import java.text.ParseException
|
||||||
|
import java.text.SimpleDateFormat
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
class Mangaeden : ParsedHttpSource() {
|
||||||
|
|
||||||
|
override val name = "Manga Eden"
|
||||||
|
|
||||||
|
override val baseUrl = "http://www.mangaeden.com"
|
||||||
|
|
||||||
|
override val lang = "it"
|
||||||
|
|
||||||
|
override val supportsLatest = true
|
||||||
|
|
||||||
|
override fun latestUpdatesRequest(page: Int): Request = GET("$baseUrl/it/it-directory/?order=3&page=$page", headers)
|
||||||
|
|
||||||
|
override fun latestUpdatesSelector() = searchMangaSelector()
|
||||||
|
|
||||||
|
override fun latestUpdatesFromElement(element: Element): SManga = searchMangaFromElement(element)
|
||||||
|
|
||||||
|
override fun latestUpdatesNextPageSelector() = searchMangaNextPageSelector()
|
||||||
|
|
||||||
|
override fun popularMangaRequest(page: Int): Request = GET("$baseUrl/it/it-directory/?page=$page", headers)
|
||||||
|
|
||||||
|
override fun popularMangaSelector() = searchMangaSelector()
|
||||||
|
|
||||||
|
override fun popularMangaFromElement(element: Element): SManga = searchMangaFromElement(element)
|
||||||
|
|
||||||
|
override fun popularMangaNextPageSelector() = searchMangaNextPageSelector()
|
||||||
|
|
||||||
|
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
|
||||||
|
val url = HttpUrl.parse("$baseUrl/it/it-directory/").newBuilder().addQueryParameter("title", query)
|
||||||
|
(if (filters.isEmpty()) getFilterList() else filters).forEach { filter ->
|
||||||
|
when (filter) {
|
||||||
|
is StatusList -> filter.state
|
||||||
|
.filter { it.state }
|
||||||
|
.map { it.id.toString() }
|
||||||
|
.forEach { url.addQueryParameter("status", it) }
|
||||||
|
is Types -> filter.state
|
||||||
|
.filter { it.state }
|
||||||
|
.map { it.id.toString() }
|
||||||
|
.forEach { url.addQueryParameter("type", it) }
|
||||||
|
is GenreList -> filter.state
|
||||||
|
.filterNot { it.isIgnored() }
|
||||||
|
.forEach { genre -> url.addQueryParameter(if (genre.isIncluded()) "categoriesInc" else "categoriesExcl", genre.id) }
|
||||||
|
is TextField -> url.addQueryParameter(filter.key, filter.state)
|
||||||
|
is OrderBy -> filter.state?.let {
|
||||||
|
val sortId = it.index
|
||||||
|
url.addQueryParameter("order", if (it.ascending) "-$sortId" else "$sortId")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
url.addQueryParameter("page", page.toString())
|
||||||
|
return GET(url.toString(), headers)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun searchMangaSelector() = "table#mangaList > tbody > tr:has(td:gt(1))"
|
||||||
|
|
||||||
|
override fun searchMangaFromElement(element: Element) = SManga.create().apply {
|
||||||
|
element.select("td > a").first()?.let {
|
||||||
|
setUrlWithoutDomain(it.attr("href"))
|
||||||
|
title = it.text()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun searchMangaNextPageSelector() = "a:has(span.next)"
|
||||||
|
|
||||||
|
override fun mangaDetailsParse(document: Document) = SManga.create().apply {
|
||||||
|
val infos = document.select("div.rightbox")
|
||||||
|
|
||||||
|
author = infos.select("a[href^=/it/it-directory/?author]").first()?.text()
|
||||||
|
artist = infos.select("a[href^=/it/it-directory/?artist]").first()?.text()
|
||||||
|
genre = infos.select("a[href^=/it/it-directory/?categoriesInc]").map { it.text() }.joinToString()
|
||||||
|
description = document.select("h2#mangaDescription").text()
|
||||||
|
status = parseStatus(infos.select("h4:containsOwn(Stato)").first()?.nextSibling().toString())
|
||||||
|
val img = infos.select("div.mangaImage2 > img").first()?.attr("src")
|
||||||
|
if (!img.isNullOrBlank()) thumbnail_url = img.let { "http:$it" }
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun parseStatus(status: String) = when {
|
||||||
|
status.contains("In Corso", true) -> SManga.ONGOING
|
||||||
|
status.contains("Completato", true) -> SManga.COMPLETED
|
||||||
|
else -> SManga.UNKNOWN
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun chapterListSelector() = "div#leftContent > table > tbody > tr"
|
||||||
|
|
||||||
|
override fun chapterFromElement(element: Element) = SChapter.create().apply {
|
||||||
|
val a = element.select("a[href^=/it/it-manga/]").first()
|
||||||
|
|
||||||
|
setUrlWithoutDomain(a?.attr("href").orEmpty())
|
||||||
|
name = a?.select("b")?.first()?.text().orEmpty()
|
||||||
|
date_upload = element.select("td.chapterDate").first()?.text()?.let { parseChapterDate(it.trim()) } ?: 0L
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun parseChapterDate(date: String): Long =
|
||||||
|
if ("Oggi" in date) {
|
||||||
|
Calendar.getInstance().apply {
|
||||||
|
set(Calendar.HOUR_OF_DAY, 0)
|
||||||
|
set(Calendar.MINUTE, 0)
|
||||||
|
set(Calendar.SECOND, 0)
|
||||||
|
set(Calendar.MILLISECOND, 0)
|
||||||
|
}.timeInMillis
|
||||||
|
} else if ("Ieri" in date) {
|
||||||
|
Calendar.getInstance().apply {
|
||||||
|
add(Calendar.DATE, -1)
|
||||||
|
set(Calendar.HOUR_OF_DAY, 0)
|
||||||
|
set(Calendar.MINUTE, 0)
|
||||||
|
set(Calendar.SECOND, 0)
|
||||||
|
set(Calendar.MILLISECOND, 0)
|
||||||
|
}.timeInMillis
|
||||||
|
} else try {
|
||||||
|
SimpleDateFormat("d MMM yyyy", Locale.ITALIAN).parse(date).time
|
||||||
|
} catch (e: ParseException) {
|
||||||
|
0L
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun pageListParse(document: Document): List<Page> = mutableListOf<Page>().apply {
|
||||||
|
document.select("option[value^=/it/it-manga/]").forEach {
|
||||||
|
add(Page(size, "$baseUrl${it.attr("value")}"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun imageUrlParse(document: Document): String = document.select("a#nextA.next > img").first()?.attr("src").let { "http$it" }
|
||||||
|
|
||||||
|
private class NamedId(name: String, val id: Int) : Filter.CheckBox(name)
|
||||||
|
private class Genre(name: String, val id: String) : Filter.TriState(name)
|
||||||
|
private class TextField(name: String, val key: String) : Filter.Text(name)
|
||||||
|
private class OrderBy : Filter.Sort("Ordina per", arrayOf("Titolo manga", "Visite", "Capitoli", "Ultimo capitolo"),
|
||||||
|
Filter.Sort.Selection(1, false))
|
||||||
|
|
||||||
|
private class StatusList(statuses: List<NamedId>) : Filter.Group<NamedId>("Stato", statuses)
|
||||||
|
private class Types(types: List<NamedId>) : Filter.Group<NamedId>("Tipo", types)
|
||||||
|
private class GenreList(genres: List<Genre>) : Filter.Group<Genre>("Generi", genres)
|
||||||
|
|
||||||
|
override fun getFilterList() = FilterList(
|
||||||
|
TextField("Autore", "author"),
|
||||||
|
TextField("Artista", "artist"),
|
||||||
|
OrderBy(),
|
||||||
|
Types(types()),
|
||||||
|
StatusList(statuses()),
|
||||||
|
GenreList(genres())
|
||||||
|
)
|
||||||
|
|
||||||
|
private fun types() = listOf(
|
||||||
|
NamedId("Japanese Manga", 0),
|
||||||
|
NamedId("Korean Manhwa", 1),
|
||||||
|
NamedId("Chinese Manhua", 2),
|
||||||
|
NamedId("Comic", 3),
|
||||||
|
NamedId("Doujinshi", 4)
|
||||||
|
)
|
||||||
|
|
||||||
|
private fun statuses() = listOf(
|
||||||
|
NamedId("In corso", 1),
|
||||||
|
NamedId("Completato", 2),
|
||||||
|
NamedId("Sospeso", 0)
|
||||||
|
)
|
||||||
|
|
||||||
|
private fun genres() = listOf(
|
||||||
|
Genre("Avventura", "4e70ea8cc092255ef70073d3"),
|
||||||
|
Genre("Azione", "4e70ea8cc092255ef70073c3"),
|
||||||
|
Genre("Bara", "4e70ea90c092255ef70074b7"),
|
||||||
|
Genre("Commedia", "4e70ea8cc092255ef70073d0"),
|
||||||
|
Genre("Demenziale", "4e70ea8fc092255ef7007475"),
|
||||||
|
Genre("Dounshinji", "4e70ea93c092255ef70074e4"),
|
||||||
|
Genre("Drama", "4e70ea8cc092255ef70073f9"),
|
||||||
|
Genre("Ecchi", "4e70ea8cc092255ef70073cd"),
|
||||||
|
Genre("Fantasy", "4e70ea8cc092255ef70073c4"),
|
||||||
|
Genre("Harem", "4e70ea8cc092255ef70073d1"),
|
||||||
|
Genre("Hentai", "4e70ea90c092255ef700749a"),
|
||||||
|
Genre("Horror", "4e70ea8cc092255ef70073ce"),
|
||||||
|
Genre("Josei", "4e70ea90c092255ef70074bd"),
|
||||||
|
Genre("Magico", "4e70ea93c092255ef700751b"),
|
||||||
|
Genre("Mecha", "4e70ea8cc092255ef70073ef"),
|
||||||
|
Genre("Misteri", "4e70ea8dc092255ef700740a"),
|
||||||
|
Genre("Musica", "4e70ea8fc092255ef7007456"),
|
||||||
|
Genre("Psicologico", "4e70ea8ec092255ef7007439"),
|
||||||
|
Genre("Raccolta", "4e70ea90c092255ef70074ae"),
|
||||||
|
Genre("Romantico", "4e70ea8cc092255ef70073c5"),
|
||||||
|
Genre("Sci-Fi", "4e70ea8cc092255ef70073e4"),
|
||||||
|
Genre("Scolastico", "4e70ea8cc092255ef70073e5"),
|
||||||
|
Genre("Seinen", "4e70ea8cc092255ef70073ea"),
|
||||||
|
Genre("Sentimentale", "4e70ea8dc092255ef7007432"),
|
||||||
|
Genre("Shota", "4e70ea90c092255ef70074b8"),
|
||||||
|
Genre("Shoujo", "4e70ea8dc092255ef7007421"),
|
||||||
|
Genre("Shounen", "4e70ea8cc092255ef70073c6"),
|
||||||
|
Genre("Sovrannaturale", "4e70ea8cc092255ef70073c7"),
|
||||||
|
Genre("Splatter", "4e70ea99c092255ef70075a3"),
|
||||||
|
Genre("Sportivo", "4e70ea8dc092255ef7007426"),
|
||||||
|
Genre("Storico", "4e70ea8cc092255ef70073f4"),
|
||||||
|
Genre("Vita Quotidiana", "4e70ea8ec092255ef700743f"),
|
||||||
|
Genre("Yaoi", "4e70ea8cc092255ef70073de"),
|
||||||
|
Genre("Yuri", "4e70ea9ac092255ef70075d1")
|
||||||
|
)
|
||||||
|
}
|
|
@ -4,6 +4,7 @@ include 'en-mangafox',
|
||||||
'en-mangasee',
|
'en-mangasee',
|
||||||
'en-readmangatoday',
|
'en-readmangatoday',
|
||||||
'de-wiemanga',
|
'de-wiemanga',
|
||||||
|
'it-mangaeden',
|
||||||
'ru-mangachan',
|
'ru-mangachan',
|
||||||
'ru-mintmanga',
|
'ru-mintmanga',
|
||||||
'ru-readmanga'
|
'ru-readmanga'
|
||||||
|
|
Loading…
Reference in New Issue