package com.hexated import com.fasterxml.jackson.annotation.JsonProperty import com.lagradost.cloudstream3.* import com.lagradost.cloudstream3.mvvm.safeApiCall import com.lagradost.cloudstream3.utils.AppUtils.parseJson import com.lagradost.cloudstream3.utils.AppUtils.toJson import com.lagradost.cloudstream3.utils.ExtractorLink import com.lagradost.cloudstream3.utils.SubtitleHelper import com.lagradost.cloudstream3.utils.getQualityFromName import kotlinx.coroutines.delay class Loklok : MainAPI() { override var name = "Loklok" override val hasMainPage = true override val hasChromecastSupport = true override val instantLinkLoading = true override val supportedTypes = setOf( TvType.Movie, TvType.TvSeries, TvType.Anime, TvType.AsianDrama, ) private val headers = mapOf( "lang" to "en", "versioncode" to "11", "clienttype" to "ios_jike_default" ) // no license found // thanks to https://github.com/napthedev/filmhot for providing API companion object { private val api = base64DecodeAPI("dg==LnQ=b2s=a2w=bG8=aS4=YXA=ZS0=aWw=b2I=LW0=Z2E=Ly8=czo=dHA=aHQ=") private val apiUrl = "$api/${base64Decode("Y21zL2FwcA==")}" private val searchApi = base64Decode("aHR0cHM6Ly9sb2tsb2suY29t") private const val mainImageUrl = "https://images.weserv.nl" private fun base64DecodeAPI(api: String): String { return api.chunked(4).map { base64Decode(it) }.reversed().joinToString("") } } private fun encode(input: String): String = java.net.URLEncoder.encode(input, "utf-8").replace("+", "%20") override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { val home = ArrayList() for (i in 0..6) { // delay(500) app.get("$apiUrl/homePage/getHome?page=$i", headers = headers) .parsedSafe()?.data?.recommendItems ?.filterNot { it.homeSectionType == "BLOCK_GROUP" } ?.filterNot { it.homeSectionType == "BANNER" } ?.mapNotNull { res -> val header = res.homeSectionName ?: return@mapNotNull null val media = res.media?.mapNotNull { media -> media.toSearchResponse() } ?: throw ErrorLoadingException("Invalid Json Reponse") home.add(HomePageList(header, media)) } } return HomePageResponse(home) } private fun Media.toSearchResponse(): SearchResponse? { return newMovieSearchResponse( title ?: name ?: return null, UrlData(id, category).toJson(), TvType.Movie, ) { this.posterUrl = (imageUrl ?: coverVerticalUrl)?.let { "$mainImageUrl/?url=${encode(it)}&w=175&h=246&fit=cover&output=webp" } } } override suspend fun search(query: String): List { val res = app.get( "$searchApi/search?keyword=$query", ).document val script = res.select("script").find { it.data().contains("function(a,b,c,d,e") }?.data() ?.substringAfter("searchResults:[")?.substringBefore("]}],fetch") return res.select("div.search-list div.search-video-card").mapIndexed { num, block -> val name = block.selectFirst("h2.title")?.text() val data = block.selectFirst("a")?.attr("href")?.split("/") val id = data?.last() val type = data?.get(2)?.toInt() val image = Regex("coverVerticalUrl:\"(.*?)\",").findAll(script.toString()) .map { it.groupValues[1] }.toList().getOrNull(num)?.replace("\\u002F", "/") newMovieSearchResponse( "$name", UrlData(id, type).toJson(), TvType.Movie, ) { this.posterUrl = image } } } override suspend fun load(url: String): LoadResponse? { val data = parseJson(url) val res = app.get( "$apiUrl/movieDrama/get?id=${data.id}&category=${data.category}", headers = headers ).parsedSafe()?.data ?: throw ErrorLoadingException("Invalid Json Reponse") val episodes = res.episodeVo?.map { eps -> val definition = eps.definitionList?.map { Definition( it.code, it.description, ) } val subtitling = eps.subtitlingList?.map { Subtitling( it.languageAbbr, it.language, it.subtitlingUrl ) } Episode( data = UrlEpisode( data.id.toString(), data.category, eps.id, definition, subtitling ).toJson(), episode = eps.seriesNo ) } ?: throw ErrorLoadingException("No Episode Found") val recommendations = res.likeList?.mapNotNull { rec -> rec.toSearchResponse() } val type = when { data.category == 0 -> { TvType.Movie } data.category != 0 && res.tagNameList?.contains("Anime") == true -> { TvType.Anime } else -> { TvType.TvSeries } } return newTvSeriesLoadResponse( res.name ?: return null, url, type, episodes ) { this.posterUrl = res.coverVerticalUrl this.year = res.year this.plot = res.introduction this.tags = res.tagNameList this.rating = res.score.toRatingInt() this.recommendations = recommendations } } private fun getLanguage(str: String): String { return when (str) { "in_ID" -> "Indonesian" "pt" -> "Portuguese" else -> str.split("_").first().let { SubtitleHelper.fromTwoLettersToLanguage(it).toString() } } } override suspend fun loadLinks( data: String, isCasting: Boolean, subtitleCallback: (SubtitleFile) -> Unit, callback: (ExtractorLink) -> Unit ): Boolean { val res = parseJson(data) res.definitionList?.apmap { video -> safeApiCall { delay(500) app.get( "$apiUrl/media/previewInfo?category=${res.category}&contentId=${res.id}&episodeId=${res.epId}&definition=${video.code}", headers = headers ).parsedSafe