diff --git a/BambooUAProvider/build.gradle.kts b/BambooUAProvider/build.gradle.kts
new file mode 100644
index 0000000..090ad47
--- /dev/null
+++ b/BambooUAProvider/build.gradle.kts
@@ -0,0 +1,29 @@
+// use an integer for version numbers
+version = 1
+
+dependencies {
+ implementation("com.google.code.gson:gson:2.8.9")
+}
+
+cloudstream {
+ language = "uk"
+ // All of these properties are optional, you can safely remove them
+
+ description = "BambooUA - Ми команда, що дарує переклади азійських фільмів / шоу / кліпів / дорам українською мовою. Приєднуйтеся до нас! З нами ти завжди будеш першим!"
+ 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",
+ "AsianDrama",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=bambooua.com&sz=%size%"
+}
\ No newline at end of file
diff --git a/BambooUAProvider/src/main/AndroidManifest.xml b/BambooUAProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/BambooUAProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/BambooUAProvider/src/main/kotlin/com/lagradost/BambooUAProvider.kt b/BambooUAProvider/src/main/kotlin/com/lagradost/BambooUAProvider.kt
new file mode 100644
index 0000000..af80ad3
--- /dev/null
+++ b/BambooUAProvider/src/main/kotlin/com/lagradost/BambooUAProvider.kt
@@ -0,0 +1,210 @@
+package com.lagradost
+
+import android.util.Log
+import com.google.gson.Gson
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId
+import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId
+import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.M3u8Helper
+import org.jsoup.nodes.Element
+
+class BambooUAProvider : MainAPI() {
+
+ // Basic Info
+ override var mainUrl = "https://bambooua.com"
+ override var name = "BambooUA"
+ override val hasMainPage = true
+ override var lang = "uk"
+ override val hasDownloadSupport = true
+ override val supportedTypes = setOf(
+ TvType.Anime,
+ TvType.AsianDrama,
+ )
+
+ // Sections
+ override val mainPage = mainPageOf(
+ "$mainUrl/dorama/page/" to "Дорама",
+ "$mainUrl/anime/page/" to "Аніме",
+ "$mainUrl/lakorn/page/" to "Лакорн",
+ "$mainUrl/cinema/page/" to "Кіно",
+ "$mainUrl/voice/page/" to "Озвучення",
+ "$mainUrl/tv-show/page/" to "ТВ-шоу",
+ "$mainUrl/done/page/" to "Завершені",
+ "$mainUrl/world-bl/page/" to "Світ ЛГБТ",
+ "$mainUrl/now/page/" to "Поточні",
+ )
+
+ // Main Page
+ private val animeSelector = "li.slide-item"
+ private val titleSelector = ".block-description > h6"
+ private val hrefSelector = ".block-images .hover-buttons"
+ private val posterSelector = ".img-fluid"
+
+ // Load info
+ // private val titleLoadSelector = ".movie-detail .trending-info .trending-text"
+ private val genresSelector = "span.full_cat a"
+ private val yearSelector = ".trending-info .text-detail span.badge-danger"
+ // private val playerSelector = ".video-responsive > iframe"
+ // private val descriptionSelector = ".full-text"
+ // private val ratingSelector = ".pmovie__subrating img"
+
+ override suspend fun getMainPage(
+ page: Int,
+ request: MainPageRequest
+ ): HomePageResponse {
+ val document = app.get(request.data + page).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()?.trim().toString()
+ val href = this.selectFirst(hrefSelector)?.attr("href").toString()
+ val posterUrl = mainUrl + this.selectFirst(posterSelector)?.attr("src")
+
+
+ val sub = this.select(".Type_project_SUB").text().substringAfter(". ")
+ val dub = this.select(".Type_project_DUB").text().substringAfter(". ")
+ // Log.d("load-debug", dub)
+ return newAnimeSearchResponse(title, href, TvType.Anime) {
+ this.posterUrl = posterUrl
+ addDubStatus(dub.isNotEmpty(), sub.isNotEmpty(), dub.toIntOrNull(), sub.toIntOrNull())
+ }
+
+ }
+
+ 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 json = tryParseJson(document.select("script[type*=json]").html())
+ val gJson = Gson().fromJson(document.select("script[type*=json]").html(), JSONModel::class.java)
+ Log.d("load-debug-json", gJson.graph[0].name)
+
+ val title = gJson.graph[0].name
+
+ val poster = gJson.graph[0].image[1]
+ val tags = document.select(genresSelector).map { it.text() }
+ val year = document.select(yearSelector).text().toIntOrNull()
+
+ val tvType = with(tags){
+ when{
+ contains("Аніме") -> TvType.Anime
+ contains("Кіно") -> TvType.Movie
+ else -> TvType.AsianDrama
+ }
+ }
+ val description = gJson.graph[0].description
+ // val rating = document.select(ratingSelector).next().text().toRatingInt()
+
+ val recommendations = document.select(".favorites-slider li.slide-item").map {
+ it.toSearchResponse()
+ }
+
+ var subEpisodes: List = emptyList()
+ var dubEpisodes: List = emptyList()
+
+ // Parse episodes (sub/dub)
+ document.select(".mt-4").forEach {
+ // Parse sub
+ if(it.select("h3.my-4").text() == "Субтитри"){
+ it.select("span.play_me").forEach{ episode ->
+ subEpisodes = subEpisodes.plus(
+ Episode(
+ episode.attr("data-file"),
+ episode.attr("data-title"),
+ episode = episode.attr("data-title").replace("Серія ","").toIntOrNull(),
+ )
+ )
+ }
+ // Parse dub
+ } else if(it.select("h3.my-4").text() == "Озвучення"){
+ it.select("span.play_me").forEach{ episode ->
+ dubEpisodes = dubEpisodes.plus(
+ Episode(
+ episode.attr("data-file"),
+ episode.attr("data-title"),
+ episode = episode.attr("data-title").replace("Серія ","").toIntOrNull(),
+ )
+ )
+ }
+ }
+ }
+
+ // Return to app
+ // Parse Episodes as Series
+ return if(tvType != TvType.Movie){
+ newAnimeLoadResponse(title, url, tvType) {
+ this.posterUrl = poster
+ this.year = year
+ this.plot = description
+ this.tags = tags
+ this.recommendations = recommendations
+ this.addEpisodes(DubStatus.Dubbed, dubEpisodes)
+ this.addEpisodes(DubStatus.Subbed, subEpisodes)
+ }
+ } else{
+ newMovieLoadResponse(title, url, tvType, url) {
+ this.posterUrl = poster
+ 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,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+
+ // Movie
+ if(data.startsWith("https://bambooua.com")){
+ val document = app.get(data).document
+ document.select("span.mr-3").forEach {
+ Log.d("load-debug", it.attr("data-file"))
+ M3u8Helper.generateM3u8(
+ source = it.attr("data-title"),
+ streamUrl = it.attr("data-file"),
+ referer = ""
+ ).forEach(callback)
+ }
+ return true
+ }
+
+ // Serial
+ M3u8Helper.generateM3u8(
+ source = "Bambooua",
+ streamUrl = data,
+ referer = ""
+ ).forEach(callback)
+
+ return true
+ }
+
+}
diff --git a/BambooUAProvider/src/main/kotlin/com/lagradost/BambooUAProviderPlugin.kt b/BambooUAProvider/src/main/kotlin/com/lagradost/BambooUAProviderPlugin.kt
new file mode 100644
index 0000000..50984cf
--- /dev/null
+++ b/BambooUAProvider/src/main/kotlin/com/lagradost/BambooUAProviderPlugin.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 BambooUAProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(BambooUAProvider())
+ }
+}
diff --git a/BambooUAProvider/src/main/kotlin/com/lagradost/JSONModel.kt b/BambooUAProvider/src/main/kotlin/com/lagradost/JSONModel.kt
new file mode 100644
index 0000000..3824eba
--- /dev/null
+++ b/BambooUAProvider/src/main/kotlin/com/lagradost/JSONModel.kt
@@ -0,0 +1,43 @@
+package com.lagradost
+
+import com.google.gson.annotations.SerializedName
+
+data class JSONModel (
+
+ @SerializedName("@context") val context : String,
+ @SerializedName("@graph") val graph : List
+)
+
+data class graph (
+
+ @SerializedName("@type") val type : String,
+ @SerializedName("@context") val context : String,
+ val publisher : Publisher,
+ val name : String,
+ val headline : String,
+ val mainEntityOfPage : MainEntityOfPage,
+ val datePublished : String,
+ val dateModified : String,
+ val author : Author,
+ val image : List,
+ val description : String
+)
+
+data class Publisher (
+
+ @SerializedName("@type") val type : String,
+ val name : String
+)
+
+data class MainEntityOfPage (
+
+ @SerializedName("@type") val type : String,
+ @SerializedName("@id") val id : String
+)
+
+data class Author (
+
+ @SerializedName("@type") val type : String,
+ val name : String,
+ val url : String
+)