2022-09-09 16:09:54 +00:00
|
|
|
|
package com.hexated
|
|
|
|
|
|
|
|
|
|
import com.fasterxml.jackson.annotation.JsonProperty
|
|
|
|
|
import com.lagradost.cloudstream3.*
|
|
|
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addActors
|
|
|
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
|
|
|
|
import com.lagradost.cloudstream3.mvvm.safeApiCall
|
|
|
|
|
import com.lagradost.cloudstream3.utils.*
|
2022-10-09 15:12:43 +00:00
|
|
|
|
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
|
2022-09-09 16:09:54 +00:00
|
|
|
|
import org.jsoup.nodes.Element
|
|
|
|
|
|
|
|
|
|
class Hdfilmcehennemi : MainAPI() {
|
|
|
|
|
override var mainUrl = "https://www.hdfilmcehennemi.live"
|
|
|
|
|
override var name = "hdfilmcehennemi"
|
|
|
|
|
override val hasMainPage = true
|
|
|
|
|
override var lang = "tr"
|
|
|
|
|
override val hasQuickSearch = true
|
|
|
|
|
override val hasDownloadSupport = true
|
|
|
|
|
override val supportedTypes = setOf(
|
|
|
|
|
TvType.Movie,
|
|
|
|
|
TvType.TvSeries,
|
|
|
|
|
)
|
|
|
|
|
|
2022-10-09 15:12:43 +00:00
|
|
|
|
companion object {
|
|
|
|
|
private const val vidmolyServer = "https://vidmoly.to"
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-09 16:09:54 +00:00
|
|
|
|
override val mainPage = mainPageOf(
|
|
|
|
|
"$mainUrl/category/tavsiye-filmler-izle1/page/" to "Tavsiye Filmler Kategorisi",
|
2022-09-20 15:20:29 +00:00
|
|
|
|
"$mainUrl/yabancidiziizle/page/" to "Son Eklenen Yabancı Diziler",
|
2022-09-09 16:09:54 +00:00
|
|
|
|
"$mainUrl/imdb-7-puan-uzeri-filmler/page/" to "Imdb 7+ Filmler",
|
|
|
|
|
"$mainUrl/en-cok-yorumlananlar/page/" to "En Çok Yorumlananlar",
|
|
|
|
|
"$mainUrl/en-cok-begenilen-filmleri-izle/page/" to "En Çok Beğenilenler",
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
override suspend fun getMainPage(
|
|
|
|
|
page: Int,
|
|
|
|
|
request: MainPageRequest
|
|
|
|
|
): HomePageResponse {
|
|
|
|
|
val document = app.get(request.data + page).document
|
|
|
|
|
val home = document.select("div.card-body div.row div.col-6.col-sm-3.poster-container")
|
|
|
|
|
.mapNotNull {
|
|
|
|
|
it.toSearchResult()
|
|
|
|
|
}
|
|
|
|
|
return newHomePageResponse(request.name, home)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private fun Element.toSearchResult(): SearchResponse? {
|
|
|
|
|
val title = this.selectFirst("a")?.text() ?: return null
|
|
|
|
|
val href = fixUrlNull(this.selectFirst("a")?.attr("href")) ?: return null
|
|
|
|
|
val posterUrl = fixUrlNull(this.selectFirst("img")?.attr("data-src"))
|
|
|
|
|
return newMovieSearchResponse(title, href, TvType.Movie) {
|
|
|
|
|
this.posterUrl = posterUrl
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private fun Media.toSearchResponse(): SearchResponse? {
|
|
|
|
|
return newMovieSearchResponse(
|
|
|
|
|
title ?: return null,
|
|
|
|
|
"$mainUrl/$slugPrefix$slug",
|
|
|
|
|
TvType.TvSeries,
|
|
|
|
|
) {
|
|
|
|
|
this.posterUrl = "$mainUrl/uploads/poster/$poster"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
override suspend fun quickSearch(query: String): List<SearchResponse> = search(query)
|
|
|
|
|
|
|
|
|
|
override suspend fun search(query: String): List<SearchResponse> {
|
|
|
|
|
return app.post(
|
|
|
|
|
"$mainUrl/search/",
|
|
|
|
|
data = mapOf("query" to query),
|
|
|
|
|
referer = "$mainUrl/",
|
|
|
|
|
headers = mapOf(
|
|
|
|
|
"Accept" to "application/json, text/javascript, */*; q=0.01",
|
|
|
|
|
"X-Requested-With" to "XMLHttpRequest"
|
|
|
|
|
)
|
|
|
|
|
).parsedSafe<Result>()?.result?.mapNotNull { media ->
|
|
|
|
|
media.toSearchResponse()
|
|
|
|
|
} ?: throw ErrorLoadingException("Invalid Json reponse")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
override suspend fun load(url: String): LoadResponse? {
|
|
|
|
|
val document = app.get(url).document
|
|
|
|
|
|
|
|
|
|
val title = document.selectFirst("div.card-header > h1, div.card-header > h2")?.text()
|
|
|
|
|
?: return null
|
|
|
|
|
val poster = fixUrlNull(document.selectFirst("img.img-fluid")?.attr("src"))
|
|
|
|
|
val tags = document.select("div.mb-0.lh-lg div:nth-child(5) a").map { it.text() }
|
|
|
|
|
val year =
|
|
|
|
|
document.selectFirst("div.mb-0.lh-lg div:nth-child(4) a")?.text()?.trim()?.toIntOrNull()
|
|
|
|
|
val tvType = if (document.select("nav#seasonsTabs").isNullOrEmpty()
|
|
|
|
|
) TvType.Movie else TvType.TvSeries
|
|
|
|
|
val description = document.selectFirst("article.text-white > p")?.text()?.trim()
|
|
|
|
|
val rating = document.selectFirst("div.rating-votes div.rate span")?.text()?.toRatingInt()
|
|
|
|
|
val actors = document.select("div.mb-0.lh-lg div:last-child a.chip").map {
|
|
|
|
|
Actor(it.text(), it.select("img").attr("src"))
|
|
|
|
|
}
|
|
|
|
|
val recommendations =
|
|
|
|
|
document.select("div.swiper-wrapper div.poster.poster-pop").mapNotNull {
|
|
|
|
|
val recName = it.selectFirst("h2.title")?.text() ?: return@mapNotNull null
|
|
|
|
|
val recHref =
|
|
|
|
|
fixUrlNull(it.selectFirst("a")?.attr("href")) ?: return@mapNotNull null
|
|
|
|
|
val recPosterUrl = fixUrlNull(it.selectFirst("img")?.attr("data-src"))
|
|
|
|
|
newTvSeriesSearchResponse(recName, recHref, TvType.TvSeries) {
|
|
|
|
|
this.posterUrl = recPosterUrl
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return if (tvType == TvType.TvSeries) {
|
|
|
|
|
val trailer =
|
|
|
|
|
document.selectFirst("button.btn.btn-fragman.btn-danger")?.attr("data-trailer")
|
|
|
|
|
?.let {
|
|
|
|
|
"https://www.youtube.com/embed/$it"
|
|
|
|
|
}
|
|
|
|
|
val episodes = document.select("div#seasonsTabs-tabContent div.card-list-item").map {
|
|
|
|
|
val href = it.select("a").attr("href")
|
|
|
|
|
val name = it.select("h3").text().trim()
|
|
|
|
|
val episode = it.select("h3").text().let { num ->
|
|
|
|
|
Regex("Sezon\\s?([0-9]+).").find(num)?.groupValues?.getOrNull(1)?.toIntOrNull()
|
|
|
|
|
}
|
|
|
|
|
val season = it.parents()[1].attr("id").substringAfter("-").toIntOrNull()
|
|
|
|
|
Episode(
|
|
|
|
|
href,
|
|
|
|
|
name,
|
|
|
|
|
season,
|
|
|
|
|
episode,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) {
|
|
|
|
|
this.posterUrl = poster
|
|
|
|
|
this.year = year
|
|
|
|
|
this.plot = description
|
|
|
|
|
this.tags = tags
|
|
|
|
|
this.rating = rating
|
|
|
|
|
addActors(actors)
|
|
|
|
|
this.recommendations = recommendations
|
|
|
|
|
addTrailer(trailer)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
val trailer =
|
|
|
|
|
document.selectFirst("nav.nav.card-nav.nav-slider a[data-bs-toggle=\"modal\"]")
|
|
|
|
|
?.attr("data-trailer")?.let {
|
|
|
|
|
"https://www.youtube.com/embed/$it"
|
|
|
|
|
}
|
|
|
|
|
newMovieLoadResponse(title, url, TvType.Movie, url) {
|
|
|
|
|
this.posterUrl = poster
|
|
|
|
|
this.year = year
|
|
|
|
|
this.plot = description
|
|
|
|
|
this.tags = tags
|
|
|
|
|
this.rating = rating
|
|
|
|
|
addActors(actors)
|
|
|
|
|
this.recommendations = recommendations
|
|
|
|
|
addTrailer(trailer)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-09 15:12:43 +00:00
|
|
|
|
private fun String.addMarks(str: String): String {
|
|
|
|
|
return this.replace(Regex("\"?$str\"?"), "\"$str\"")
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-09 16:09:54 +00:00
|
|
|
|
private suspend fun invokeLocalSource(
|
|
|
|
|
source: String,
|
|
|
|
|
url: String,
|
2022-09-20 10:58:40 +00:00
|
|
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
|
|
|
callback: (ExtractorLink) -> Unit
|
2022-09-09 16:09:54 +00:00
|
|
|
|
) {
|
2022-10-09 15:12:43 +00:00
|
|
|
|
val script = app.get(
|
|
|
|
|
url,
|
|
|
|
|
referer = "${mainUrl}/"
|
|
|
|
|
).document.select("script")
|
|
|
|
|
.find { it.data().contains("sources:") }?.data()
|
|
|
|
|
val videoData = script?.substringAfter("sources: [")
|
|
|
|
|
?.substringBefore("],")?.addMarks("file")
|
|
|
|
|
val subData = script?.substringAfter("tracks: [")?.substringBefore("]")?.addMarks("file")
|
|
|
|
|
?.addMarks("label")?.addMarks("kind")
|
|
|
|
|
|
|
|
|
|
tryParseJson<Source>(videoData)?.file?.let { m3uLink ->
|
|
|
|
|
M3u8Helper.generateM3u8(
|
|
|
|
|
source,
|
|
|
|
|
m3uLink,
|
|
|
|
|
if (url.startsWith(mainUrl)) "$mainUrl/" else "$vidmolyServer/"
|
|
|
|
|
).forEach(callback)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tryParseJson<List<SubSource>>("[${subData}]")
|
|
|
|
|
?.filter { it.kind == "captions" }?.map {
|
2022-09-20 10:58:40 +00:00
|
|
|
|
subtitleCallback.invoke(
|
|
|
|
|
SubtitleFile(
|
2022-10-09 15:12:43 +00:00
|
|
|
|
it.label.toString(),
|
2022-10-09 15:40:51 +00:00
|
|
|
|
"$vidmolyServer${it.file}"
|
2022-09-20 10:58:40 +00:00
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
}
|
2022-09-09 16:09:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
override suspend fun loadLinks(
|
|
|
|
|
data: String,
|
|
|
|
|
isCasting: Boolean,
|
|
|
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
|
|
|
callback: (ExtractorLink) -> Unit
|
|
|
|
|
): Boolean {
|
|
|
|
|
app.get(data).document.select("nav.nav.card-nav.nav-slider a.nav-link").map {
|
|
|
|
|
Pair(it.attr("href"), it.text())
|
|
|
|
|
}.apmap { (url, source) ->
|
|
|
|
|
safeApiCall {
|
|
|
|
|
app.get(url).document.select("div.card-video > iframe").attr("data-src")
|
|
|
|
|
.let { link ->
|
2022-10-09 15:12:43 +00:00
|
|
|
|
invokeLocalSource(source, link, subtitleCallback, callback)
|
2022-09-09 16:09:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-09 15:12:43 +00:00
|
|
|
|
private data class Source(
|
|
|
|
|
@JsonProperty("file") val file: String? = null,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
private data class SubSource(
|
|
|
|
|
@JsonProperty("file") val file: String? = null,
|
|
|
|
|
@JsonProperty("label") val label: String? = null,
|
|
|
|
|
@JsonProperty("kind") val kind: String? = null,
|
|
|
|
|
)
|
|
|
|
|
|
2022-09-09 16:09:54 +00:00
|
|
|
|
data class Result(
|
|
|
|
|
@JsonProperty("result") val result: ArrayList<Media>? = arrayListOf(),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
data class Media(
|
|
|
|
|
@JsonProperty("title") val title: String? = null,
|
|
|
|
|
@JsonProperty("poster") val poster: String? = null,
|
|
|
|
|
@JsonProperty("slug") val slug: String? = null,
|
|
|
|
|
@JsonProperty("slug_prefix") val slugPrefix: String? = null,
|
|
|
|
|
)
|
|
|
|
|
}
|