Add MangaXY (#14870)
* Add MangaXY * Update src/vi/mangaxy/src/eu/kanade/tachiyomi/extension/vi/MangaXY/MangaXY.kt Co-authored-by: beerpsi <92439990+beerpiss@users.noreply.github.com> * Delete search id because it doesn't work * Fix icon reduces image * Edit code as recommended by alessandrojean Co-authored-by: beerpsi <92439990+beerpiss@users.noreply.github.com>
This commit is contained in:
parent
9a99a7ecca
commit
fd0a489887
|
@ -0,0 +1,2 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest package="eu.kanade.tachiyomi.extension" />
|
|
@ -0,0 +1,11 @@
|
|||
apply plugin: 'com.android.application'
|
||||
apply plugin: 'kotlin-android'
|
||||
|
||||
ext {
|
||||
extName = 'MangaXY'
|
||||
pkgNameSuffix = 'vi.MangaXY'
|
||||
extClass = '.MangaXY'
|
||||
extVersionCode = 1
|
||||
}
|
||||
|
||||
apply from: "$rootDir/common.gradle"
|
Binary file not shown.
After Width: | Height: | Size: 3.5 KiB |
Binary file not shown.
After Width: | Height: | Size: 2.0 KiB |
Binary file not shown.
After Width: | Height: | Size: 4.7 KiB |
Binary file not shown.
After Width: | Height: | Size: 8.2 KiB |
Binary file not shown.
After Width: | Height: | Size: 11 KiB |
Binary file not shown.
After Width: | Height: | Size: 47 KiB |
|
@ -0,0 +1,342 @@
|
|||
package eu.kanade.tachiyomi.extension.vi.MangaXY
|
||||
|
||||
import eu.kanade.tachiyomi.network.GET
|
||||
import eu.kanade.tachiyomi.source.model.Filter
|
||||
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 org.jsoup.nodes.Document
|
||||
import org.jsoup.nodes.Element
|
||||
import java.text.SimpleDateFormat
|
||||
import java.util.Locale
|
||||
import java.util.TimeZone
|
||||
|
||||
class MangaXY : ParsedHttpSource() {
|
||||
|
||||
override val name = "MangaXY"
|
||||
|
||||
override val baseUrl = "https://mangaxy.com"
|
||||
|
||||
override val lang = "vi"
|
||||
|
||||
override val supportsLatest = true
|
||||
|
||||
override val client: OkHttpClient = network.cloudflareClient
|
||||
|
||||
override fun headersBuilder() = Headers.Builder()
|
||||
.add("Referer", "$baseUrl/")
|
||||
|
||||
private val dateFormat = SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.US).apply {
|
||||
timeZone = TimeZone.getTimeZone("Asia/Ho_Chi_Minh")
|
||||
}
|
||||
|
||||
override fun popularMangaRequest(page: Int) = GET(
|
||||
"$baseUrl/search.php".toHttpUrl().newBuilder()
|
||||
.addQueryParameter("act", "search")
|
||||
.addQueryParameter("sort", "xem")
|
||||
.addQueryParameter("view", "thumb")
|
||||
.addQueryParameter("page", page.toString())
|
||||
.toString(),
|
||||
headers
|
||||
)
|
||||
override fun popularMangaSelector() = ".container > .row > div.col-12.col-lg-9 > #tblChap > .thumb"
|
||||
|
||||
override fun latestUpdatesSelector() = popularMangaSelector()
|
||||
|
||||
override fun latestUpdatesRequest(page: Int) = GET(
|
||||
"$baseUrl/search.php".toHttpUrl().newBuilder()
|
||||
.addQueryParameter("act", "search")
|
||||
.addQueryParameter("sort", "chap")
|
||||
.addQueryParameter("view", "thumb")
|
||||
.addQueryParameter("page", page.toString())
|
||||
.toString(),
|
||||
headers
|
||||
)
|
||||
|
||||
override fun popularMangaFromElement(element: Element): SManga {
|
||||
val manga = SManga.create()
|
||||
element.select("a.name").first().let {
|
||||
manga.setUrlWithoutDomain(it.attr("href"))
|
||||
manga.title = it.text().trim()
|
||||
}
|
||||
manga.thumbnail_url = element.select(".item img")
|
||||
.first()
|
||||
.attr("style")
|
||||
.substringAfter("url('")
|
||||
.substringBefore("')")
|
||||
.replace("//", "https:")
|
||||
.replace("http:", "")
|
||||
return manga
|
||||
}
|
||||
|
||||
override fun latestUpdatesFromElement(element: Element): SManga = popularMangaFromElement(element)
|
||||
|
||||
override fun popularMangaNextPageSelector(): String = "div#tblChap p.page a:contains(Cuối)"
|
||||
|
||||
override fun latestUpdatesNextPageSelector() = popularMangaNextPageSelector()
|
||||
|
||||
open class UriPartFilter(displayName: String, private val vals: Array<Pair<String, String>>, state: Int = 0) :
|
||||
Filter.Select<String>(displayName, vals.map { it.first }.toTypedArray(), state) {
|
||||
fun toUriPart() = vals[state].second
|
||||
}
|
||||
private class SortByFilter : UriPartFilter(
|
||||
"Sắp xếp theo",
|
||||
arrayOf(
|
||||
Pair("Chap mới", "chap"),
|
||||
Pair("Truyện mới", "truyen"),
|
||||
Pair("Xem nhiều", "xem"),
|
||||
Pair("Theo ABC", "ten"),
|
||||
Pair("Số Chương", "sochap"),
|
||||
),
|
||||
2
|
||||
)
|
||||
private class SearchTypeFilter : UriPartFilter(
|
||||
"Kiểu tìm",
|
||||
arrayOf(
|
||||
Pair("AND/và", "and"),
|
||||
Pair("OR/hoặc", "or"),
|
||||
)
|
||||
)
|
||||
private class ForFilter : UriPartFilter(
|
||||
"Dành cho",
|
||||
arrayOf(
|
||||
Pair("Bất kì", ""),
|
||||
Pair("Con gái", "gai"),
|
||||
Pair("Con trai", "trai"),
|
||||
Pair("Con nít", "nit"),
|
||||
)
|
||||
)
|
||||
private class AgeFilter : UriPartFilter(
|
||||
"Bất kỳ",
|
||||
arrayOf(
|
||||
Pair("Bất kì", ""),
|
||||
Pair("= 13", "13"),
|
||||
Pair("= 14", "14"),
|
||||
Pair("= 15", "15"),
|
||||
Pair("= 16", "16"),
|
||||
Pair("= 17", "17"),
|
||||
Pair("= 18", "18"),
|
||||
)
|
||||
)
|
||||
private class StatusFilter : UriPartFilter(
|
||||
"Tình trạng",
|
||||
arrayOf(
|
||||
Pair("Bất kì", ""),
|
||||
Pair("Đang dịch", "Ongoing"),
|
||||
Pair("Hoàn thành", "Complete"),
|
||||
Pair("Tạm ngưng", "Drop"),
|
||||
)
|
||||
)
|
||||
private class OriginFilter : UriPartFilter(
|
||||
"Quốc gia",
|
||||
arrayOf(
|
||||
Pair("Bất kì", ""),
|
||||
Pair("Nhật Bản", "nhat"),
|
||||
Pair("Trung Quốc", "trung"),
|
||||
Pair("Hàn Quốc", "han"),
|
||||
Pair("Việt Nam", "vietnam"),
|
||||
)
|
||||
)
|
||||
private class ReadingModeFilter : UriPartFilter(
|
||||
"Kiểu đọc",
|
||||
arrayOf(
|
||||
Pair("Bất kì", ""),
|
||||
Pair("Chưa xác định", "chưa xác định"),
|
||||
Pair("Phải qua trái", "xem từ phải qua trái"),
|
||||
Pair("Trái qua phải", "xem từ trái qua phải"),
|
||||
)
|
||||
)
|
||||
private class YearFilter : Filter.Text("Năm phát hành")
|
||||
private class UserFilter : Filter.Text("Đăng bởi thành viên")
|
||||
private class AuthorFilter : Filter.Text("Tên tác giả")
|
||||
private class SourceFilter : Filter.Text("Nguồn/Nhóm dịch")
|
||||
|
||||
override fun searchMangaRequest(page: Int, query: String, filters: FilterList) = GET(
|
||||
"$baseUrl/search.php".toHttpUrl().newBuilder().apply {
|
||||
addQueryParameter("act", "timnangcao")
|
||||
addQueryParameter("view", "thumb")
|
||||
addQueryParameter("page", page.toString())
|
||||
|
||||
if (query.isNotEmpty()) {
|
||||
addQueryParameter("q", query)
|
||||
}
|
||||
|
||||
(if (filters.isEmpty()) getFilterList() else filters).forEach { filter ->
|
||||
when (filter) {
|
||||
is SortByFilter -> addQueryParameter("sort", filter.toUriPart())
|
||||
is SearchTypeFilter -> addQueryParameter("andor", filter.toUriPart())
|
||||
is ForFilter -> if (filter.state != 0) {
|
||||
addQueryParameter("danhcho", filter.toUriPart())
|
||||
}
|
||||
is AgeFilter -> if (filter.state != 0) {
|
||||
addQueryParameter("DoTuoi", filter.toUriPart())
|
||||
}
|
||||
is StatusFilter -> if (filter.state != 0) {
|
||||
addQueryParameter("TinhTrang", filter.toUriPart())
|
||||
}
|
||||
is OriginFilter -> if (filter.state != 0) {
|
||||
addQueryParameter("quocgia", filter.toUriPart())
|
||||
}
|
||||
is ReadingModeFilter -> if (filter.state != 0) {
|
||||
addQueryParameter("KieuDoc", filter.toUriPart())
|
||||
}
|
||||
is YearFilter -> if (filter.state.isNotEmpty()) {
|
||||
addQueryParameter("NamPhaHanh", filter.state)
|
||||
}
|
||||
is UserFilter -> if (filter.state.isNotEmpty()) {
|
||||
addQueryParameter("u", filter.state)
|
||||
}
|
||||
is AuthorFilter -> if (filter.state.isNotEmpty()) {
|
||||
addQueryParameter("TacGia", filter.state)
|
||||
}
|
||||
is SourceFilter -> if (filter.state.isNotEmpty()) {
|
||||
addQueryParameter("Nguon", filter.state)
|
||||
}
|
||||
is GenreList -> {
|
||||
addQueryParameter(
|
||||
"baogom",
|
||||
filter.state
|
||||
.filter { it.state == Filter.TriState.STATE_INCLUDE }
|
||||
.joinToString(",") { it.id }
|
||||
)
|
||||
addQueryParameter(
|
||||
"khonggom",
|
||||
filter.state
|
||||
.filter { it.state == Filter.TriState.STATE_EXCLUDE }
|
||||
.joinToString(",") { it.id }
|
||||
)
|
||||
}
|
||||
else -> {}
|
||||
}
|
||||
}
|
||||
}.build().toString(),
|
||||
headers
|
||||
)
|
||||
|
||||
override fun searchMangaSelector() = popularMangaSelector()
|
||||
|
||||
override fun searchMangaFromElement(element: Element): SManga {
|
||||
return popularMangaFromElement(element)
|
||||
}
|
||||
|
||||
override fun searchMangaNextPageSelector() = popularMangaNextPageSelector()
|
||||
|
||||
override fun mangaDetailsParse(document: Document): SManga = SManga.create().apply {
|
||||
val infoElement = document.selectFirst(".tab-content")
|
||||
val infoTop = document.selectFirst(".detail-top-wrap")
|
||||
title = infoTop.select("h1.comics-title").text()
|
||||
author = infoTop.select(".created-by").joinToString { it.text() }
|
||||
genre = infoTop.select(".top-comics-type a")
|
||||
.filter { it.text().isNotEmpty() }
|
||||
.joinToString(", ") { it.text() }
|
||||
description = infoElement.select(".manga-info p").text()
|
||||
thumbnail_url = infoTop.select(".detail-top-right img")
|
||||
.first()
|
||||
.attr("style")
|
||||
.substringAfter("url('")
|
||||
.substringBefore("')")
|
||||
.replace("//", "https:")
|
||||
.replace("http:", "")
|
||||
status = when (infoElement.select(".manga-info ul li a").first().text().trim()) {
|
||||
"Đang tiến hành" -> SManga.ONGOING
|
||||
"Đã Hoàn Thành" -> SManga.COMPLETED
|
||||
"Tạm ngưng" -> SManga.ON_HIATUS
|
||||
else -> SManga.UNKNOWN
|
||||
}
|
||||
}
|
||||
|
||||
override fun chapterListSelector() = "#ChapList > .episode-item"
|
||||
override fun chapterFromElement(element: Element) = SChapter.create().apply {
|
||||
setUrlWithoutDomain(element.select(".episode-item").first().attr("abs:href"))
|
||||
name = element.select(".episode-title").first().text()
|
||||
date_upload = runCatching {
|
||||
dateFormat.parse(element.select("div.episode-date > time").attr("datetime"))?.time
|
||||
}.getOrNull() ?: 0L
|
||||
}
|
||||
|
||||
override fun pageListParse(document: Document): List<Page> {
|
||||
return document.select("img.img-fluid").mapIndexed { i, img ->
|
||||
Page(i, imageUrl = img.attr("abs:src"))
|
||||
}
|
||||
}
|
||||
|
||||
override fun imageUrlParse(document: Document) = throw UnsupportedOperationException("Not used")
|
||||
|
||||
open class Genre(name: String, val id: String) : Filter.TriState(name)
|
||||
private class GenreList(genres: List<Genre>) : Filter.Group<Genre>("Thể loại", genres)
|
||||
override fun getFilterList() = FilterList(
|
||||
GenreList(getGenreList()),
|
||||
SortByFilter(),
|
||||
SearchTypeFilter(),
|
||||
ForFilter(),
|
||||
AgeFilter(),
|
||||
StatusFilter(),
|
||||
OriginFilter(),
|
||||
ReadingModeFilter(),
|
||||
YearFilter(),
|
||||
UserFilter(),
|
||||
AuthorFilter(),
|
||||
SourceFilter(),
|
||||
)
|
||||
|
||||
private fun getGenreList() = listOf(
|
||||
Genre("Phát Hành Tại TT8", "106"),
|
||||
Genre("Webtoons", "112"),
|
||||
Genre("Manga", "141"),
|
||||
Genre("Truyện Màu", "113"),
|
||||
Genre("Action", "52"),
|
||||
Genre("Adult", "53"),
|
||||
Genre("Adventure", "65"),
|
||||
Genre("Anime", "107"),
|
||||
Genre("Biseinen", "123"),
|
||||
Genre("Bishounen", "122"),
|
||||
Genre("Comedy", "50"),
|
||||
Genre("Demons", ""),
|
||||
Genre("Doujinshi", "72"),
|
||||
Genre("Drama", "73"),
|
||||
Genre("Ecchi", "74"),
|
||||
Genre("Fantasy", "75"),
|
||||
Genre("Gender Bender", "76"),
|
||||
Genre("Harem", "77"),
|
||||
Genre("Hentai", ""),
|
||||
Genre("Historical", "78"),
|
||||
Genre("Horror", "79"),
|
||||
Genre("Isekai", "139"),
|
||||
Genre("Josei", "80"),
|
||||
Genre("Live action", "81"),
|
||||
Genre("Magic", "116"),
|
||||
Genre("Martial Arts", "84"),
|
||||
Genre("Mature", "85"),
|
||||
Genre("Manhua", "82"),
|
||||
Genre("Manhwa", "83"),
|
||||
Genre("Mecha", "86"),
|
||||
Genre("Mystery", "87"),
|
||||
Genre("One-shot", "88"),
|
||||
Genre("Oneshot", ""),
|
||||
Genre("Other", ""),
|
||||
Genre("Psychological", "89"),
|
||||
Genre("Romance", "90"),
|
||||
Genre("School Life", "91"),
|
||||
Genre("Sci fi", "92"),
|
||||
Genre("Seinen", "93"),
|
||||
Genre("Shotacon", ""),
|
||||
Genre("Shoujo", "94"),
|
||||
Genre("Shoujo Ai", "66"),
|
||||
Genre("Shounen", "96"),
|
||||
Genre("Shounen Ai", "97"),
|
||||
Genre("Slash", "121"),
|
||||
Genre("Slice of Life", "98"),
|
||||
Genre("Smut", "99"),
|
||||
Genre("Sports", "101"),
|
||||
Genre("Super power", ""),
|
||||
Genre("Supernatural", "102"),
|
||||
Genre("Tragedy", "104"),
|
||||
Genre("Yaoi", "114"),
|
||||
Genre("Yuri", "111")
|
||||
)
|
||||
}
|
Loading…
Reference in New Issue