2023-07-24 14:25:57 +00:00
|
|
|
package com.hexated
|
|
|
|
|
|
|
|
import com.fasterxml.jackson.annotation.JsonProperty
|
|
|
|
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
|
2023-07-25 07:46:32 +00:00
|
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
2023-07-24 14:25:57 +00:00
|
|
|
import com.lagradost.cloudstream3.utils.*
|
|
|
|
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
|
|
|
|
import org.jsoup.nodes.Element
|
|
|
|
import org.jsoup.select.Elements
|
|
|
|
|
|
|
|
class Nimegami : MainAPI() {
|
|
|
|
override var mainUrl = "https://nimegami.id"
|
|
|
|
override var name = "Nimegami"
|
|
|
|
override val hasMainPage = true
|
|
|
|
override var lang = "id"
|
|
|
|
override val supportedTypes = setOf(
|
|
|
|
TvType.Anime,
|
|
|
|
TvType.AnimeMovie,
|
|
|
|
TvType.OVA
|
|
|
|
)
|
|
|
|
|
|
|
|
companion object {
|
|
|
|
fun getType(t: String): TvType {
|
2023-07-25 07:46:32 +00:00
|
|
|
return when {
|
2023-07-25 08:33:26 +00:00
|
|
|
t.contains("Tv", true) -> TvType.Anime
|
2023-07-25 07:46:32 +00:00
|
|
|
t.contains("Movie", true) -> TvType.AnimeMovie
|
|
|
|
t.contains("OVA", true) || t.contains("Special", true) -> TvType.OVA
|
|
|
|
else -> TvType.Anime
|
|
|
|
}
|
2023-07-24 14:25:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fun getStatus(t: String?): ShowStatus {
|
|
|
|
return when {
|
|
|
|
t?.contains("On-Going", true) == true -> ShowStatus.Ongoing
|
|
|
|
else -> ShowStatus.Completed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override val mainPage = mainPageOf(
|
|
|
|
"" to "Updated Anime",
|
|
|
|
"/type/tv" to "Anime",
|
|
|
|
"/type/movie" to "Movie",
|
|
|
|
"/type/ona" to "ONA",
|
|
|
|
"/type/live-action" to "Live Action",
|
|
|
|
)
|
|
|
|
|
|
|
|
override suspend fun getMainPage(
|
|
|
|
page: Int,
|
|
|
|
request: MainPageRequest
|
|
|
|
): HomePageResponse {
|
|
|
|
val document = app.get("$mainUrl${request.data}/page/$page").document
|
|
|
|
val home = document.select("div.post-article article, div.archive article").mapNotNull {
|
|
|
|
it.toSearchResult()
|
|
|
|
}
|
|
|
|
return newHomePageResponse(
|
|
|
|
list = HomePageList(
|
|
|
|
name = request.name,
|
|
|
|
list = home,
|
|
|
|
isHorizontalImages = request.name != "Updated Anime"
|
|
|
|
),
|
|
|
|
hasNext = true
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
private fun Element.toSearchResult(): AnimeSearchResponse? {
|
|
|
|
val href = fixUrl(this.selectFirst("a")!!.attr("href"))
|
|
|
|
val title = this.selectFirst("h2 a")?.text() ?: return null
|
|
|
|
val posterUrl = (this.selectFirst("noscript img") ?: this.selectFirst("img"))?.attr("src")
|
|
|
|
val episode = this.selectFirst("ul li:contains(Episode), div.eps-archive")?.ownText()
|
|
|
|
?.filter { it.isDigit() }?.toIntOrNull()
|
|
|
|
|
|
|
|
return newAnimeSearchResponse(title, href, TvType.Anime) {
|
|
|
|
this.posterUrl = posterUrl
|
|
|
|
addSub(episode)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun search(query: String): List<SearchResponse> {
|
2024-01-16 13:05:49 +00:00
|
|
|
val searchResponse = mutableListOf<SearchResponse>()
|
|
|
|
for (i in 1..2) {
|
2024-01-17 10:01:09 +00:00
|
|
|
val res = app.get("$mainUrl/page/$i/?s=$query&post_type=post").document.select("div.archive article")
|
2024-01-16 13:05:49 +00:00
|
|
|
.mapNotNull {
|
|
|
|
it.toSearchResult()
|
|
|
|
}
|
|
|
|
searchResponse.addAll(res)
|
|
|
|
}
|
|
|
|
return searchResponse
|
2023-07-24 14:25:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun load(url: String): LoadResponse {
|
|
|
|
val document = app.get(url).document
|
|
|
|
|
|
|
|
val table = document.select("div#Info table tbody")
|
|
|
|
val title = table.getContent("Judul :").text()
|
|
|
|
val poster = document.selectFirst("div.coverthumbnail img")?.attr("src")
|
|
|
|
val bgPoster = document.selectFirst("div.thumbnail-a img")?.attr("src")
|
|
|
|
val tags = table.getContent("Kategori").select("a").map { it.text() }
|
|
|
|
|
|
|
|
val year = table.getContent("Musim / Rilis").text().filter { it.isDigit() }.toIntOrNull()
|
|
|
|
val status = getStatus(document.selectFirst("h1[itemprop=headline]")?.text())
|
2023-09-11 15:25:05 +00:00
|
|
|
val type = getType(table.getContent("Type").text())
|
2023-07-24 14:25:57 +00:00
|
|
|
val description = document.select("div#Sinopsis p").text().trim()
|
2023-07-25 07:46:32 +00:00
|
|
|
val trailer = document.selectFirst("div#Trailer iframe")?.attr("src")
|
2023-07-24 14:25:57 +00:00
|
|
|
|
|
|
|
val episodes = document.select("div.list_eps_stream li")
|
|
|
|
.mapNotNull {
|
2023-09-22 09:38:38 +00:00
|
|
|
val episode = Regex("Episode\\s?(\\d+)").find(it.text())?.groupValues?.getOrNull(0)
|
|
|
|
?.toIntOrNull()
|
2023-07-24 14:25:57 +00:00
|
|
|
val link = it.attr("data")
|
2023-09-11 15:25:05 +00:00
|
|
|
Episode(link, episode = episode)
|
2023-07-24 14:25:57 +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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-22 09:38:38 +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) {
|
2023-07-24 14:25:57 +00:00
|
|
|
engName = title
|
2023-09-11 15:25:05 +00:00
|
|
|
posterUrl = tracker?.image ?: poster
|
|
|
|
backgroundPosterUrl = tracker?.cover ?: bgPoster
|
2023-07-24 14:25:57 +00:00
|
|
|
this.year = year
|
|
|
|
addEpisodes(DubStatus.Subbed, episodes)
|
|
|
|
showStatus = status
|
|
|
|
plot = description
|
|
|
|
this.tags = tags
|
|
|
|
this.recommendations = recommendations
|
2023-07-25 07:46:32 +00:00
|
|
|
addTrailer(trailer)
|
2023-09-11 15:25:05 +00:00
|
|
|
addMalId(tracker?.malId)
|
|
|
|
addAniListId(tracker?.aniId?.toIntOrNull())
|
2023-07-24 14:25:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun loadLinks(
|
|
|
|
data: String,
|
|
|
|
isCasting: Boolean,
|
|
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
|
|
callback: (ExtractorLink) -> Unit
|
|
|
|
): Boolean {
|
|
|
|
|
|
|
|
tryParseJson<ArrayList<Sources>>(base64Decode(data))?.map { sources ->
|
|
|
|
sources.url?.apmap { url ->
|
2023-09-22 09:38:38 +00:00
|
|
|
loadFixedExtractor(
|
2024-01-15 13:04:54 +00:00
|
|
|
url,
|
2023-09-22 09:38:38 +00:00
|
|
|
sources.format,
|
|
|
|
"$mainUrl/",
|
|
|
|
subtitleCallback,
|
|
|
|
callback
|
|
|
|
)
|
2023-07-24 14:25:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
private suspend fun loadFixedExtractor(
|
|
|
|
url: String,
|
|
|
|
quality: String?,
|
|
|
|
referer: String? = null,
|
|
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
|
|
callback: (ExtractorLink) -> Unit
|
|
|
|
) {
|
|
|
|
loadExtractor(url, referer, subtitleCallback) { link ->
|
|
|
|
callback.invoke(
|
|
|
|
ExtractorLink(
|
|
|
|
link.name,
|
|
|
|
link.name,
|
|
|
|
link.url,
|
|
|
|
link.referer,
|
|
|
|
getQualityFromName(quality),
|
2023-09-09 12:16:04 +00:00
|
|
|
link.type,
|
2023-07-24 14:25:57 +00:00
|
|
|
link.headers,
|
|
|
|
link.extractorData
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-22 09:38:38 +00:00
|
|
|
private fun Elements.getContent(css: String): Elements {
|
2023-07-24 14:25:57 +00:00
|
|
|
return this.select("tr:contains($css) td:last-child")
|
|
|
|
}
|
|
|
|
|
|
|
|
data class Sources(
|
|
|
|
@JsonProperty("format") val format: String? = null,
|
|
|
|
@JsonProperty("url") val url: ArrayList<String>? = arrayListOf(),
|
|
|
|
)
|
|
|
|
|
|
|
|
}
|