Compare commits
No commits in common. "0d6c62870bc966e2b0a0fe1325a6134fcf38bad2" and "f34f268690a46a28b7eead9d793d14790933d978" have entirely different histories.
0d6c62870b
...
f34f268690
9 changed files with 328 additions and 5 deletions
25
HigoTVProvider/build.gradle.kts
Normal file
25
HigoTVProvider/build.gradle.kts
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
// use an integer for version numbers
|
||||||
|
version = 7
|
||||||
|
|
||||||
|
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%"
|
||||||
|
}
|
2
HigoTVProvider/src/main/AndroidManifest.xml
Normal file
2
HigoTVProvider/src/main/AndroidManifest.xml
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<manifest package="com.lagradost"/>
|
248
HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProvider.kt
Normal file
248
HigoTVProvider/src/main/kotlin/com/lagradost/HigoTVProvider.kt
Normal file
|
@ -0,0 +1,248 @@
|
||||||
|
package com.lagradost
|
||||||
|
|
||||||
|
import com.google.gson.GsonBuilder
|
||||||
|
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 com.lagradost.models.Search
|
||||||
|
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)
|
||||||
|
|
||||||
|
private val searchUrl = "$mainUrl/search?query="
|
||||||
|
|
||||||
|
// 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<List<PlayerJson>>() {}.type
|
||||||
|
private val listSearch = object : TypeToken<List<Search>>() {}.type
|
||||||
|
|
||||||
|
private val TAG = "$name-Debug"
|
||||||
|
|
||||||
|
private val gson = GsonBuilder()
|
||||||
|
.setLenient()
|
||||||
|
.create()
|
||||||
|
|
||||||
|
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.selectFirst(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<SearchResponse> {
|
||||||
|
val document = app.get(searchUrl + query).text
|
||||||
|
return gson.fromJson<List<Search>>(document, listSearch).map {
|
||||||
|
newAnimeSearchResponse(it.name, "$mainUrl/movies/view/${it.slug}", TvType.Anime) {
|
||||||
|
// this.otherName = engTitle
|
||||||
|
this.posterUrl = it.poster
|
||||||
|
addDubStatus(isDub = true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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 = document.select(".main-poster").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()
|
||||||
|
|
||||||
|
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<Episode>()
|
||||||
|
|
||||||
|
document.select("iframe").forEach {
|
||||||
|
if(it.attr("src").isNotEmpty()){
|
||||||
|
val playerDocument = app.get(it.attr("src"),
|
||||||
|
headers = mapOf(
|
||||||
|
"Host" to "moonanime.art",
|
||||||
|
"Accept" to "*/*",
|
||||||
|
"User-Agent" to "Mozilla/5.0 (Windows NT 10.0; rv:126.0) Gecko/20100101 Firefox/126.0",
|
||||||
|
"accept-language" to "en-US,en;q=0.5"
|
||||||
|
)).document
|
||||||
|
val parsedJSON = gson.fromJson<List<PlayerJson>>(
|
||||||
|
playerDocument.select("script[type*=text/javascript]").html().substringAfter("file: '").substringBefore("',"), listPlayer
|
||||||
|
)
|
||||||
|
|
||||||
|
parsedJSON?.forEach {
|
||||||
|
episodes.add(
|
||||||
|
Episode(
|
||||||
|
"${url}, ${it.title}",
|
||||||
|
it.title,
|
||||||
|
episode = it.title!!.replace("Серія ", "").toIntOrNull(),
|
||||||
|
posterUrl = it.poster
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Log.d("CakesTwix-Debug", playerRawJson)
|
||||||
|
|
||||||
|
// 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, "${url}, 1 Серія") {
|
||||||
|
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) [url, episode 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
|
||||||
|
|
||||||
|
document.select("iframe").forEach {
|
||||||
|
if(it.attr("src").isNotEmpty()){
|
||||||
|
val playerDocument = app.get(it.attr("src"),
|
||||||
|
headers = mapOf(
|
||||||
|
"Host" to "moonanime.art",
|
||||||
|
"Accept" to "*/*",
|
||||||
|
"User-Agent" to "Mozilla/5.0 (Windows NT 10.0; rv:126.0) Gecko/20100101 Firefox/126.0",
|
||||||
|
"accept-language" to "en-US,en;q=0.5"
|
||||||
|
)).document
|
||||||
|
// Log.d("CakesTwix-Debug", playerDocument.select("script[type*=text/javascript]").html().substringAfter("file:'").substringBefore("',"))
|
||||||
|
val parsedJSON = gson.fromJson<List<PlayerJson>>(
|
||||||
|
playerDocument.select("script[type*=text/javascript]").html().substringAfter("file: '").substringBefore("',"), listPlayer
|
||||||
|
)
|
||||||
|
|
||||||
|
parsedJSON?.forEach {
|
||||||
|
if(it.title != dataList[1]) return@forEach
|
||||||
|
M3u8Helper.generateM3u8(
|
||||||
|
source = it.title,
|
||||||
|
streamUrl = it.file!!,
|
||||||
|
referer = "https://moonanime.art"
|
||||||
|
).last().let(callback)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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()
|
||||||
|
}
|
||||||
|
}
|
|
@ -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())
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,9 @@
|
||||||
|
package com.lagradost.models
|
||||||
|
|
||||||
|
data class PlayerJson (
|
||||||
|
|
||||||
|
val title : String?,
|
||||||
|
val file : String?,
|
||||||
|
val id: String?,
|
||||||
|
val poster: String?
|
||||||
|
)
|
|
@ -0,0 +1,26 @@
|
||||||
|
package com.lagradost.models
|
||||||
|
|
||||||
|
data class Search (
|
||||||
|
|
||||||
|
val id : Int,
|
||||||
|
val slug : String,
|
||||||
|
val name : String,
|
||||||
|
val longname : String,
|
||||||
|
val engname : String,
|
||||||
|
val age : String,
|
||||||
|
val genre : String,
|
||||||
|
val year : Int,
|
||||||
|
val add_date : String,
|
||||||
|
val country : String,
|
||||||
|
val flagicon : String,
|
||||||
|
val regiser : String,
|
||||||
|
val studio : String,
|
||||||
|
val serii : String,
|
||||||
|
val stan : String,
|
||||||
|
val rating : Int,
|
||||||
|
val description : String,
|
||||||
|
val info_pro_anime : String,
|
||||||
|
val poster : String,
|
||||||
|
val trailer : String,
|
||||||
|
val typeofanime : String,
|
||||||
|
)
|
|
@ -25,7 +25,7 @@ https://git.cakestwix.com/CakesTwix/cloudstream-extensions-uk/raw/branch/master/
|
||||||
|
|
||||||
<!-- Support -->
|
<!-- Support -->
|
||||||
## ✅ Підтримка
|
## ✅ Підтримка
|
||||||
- [Fediverse](https://pl.m0e.space/mangane/@CakesTwix) aka @CakesTwix@shkey.cakestwix.com
|
- [Mastodon](https://pl.m0e.space/mangane/@CakesTwix) aka @CakesTwix@pl.m0e.space
|
||||||
- [Matrix-чат](https://matrix.to/#/#cakestwix:matrix.org)
|
- [Matrix-чат](https://matrix.to/#/#cakestwix:matrix.org)
|
||||||
|
|
||||||
<!-- Contributing -->
|
<!-- Contributing -->
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 4
|
version = 3
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation("com.google.code.gson:gson:2.9.0")
|
implementation("com.google.code.gson:gson:2.9.0")
|
||||||
|
|
|
@ -167,7 +167,7 @@ class UAFlixProvider : MainAPI() {
|
||||||
}
|
}
|
||||||
|
|
||||||
} else { // Player in site
|
} else { // Player in site
|
||||||
val playerRawJson = app.get(playerUrl, referer = "https://uafix.net").document.select("script").html()
|
val playerRawJson = app.get(playerUrl).document.select("script").html()
|
||||||
.substringAfterLast("file:\'")
|
.substringAfterLast("file:\'")
|
||||||
.substringBefore("\',")
|
.substringBefore("\',")
|
||||||
|
|
||||||
|
@ -245,7 +245,7 @@ class UAFlixProvider : MainAPI() {
|
||||||
).last().let(callback)
|
).last().let(callback)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
val playerRawJson = app.get(playerUrl, referer = "https://uafix.net").document.select("script").html()
|
val playerRawJson = app.get(playerUrl).document.select("script").html()
|
||||||
.substringAfterLast("file:\'")
|
.substringAfterLast("file:\'")
|
||||||
.substringBefore("\',")
|
.substringBefore("\',")
|
||||||
tryParseJson<List<PlayerJson>>(playerRawJson)?.map { dubs -> // Dubs
|
tryParseJson<List<PlayerJson>>(playerRawJson)?.map { dubs -> // Dubs
|
||||||
|
@ -263,7 +263,7 @@ class UAFlixProvider : MainAPI() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
val playerRawJson = app.get(dataList[2], referer = "https://uafix.net").document.select("script").html()
|
val playerRawJson = app.get(dataList[2]).document.select("script").html()
|
||||||
.substringAfterLast("file:\'")
|
.substringAfterLast("file:\'")
|
||||||
.substringBefore("\',")
|
.substringBefore("\',")
|
||||||
tryParseJson<List<PlayerJson>>(playerRawJson)?.map { dubs -> // Dubs
|
tryParseJson<List<PlayerJson>>(playerRawJson)?.map { dubs -> // Dubs
|
||||||
|
|
Loading…
Reference in a new issue