From ee5799b734e1590019f73167529136b65df6c776 Mon Sep 17 00:00:00 2001 From: CakesTwix Date: Thu, 18 Jan 2024 15:50:00 +0200 Subject: [PATCH] higotv: Init --- HigoTVProvider/build.gradle.kts | 25 ++ HigoTVProvider/src/main/AndroidManifest.xml | 2 + .../kotlin/com/lagradost/HigoTVProvider.kt | 253 ++++++++++++++++++ .../com/lagradost/HigoTVProviderPlugin.kt | 13 + .../kotlin/com/lagradost/models/PlayerJson.kt | 13 + 5 files changed, 306 insertions(+) create mode 100644 HigoTVProvider/build.gradle.kts create mode 100644 HigoTVProvider/src/main/AndroidManifest.xml create mode 100644 HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProvider.kt create mode 100644 HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProviderPlugin.kt create mode 100644 HigoTVProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt diff --git a/HigoTVProvider/build.gradle.kts b/HigoTVProvider/build.gradle.kts new file mode 100644 index 0000000..fb82e1b --- /dev/null +++ b/HigoTVProvider/build.gradle.kts @@ -0,0 +1,25 @@ +// use an integer for version numbers +version = 1 + +dependencies{ + implementation("com.google.code.gson:gson:2.9.0") +} + +cloudstream { + language = "uk" + // All of these properties are optional, you can safely remove them + + description = "Новий сайт із бібліотекою аніме з українською озвучкою." + authors = listOf("CakesTwix") + + /** + * Status int as the following: + * 0: Down + * 1: Ok + * 2: Slow + * 3: Beta only + * */ + status = 1 // will be 3 if unspecified + + iconUrl = "https://www.google.com/s2/favicons?domain=higotv.fun&sz=%size%" +} diff --git a/HigoTVProvider/src/main/AndroidManifest.xml b/HigoTVProvider/src/main/AndroidManifest.xml new file mode 100644 index 0000000..29aec9d --- /dev/null +++ b/HigoTVProvider/src/main/AndroidManifest.xml @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProvider.kt b/HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProvider.kt new file mode 100644 index 0000000..6087dc6 --- /dev/null +++ b/HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProvider.kt @@ -0,0 +1,253 @@ +package com.lagradost + +import android.util.Log +import com.google.gson.Gson +import com.google.gson.reflect.TypeToken +import com.lagradost.cloudstream3.AnimeSearchResponse +import com.lagradost.cloudstream3.DubStatus +import com.lagradost.cloudstream3.Episode +import com.lagradost.cloudstream3.HomePageResponse +import com.lagradost.cloudstream3.LoadResponse +import com.lagradost.cloudstream3.MainAPI +import com.lagradost.cloudstream3.MainPageRequest +import com.lagradost.cloudstream3.SearchResponse +import com.lagradost.cloudstream3.SubtitleFile +import com.lagradost.cloudstream3.TvType +import com.lagradost.cloudstream3.addDubStatus +import com.lagradost.cloudstream3.addEpisodes +import com.lagradost.cloudstream3.app +import com.lagradost.cloudstream3.fixUrl +import com.lagradost.cloudstream3.mainPageOf +import com.lagradost.cloudstream3.newAnimeLoadResponse +import com.lagradost.cloudstream3.newAnimeSearchResponse +import com.lagradost.cloudstream3.newHomePageResponse +import com.lagradost.cloudstream3.newMovieLoadResponse +import com.lagradost.cloudstream3.toRatingInt +import com.lagradost.cloudstream3.utils.ExtractorLink +import com.lagradost.cloudstream3.utils.M3u8Helper +import com.lagradost.models.PlayerJson +import org.jsoup.nodes.Element + +class HigoTVProvider : MainAPI() { + + // Basic Info + override var mainUrl = "https://higotv.fun" + override var name = "HigoTV" + override val hasMainPage = true + override var lang = "uk" + override val hasDownloadSupport = true + override val supportedTypes = setOf( + TvType.Anime + ) + + // Sections + override val mainPage = mainPageOf( + "vsevishlo" to "Виходить, Вийшло", + ) + + // Main Page + private val animeSelector = "div.poster" + private val titleSelector = ".greed-item__title_p" + // private val engTitleSelector = "div.th-title-oname.truncate" + private val hrefSelector = "a:contains(Перейти)" + private val posterSelector = ".poster > img" + + // Load info + // private val titleLoadSelector = ".page__subcol-main h1" + // private val genresSelector = "li span:contains(Жанр:) a" + // private val yearSelector = "a[href*=https://uaserials.pro/year/]" + // private val playerSelector = "iframe" + private val descriptionSelector = ".anim-txt-all" + private val ratingSelector = ".rt-tb" + + private val listPlayer = object : TypeToken>() { }.type + + private val TAG = "$name-Debug" + + override suspend fun getMainPage( + page: Int, + request: MainPageRequest + ): HomePageResponse { + val document = app.get("$mainUrl/${request.data}").document + + val home = document.select(animeSelector).map { + it.toSearchResponse() + } + return newHomePageResponse(request.name, home) + } + + private fun Element.toSearchResponse(): AnimeSearchResponse { + val title = this.select(titleSelector).text().replace("Назва: ", "") + // val engTitle = this.selectFirst(engTitleSelector)?.text()?.trim().toString() + val href = this.selectFirst(hrefSelector)?.attr("href").toString() + val posterUrl = fixUrl(this.select(posterSelector).attr("src")) + + return newAnimeSearchResponse(title, href, TvType.Anime) { + // this.otherName = engTitle + this.posterUrl = posterUrl + addDubStatus(isDub = true) + } + + } + + override suspend fun search(query: String): List { + val document = app.post( + url = mainUrl, + data = mapOf( + "do" to "search", + "subaction" to "search", + "story" to query.replace(" ", "+") + ) + ).document + + return document.select(animeSelector).map { + it.toSearchResponse() + } + } + + // Detailed information + override suspend fun load(url: String): LoadResponse { + val document = app.get(url).document + // Parse info + + val title = document.select(".anime-op__txt").text() + val engTitle = document.select(".anime-eng__txt").text() + val poster = fixUrl(document.select(".anime-op__img > img").attr("src")) + val tags = document.select(".span-menu-ogg a").map { it.text() } + + val tvType = with(document.select(".tup-anim").text()){ + when{ + contains("TV SHORT") -> TvType.OVA + contains("OVA") -> TvType.OVA + contains("Фільм") -> TvType.Movie + else -> TvType.Anime + } + } + val description = document.selectFirst(descriptionSelector)?.text()?.trim() + val rating = extractIntFromString(document.select(ratingSelector).text()).toString().toRatingInt() + + // TODO: Fix + val recommendations = document.select("div.owl-item").map { + val title = it.select(".popular-item-title").text().trim() + // val engTitle = this.selectFirst(engTitleSelector)?.text()?.trim().toString() + val href = it.select("a.popular-item-img").attr("href").toString() + val posterUrl = fixUrl(it.select(".img-fit img").attr("src")) + + newAnimeSearchResponse(title, href, tvType) { + this.otherName = engTitle + this.posterUrl = posterUrl + addDubStatus(isDub = true) + } + } + + // Parse episodes + val episodes = mutableListOf() + + val playerRawJson = document.select("div.player").select("script").html() + .substringAfterLast("file:") + .substringBeforeLast("},") + + val parsedJSON = Gson().fromJson>(playerRawJson, listPlayer) + + parsedJSON[1].folder?.forEach { voices -> + if(voices != null){ + if(!voices.title.isNullOrEmpty()){ + voices.folder?.forEach { + if(it != null) { + if (!it.file.isNullOrBlank()){ + episodes.add( + Episode( + "${url}, ${it.title}", + it.title, + episode = it.title!!.replace(" Серія","").toIntOrNull(), + ) + ) + } + } + } + } + } + } + // Parse Episodes as Series + return if (tvType != TvType.Movie) { + + newAnimeLoadResponse(title, url, tvType) { + this.posterUrl = poster + this.engName = engTitle + this.year = year + this.plot = description + this.tags = tags + this.rating = rating + this.recommendations = recommendations + addEpisodes(DubStatus.Dubbed, episodes) + } + } else { // Parse as Movie. + + newMovieLoadResponse(title, url, tvType, "$title, ") { + this.posterUrl = poster + this.name = engTitle + this.year = year + this.plot = description + this.tags = tags + this.rating = rating + this.recommendations = recommendations + } + } + } + + // It works when I click to view the series + override suspend fun loadLinks( + data: String, // (Serial) [Voice name, Series name] + isCasting: Boolean, + subtitleCallback: (SubtitleFile) -> Unit, + callback: (ExtractorLink) -> Unit + ): Boolean { + Log.d(TAG, "loadLinks: $data") + val dataList = data.split(", ") + + val document = app.get(dataList[0]).document + + val playerRawJson = document.select("div.player").select("script").html() + .substringAfterLast("file:") + .substringBeforeLast("},") + + val parsedJSON = Gson().fromJson>(playerRawJson, listPlayer) + + parsedJSON[1].folder?.forEach { voices -> + if(voices != null){ + if(!voices.title.isNullOrEmpty()){ + voices.folder?.forEach { + if(it != null) { + if (!it.file.isNullOrBlank()){ + if (it.title == dataList[1]){ + callback.invoke( + ExtractorLink( + it.file, + name = "${voices.title}", + it.file, + "", + 0, + isM3u8 = false, + ) + ) + } + } + } + } + } + } + } + + return true + } + + private fun extractIntFromString(string: String): Int? { + val value = Regex("(\\d+)").findAll(string).lastOrNull() ?: return null + if(value.value[0].toString() == "0"){ + return value.value.drop(1).toIntOrNull() + } + + return value.value.toIntOrNull() + + } +} diff --git a/HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProviderPlugin.kt b/HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProviderPlugin.kt new file mode 100644 index 0000000..d3e2431 --- /dev/null +++ b/HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProviderPlugin.kt @@ -0,0 +1,13 @@ +package com.lagradost + +import com.lagradost.cloudstream3.plugins.CloudstreamPlugin +import com.lagradost.cloudstream3.plugins.Plugin +import android.content.Context + +@CloudstreamPlugin +class HigoTVProviderPlugin: Plugin() { + override fun load(context: Context) { + // All providers should be added in this manner. Please don't edit the providers list directly. + registerMainAPI(HigoTVProvider()) + } +} diff --git a/HigoTVProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt b/HigoTVProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt new file mode 100644 index 0000000..2eaa37d --- /dev/null +++ b/HigoTVProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt @@ -0,0 +1,13 @@ +package com.lagradost.models + +data class ListPlayerJson ( + + val list: List +) + +data class PlayerJson ( + + val title : String?, + val file : String?, + val folder: List?, +)