diff --git a/EneyidaProvider/build.gradle.kts b/EneyidaProvider/build.gradle.kts
new file mode 100644
index 0000000..3c2f1d2
--- /dev/null
+++ b/EneyidaProvider/build.gradle.kts
@@ -0,0 +1,27 @@
+// use an integer for version numbers
+version = 2
+
+
+cloudstream {
+ language = "uk"
+ // All of these properties are optional, you can safely remove them
+
+ description = "Дивитися фільми та серіали онлайн в HD якості. У нас можна дивитися кіно онлайн безкоштовно, у високій якості та з якісним українським дубляжем."
+ authors = listOf("CakesTwix")
+
+ /**
+ * Status int as the following:
+ * 0: Down
+ * 1: Ok
+ * 2: Slow
+ * 3: Beta only
+ * */
+ status = 1 // will be 3 if unspecified
+ tvTypes = listOf(
+ "Anime",
+ "TvSeries",
+ "Movie",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=uakino.club&sz=%size%"
+}
\ No newline at end of file
diff --git a/EneyidaProvider/src/main/AndroidManifest.xml b/EneyidaProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/EneyidaProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/EneyidaProvider/src/main/kotlin/com/lagradost/EneyidaProvider.kt b/EneyidaProvider/src/main/kotlin/com/lagradost/EneyidaProvider.kt
new file mode 100644
index 0000000..318beea
--- /dev/null
+++ b/EneyidaProvider/src/main/kotlin/com/lagradost/EneyidaProvider.kt
@@ -0,0 +1,260 @@
+package com.lagradost
+
+import android.util.Log
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addActors
+import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.M3u8Helper
+import org.jsoup.Jsoup
+import org.jsoup.nodes.Document
+import org.jsoup.nodes.Element
+import java.util.*
+
+class EneyidaProvider : MainAPI() {
+
+ // Basic Info
+ override var mainUrl = "https://eneyida.tv"
+ override var name = "Eneyida"
+ override val hasMainPage = true
+ override var lang = "uk"
+ override val hasDownloadSupport = true
+ override val supportedTypes = setOf(
+ TvType.Movie,
+ TvType.TvSeries,
+ TvType.Anime,
+ )
+
+ // Sections
+ override val mainPage = mainPageOf(
+ "$mainUrl/films/page/" to "Фільми",
+ "$mainUrl/series/page/" to "Серіали",
+ "$mainUrl/anime/page/" to "Аніме",
+ "$mainUrl/cartoon/page/" to "Мультфільми",
+ "$mainUrl/cartoon-series/page/" to "Мультсеріали",
+ )
+
+ override suspend fun getMainPage(
+ page: Int,
+ request: MainPageRequest
+ ): HomePageResponse {
+ var document = app.get(request.data + page).document
+
+ val home = document.select("article.short").map {
+ it.toSearchResponse()
+ }
+ return newHomePageResponse(request.name, home)
+ }
+
+ private fun Element.toSearchResponse(): SearchResponse {
+ val title = this.selectFirst("a.short_title")?.text()?.trim().toString()
+ val href = this.selectFirst("div.short_title")?.attr("href").toString()
+ val posterUrl = mainUrl + this.selectFirst("a.short_img img")?.attr("data-src")
+
+ return newMovieSearchResponse(title, href, TvType.Movie) {
+ this.posterUrl = posterUrl
+ }
+
+ }
+
+ 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("div.movie-item.short-item").map {
+ it.toSearchResponse()
+ }
+ }
+
+ // Detailed information
+ override suspend fun load(url: String): LoadResponse {
+ val document = app.get(url).document
+
+ // Parse info
+ val title = document.selectFirst("h1 span.solototle")?.text()?.trim().toString()
+ val poster = fixUrl(document.selectFirst("div.film-poster img")?.attr("src").toString())
+ val tags = document.select("div.film-info > div:nth-child(4) a").map { it.text() }
+ val year = document.select("div.film-info > div:nth-child(2) a").text().toIntOrNull()
+ // TODO: Fix Naruto. Its series, not Movie Lol
+ // https://uakino.club/animeukr/6268-naruto-1-sezon.html
+ val tvType =
+ if (url.contains(Regex("(/anime-series)|(/seriesss)|(/cartoonseries)"))) TvType.TvSeries else TvType.Movie
+ val description = document.selectFirst("div[itemprop=description]")?.text()?.trim()
+ val trailer = document.selectFirst("iframe#pre")?.attr("data-src")
+ val rating = document.selectFirst("div.film-info > div:nth-child(8) div.fi-desc")?.text()
+ ?.substringBefore("/").toRatingInt()
+ val actors = document.select("div.film-info > div:nth-child(6) a").map { it.text() }
+
+ val recommendations = document.select("div#full-slides div.owl-item").map {
+ it.toSearchResponse()
+ }
+
+ // Return to app
+ // Parse Episodes as Series
+ return if (tvType == TvType.TvSeries) {
+ val id = url.split("/").last().split("-").first()
+ val episodes =
+ app.get("$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}")
+ .parsedSafe()?.response.let {
+ Jsoup.parse(it.toString()).select("div.playlists-videos li").mapNotNull { eps ->
+ val href = "$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}"
+ val name = eps.text().trim() // Серія 1
+ if (href.isNotEmpty()) {
+ Episode(
+ "$href,$name", // link, Серія 1
+ name,
+ )
+ } else {
+ null
+ }
+ }
+ }
+ newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes.distinctBy{ it.name }) {
+ this.posterUrl = poster
+ this.year = year
+ this.plot = description
+ this.tags = tags
+ this.rating = rating
+ addActors(actors)
+ this.recommendations = recommendations
+ addTrailer(trailer)
+ }
+ } else { // Parse as Movie.
+ 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)
+ }
+ }
+ }
+
+ // It works when I click to view the series
+ override suspend fun loadLinks(
+ data: String, // link, episode name
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+ val dataList = data.split(",")
+ // TODO: OPTIMIZE code!!! Remove this shitty code as soon as possible!!!!!!
+ if(dataList.size == 1){
+ val id = data.split("/").last().split("-").first()
+ val responseGet = app.get("$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}").parsedSafe()
+ if (responseGet?.success == true) { // Its serial
+ responseGet?.response?.let {
+ Jsoup.parse(it).select("div.playlists-videos li")
+ .mapNotNull { eps ->
+ var href = eps.attr("data-file") // ashdi
+ // Can be without https:
+ if (!href.contains("https://")) {
+ href = "https:$href"
+ }
+ val dub = eps.attr("data-voice") // FanWoxUA
+
+ // Get m3u from player script
+ app.get(href, referer = "$mainUrl/").document.select("script")
+ .map { script ->
+ if (script.data().contains("var player = new Playerjs({")) {
+ val m3uLink = script.data().substringAfterLast("file:\"")
+ .substringBefore("\",")
+
+ // Add as source
+ M3u8Helper.generateM3u8(
+ source = dub,
+ streamUrl = m3uLink,
+ referer = "https://ashdi.vip/"
+ ).forEach(callback)
+ }
+ }
+ }
+ }
+ } else {
+ // Its maybe film
+ val document = app.get(data).document
+ val iframeUrl = document.selectFirst("iframe#pre")?.attr("src")
+ // Get m3u from player script
+ if (iframeUrl != null) {
+ app.get(iframeUrl, referer = "$mainUrl/").document.select("script").map { script ->
+ if (script.data().contains("var player = new Playerjs({")) {
+ val m3uLink = script.data().substringAfterLast("file:\"").substringBefore("\",")
+
+ // Add as source
+ M3u8Helper.generateM3u8(
+ source = document.selectFirst("h1 span.solototle")?.text()?.trim().toString(),
+ streamUrl = m3uLink,
+ referer = "https://ashdi.vip/"
+ ).forEach(callback)
+ }
+ }
+ }
+ }
+ return true
+ }
+
+ val responseGet = app.get(dataList[0]).parsedSafe() // ajax link
+ if (responseGet?.success == true){ // Its serial
+ responseGet?.response?.let {
+ Jsoup.parse(it).select("div.playlists-videos li:contains(${dataList[1]})").mapNotNull { eps ->
+ var href = eps.attr("data-file") // ashdi
+ // Can be without https:
+ if (! href.contains("https://")) {
+ href = "https:$href"
+ }
+ val dub = eps.attr("data-voice") // FanWoxUA
+
+ // Get m3u from player script
+ app.get(href, referer = "$mainUrl/").document.select("script").map { script ->
+ if (script.data().contains("var player = new Playerjs({")) {
+ val m3uLink = script.data().substringAfterLast("file:\"").substringBefore("\",")
+
+ // Add as source
+ M3u8Helper.generateM3u8(
+ source = dub,
+ streamUrl = m3uLink,
+ referer = "https://ashdi.vip/"
+ ).forEach(callback)
+ }
+ }
+ }
+ }
+ } else {
+ // Its maybe film
+ val document = app.get(data).document
+ val iframeUrl = document.selectFirst("iframe#pre")?.attr("src")
+ // Get m3u from player script
+ if (iframeUrl != null) {
+ app.get(iframeUrl, referer = "$mainUrl/").document.select("script").map { script ->
+ if (script.data().contains("var player = new Playerjs({")) {
+ val m3uLink = script.data().substringAfterLast("file:\"").substringBefore("\",")
+
+ // Add as source
+ M3u8Helper.generateM3u8(
+ source = document.selectFirst("h1 span.solototle")?.text()?.trim().toString(),
+ streamUrl = m3uLink,
+ referer = "https://ashdi.vip/"
+ ).forEach(callback)
+ }
+ }
+ }
+ }
+
+
+ return true
+ }
+
+ data class Responses(
+ val success: Boolean?,
+ val response: String,
+ )
+
+}
\ No newline at end of file
diff --git a/EneyidaProvider/src/main/kotlin/com/lagradost/EneyidaProviderPlugin.kt b/EneyidaProvider/src/main/kotlin/com/lagradost/EneyidaProviderPlugin.kt
new file mode 100644
index 0000000..70c405b
--- /dev/null
+++ b/EneyidaProvider/src/main/kotlin/com/lagradost/EneyidaProviderPlugin.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 EneyidaProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(EneyidaProvider())
+ }
+}
\ No newline at end of file