2022-09-09 16:09:54 +00:00
|
|
|
package com.hexated
|
|
|
|
|
|
|
|
import com.lagradost.cloudstream3.*
|
2023-09-11 15:25:05 +00:00
|
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId
|
|
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId
|
2022-09-09 16:09:54 +00:00
|
|
|
import com.lagradost.cloudstream3.utils.ExtractorLink
|
|
|
|
import org.jsoup.Jsoup
|
|
|
|
import org.jsoup.nodes.Element
|
|
|
|
|
|
|
|
class KuramanimeProvider : MainAPI() {
|
2023-08-26 23:13:47 +00:00
|
|
|
override var mainUrl = "https://kuramanime.pro"
|
2022-09-09 16:09:54 +00:00
|
|
|
override var name = "Kuramanime"
|
|
|
|
override val hasQuickSearch = false
|
|
|
|
override val hasMainPage = true
|
|
|
|
override var lang = "id"
|
|
|
|
override val hasDownloadSupport = true
|
|
|
|
override val supportedTypes = setOf(
|
2024-01-07 19:17:40 +00:00
|
|
|
TvType.Anime,
|
|
|
|
TvType.AnimeMovie,
|
|
|
|
TvType.OVA
|
2022-09-09 16:09:54 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
companion object {
|
2023-04-04 00:44:24 +00:00
|
|
|
fun getType(t: String, s: Int): TvType {
|
2022-11-24 14:44:35 +00:00
|
|
|
return if (t.contains("OVA", true) || t.contains("Special")) TvType.OVA
|
2023-04-04 00:44:24 +00:00
|
|
|
else if (t.contains("Movie", true) && s == 1) TvType.AnimeMovie
|
2022-11-24 14:44:35 +00:00
|
|
|
else TvType.Anime
|
|
|
|
}
|
|
|
|
|
2022-09-09 16:09:54 +00:00
|
|
|
fun getStatus(t: String): ShowStatus {
|
|
|
|
return when (t) {
|
|
|
|
"Selesai Tayang" -> ShowStatus.Completed
|
|
|
|
"Sedang Tayang" -> ShowStatus.Ongoing
|
|
|
|
else -> ShowStatus.Completed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override val mainPage = mainPageOf(
|
2024-01-07 19:17:40 +00:00
|
|
|
"$mainUrl/anime/ongoing?order_by=updated&page=" to "Sedang Tayang",
|
|
|
|
"$mainUrl/anime/finished?order_by=updated&page=" to "Selesai Tayang",
|
|
|
|
"$mainUrl/properties/season/summer-2022?order_by=most_viewed&page=" to "Dilihat Terbanyak Musim Ini",
|
|
|
|
"$mainUrl/anime/movie?order_by=updated&page=" to "Film Layar Lebar",
|
2022-09-09 16:09:54 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
override suspend fun getMainPage(
|
2024-01-07 19:17:40 +00:00
|
|
|
page: Int,
|
|
|
|
request: MainPageRequest
|
2022-09-09 16:09:54 +00:00
|
|
|
): HomePageResponse {
|
|
|
|
val document = app.get(request.data + page).document
|
|
|
|
|
|
|
|
val home = document.select("div.col-lg-4.col-md-6.col-sm-6").mapNotNull {
|
|
|
|
it.toSearchResult()
|
|
|
|
}
|
|
|
|
|
|
|
|
return newHomePageResponse(request.name, home)
|
|
|
|
}
|
|
|
|
|
|
|
|
private fun getProperAnimeLink(uri: String): String {
|
|
|
|
return if (uri.contains("/episode")) {
|
|
|
|
Regex("(.*)/episode/.+").find(uri)?.groupValues?.get(1).toString() + "/"
|
|
|
|
} else {
|
|
|
|
uri
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private fun Element.toSearchResult(): AnimeSearchResponse? {
|
|
|
|
val href = getProperAnimeLink(fixUrl(this.selectFirst("a")!!.attr("href")))
|
|
|
|
val title = this.selectFirst("h5 a")?.text() ?: return null
|
|
|
|
val posterUrl = fixUrl(this.select("div.product__item__pic.set-bg").attr("data-setbg"))
|
2023-01-14 09:40:35 +00:00
|
|
|
val episode = this.select("div.ep span").text().let {
|
2023-01-23 23:21:04 +00:00
|
|
|
Regex("Ep\\s(\\d+)\\s/").find(it)?.groupValues?.getOrNull(1)?.toIntOrNull()
|
2023-01-14 09:40:35 +00:00
|
|
|
}
|
2022-09-09 16:09:54 +00:00
|
|
|
|
|
|
|
return newAnimeSearchResponse(title, href, TvType.Anime) {
|
|
|
|
this.posterUrl = posterUrl
|
|
|
|
addSub(episode)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun search(query: String): List<SearchResponse> {
|
|
|
|
val link = "$mainUrl/anime?search=$query&order_by=latest"
|
|
|
|
val document = app.get(link).document
|
|
|
|
|
2022-12-29 11:48:41 +00:00
|
|
|
return document.select("div#animeList div.product__item").mapNotNull {
|
2022-09-09 16:09:54 +00:00
|
|
|
it.toSearchResult()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun load(url: String): LoadResponse {
|
|
|
|
val document = app.get(url).document
|
|
|
|
|
|
|
|
val title = document.selectFirst(".anime__details__title > h3")!!.text().trim()
|
|
|
|
val poster = document.selectFirst(".anime__details__pic")?.attr("data-setbg")
|
2024-01-07 19:17:40 +00:00
|
|
|
val tags =
|
|
|
|
document.select("div.anime__details__widget > div > div:nth-child(2) > ul > li:nth-child(1)")
|
|
|
|
.text().trim().replace("Genre: ", "").split(", ")
|
2022-09-09 16:09:54 +00:00
|
|
|
|
2023-01-23 23:21:04 +00:00
|
|
|
val year = Regex("\\D").replace(
|
2024-01-07 19:17:40 +00:00
|
|
|
document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(5)")
|
|
|
|
.text().trim().replace("Musim: ", ""), ""
|
2022-09-09 16:09:54 +00:00
|
|
|
).toIntOrNull()
|
|
|
|
val status = getStatus(
|
2024-01-07 19:17:40 +00:00
|
|
|
document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(3)")
|
|
|
|
.text().trim().replace("Status: ", "")
|
2022-09-09 16:09:54 +00:00
|
|
|
)
|
|
|
|
val description = document.select(".anime__details__text > p").text().trim()
|
2022-11-24 14:44:35 +00:00
|
|
|
|
2022-12-29 11:48:41 +00:00
|
|
|
val episodes = mutableListOf<Episode>()
|
|
|
|
|
2024-01-07 19:17:40 +00:00
|
|
|
for (i in 1..10) {
|
2022-12-29 11:48:41 +00:00
|
|
|
val doc = app.get("$url?page=$i").document
|
2024-01-07 19:17:40 +00:00
|
|
|
val eps = Jsoup.parse(doc.select("#episodeLists").attr("data-content"))
|
|
|
|
.select("a.btn.btn-sm.btn-danger")
|
|
|
|
.mapNotNull {
|
|
|
|
val name = it.text().trim()
|
|
|
|
val episode = Regex("(\\d+[.,]?\\d*)").find(name)?.groupValues?.getOrNull(0)
|
|
|
|
?.toIntOrNull()
|
|
|
|
val link = it.attr("href")
|
|
|
|
Episode(link, episode = episode)
|
|
|
|
}
|
|
|
|
if (eps.isEmpty()) break else episodes.addAll(eps)
|
2022-12-29 11:48:41 +00:00
|
|
|
}
|
2022-09-09 16:09:54 +00:00
|
|
|
|
2024-01-07 19:17:40 +00:00
|
|
|
val type = getType(
|
|
|
|
document.selectFirst("div.col-lg-6.col-md-6 ul li:contains(Tipe:) a")?.text()
|
|
|
|
?.lowercase() ?: "tv", episodes.size
|
|
|
|
)
|
2022-09-09 16:09:54 +00:00
|
|
|
val recommendations = document.select("div#randomList > a").mapNotNull {
|
|
|
|
val epHref = it.attr("href")
|
|
|
|
val epTitle = it.select("h5.sidebar-title-h5.px-2.py-2").text()
|
|
|
|
val epPoster = it.select(".product__sidebar__view__item.set-bg").attr("data-setbg")
|
|
|
|
newAnimeSearchResponse(epTitle, epHref, TvType.Anime) {
|
|
|
|
this.posterUrl = epPoster
|
|
|
|
addDubStatus(dubExist = false, subExist = true)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-07 19:17:40 +00:00
|
|
|
val tracker = APIHolder.getTracker(listOf(title), TrackerType.getTypes(type), year, true)
|
2023-09-11 15:25:05 +00:00
|
|
|
|
|
|
|
return newAnimeLoadResponse(title, url, type) {
|
2022-09-09 16:09:54 +00:00
|
|
|
engName = title
|
2023-09-11 15:25:05 +00:00
|
|
|
posterUrl = tracker?.image ?: poster
|
|
|
|
backgroundPosterUrl = tracker?.cover
|
2022-09-09 16:09:54 +00:00
|
|
|
this.year = year
|
|
|
|
addEpisodes(DubStatus.Subbed, episodes)
|
|
|
|
showStatus = status
|
|
|
|
plot = description
|
|
|
|
this.tags = tags
|
|
|
|
this.recommendations = recommendations
|
2023-09-11 15:25:05 +00:00
|
|
|
addMalId(tracker?.malId)
|
|
|
|
addAniListId(tracker?.aniId?.toIntOrNull())
|
2022-09-09 16:09:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun loadLinks(
|
2024-01-07 19:17:40 +00:00
|
|
|
data: String,
|
|
|
|
isCasting: Boolean,
|
|
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
|
|
callback: (ExtractorLink) -> Unit
|
2022-09-09 16:09:54 +00:00
|
|
|
): Boolean {
|
2023-11-21 14:46:11 +00:00
|
|
|
|
2022-09-09 16:09:54 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|