Skip to content

Commit

Permalink
added Raveeflix #566
Browse files Browse the repository at this point in the history
  • Loading branch information
olivia committed Jan 31, 2024
1 parent 1040dd6 commit 15d9b59
Show file tree
Hide file tree
Showing 4 changed files with 241 additions and 0 deletions.
27 changes: 27 additions & 0 deletions Raveeflix/build.gradle.kts
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
// use an integer for version numbers
version = 1


cloudstream {
language = "id"
// All of these properties are optional, you can safely remove them

// description = "Lorem Ipsum"
authors = listOf("Hexated")

/**
* Status int as the following:
* 0: Down
* 1: Ok
* 2: Slow
* 3: Beta only
* */
status = 1 // will be 3 if unspecified
tvTypes = listOf(
"AsianDrama",
"TvSeries",
"Movie",
)

iconUrl = "https://www.google.com/s2/favicons?domain=raveeflix.my.id&sz=%size%"
}
2 changes: 2 additions & 0 deletions Raveeflix/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest package="com.hexated"/>
198 changes: 198 additions & 0 deletions Raveeflix/src/main/kotlin/com/hexated/Raveeflix.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,198 @@
package com.hexated

import com.fasterxml.jackson.annotation.JsonProperty
import com.lagradost.cloudstream3.*
import com.lagradost.cloudstream3.LoadResponse.Companion.addActors
import com.lagradost.cloudstream3.utils.*
import org.jsoup.nodes.Element

class Raveeflix : MainAPI() {
override var mainUrl = "https://raveeflix.my.id"
override var name = "Raveeflix"
override val hasMainPage = true
override var lang = "id"
override val supportedTypes =
setOf(
TvType.Movie,
TvType.TvSeries,
TvType.AsianDrama,
)

override val mainPage =
mainPageOf(
"categories/trending" to "Trending",
"tv" to "Tv-Shows",
"drakor" to "Drakor",
"categories/anime" to "Anime",
)

override suspend fun getMainPage(
page: Int,
request: MainPageRequest,
): HomePageResponse {
val pages = if (page > 1) "page/$page/" else ""
val document = app.get("$mainUrl/${request.data}/$pages").document
val home = document.select("section.w-full a.min-w-full").mapNotNull { it.toSearchResult() }
return newHomePageResponse(
HomePageList(
request.name, home, true
)
)
}

private fun Element.toSearchResult(): SearchResponse? {
val title = this.selectFirst("div.text-xl")?.text() ?: return null
val href = fixUrl(this.attr("href"))
val posterUrl = this.selectFirst("div.thumbnail_card")?.attr("style")?.getPoster()

return newMovieSearchResponse(title, href, TvType.Movie) {
this.posterUrl = posterUrl
}
}

override suspend fun search(query: String): List<SearchResponse>? {
val res = app.get("$mainUrl/index.json").text.let { AppUtils.tryParseJson<ArrayList<Index>>(it) }
return res?.filter {
it.title?.contains(
query,
true
) == true && !it.section.equals("Categories", true) && !it.section.equals("Tags", true) && it.permalink?.contains("/episode") == false
}?.mapNotNull {
newMovieSearchResponse(
it.title ?: return@mapNotNull null,
fixUrl(
it.permalink?.substringBefore("episode")?.substringBefore("season")
?: return@mapNotNull null,
),
TvType.Movie,
)
}
}

override suspend fun load(url: String): LoadResponse {
val document = app.get(url).document
val title = document.selectFirst("h1.text-4xl")?.text() ?: "No Title"
val poster = document.selectFirst("div.thumbnail_card, div.w-full.thumbnail_card_related")
?.attr("style")?.getPoster()
val type =
if (document.select("mux-player").isNullOrEmpty()) TvType.TvSeries else TvType.Movie
val tags =
if (type == TvType.TvSeries) {
document.selectFirst("div.movie-details > p:nth-child(1)")
?.ownText()?.split(",")
?.map { it.trim() }
} else {
document.select("span.mr-2")
.map { it.text() }.distinct()
}

val year =
document.selectFirst("div.movie-details > p:nth-child(2), div.max-w-prose.mb-20 > ul > li:nth-child(2) span")
?.ownText()?.substringAfter(",")?.toIntOrNull()
val description =
document.selectFirst("div.lead.text-neutral-500, span#storyline")
?.text()?.trim()
val rating =
document.selectFirst("span#rating")?.text()
?.toRatingInt()
val actors =
document.select("span#cast").text().split(", ")
.map { it.trim() }

val recommendations =
document.select("section.w-full a.min-w-full").mapNotNull { it.toSearchResult() }

return if (type == TvType.TvSeries) {
val section = document.select("div.relative > section.w-full a.min-w-full")
val hasMultipleSeason = section.any { it.attr("href").contains("/season-") }
val episodes =
if (hasMultipleSeason) {
section.apmap { ss ->
val season = ss.selectFirst("div.text-xl")?.text()?.filter { it.isDigit() }
?.toIntOrNull()
app.get(fixUrl(ss.attr("href"))).document.select("div.relative > section.w-full a.min-w-full")
.mapNotNull { eps ->
val name = eps.selectFirst("div.text-xl")?.text()
?: return@mapNotNull null
val href = fixUrl(eps.attr("href"))
val posterUrl = eps.selectFirst("div.thumbnail_card")?.attr("style")
?.getPoster()
Episode(
href,
name,
posterUrl = posterUrl,
season = season
)
}
}.flatten()
} else {
section.mapNotNull { eps ->
val name = eps.selectFirst("div.text-xl")?.text() ?: return@mapNotNull null
val href = fixUrl(eps.attr("href"))
val posterUrl =
eps.selectFirst("div.thumbnail_card")?.attr("style")?.getPoster()
Episode(
href,
name,
posterUrl = posterUrl,
)
}
}
newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes.reversed()) {
this.posterUrl = poster
this.year = year
this.seasonNames
this.plot = description
this.tags = tags
this.rating = rating
addActors(actors)
this.recommendations = recommendations
}
} else {
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
}
}
}

override suspend fun loadLinks(
data: String,
isCasting: Boolean,
subtitleCallback: (SubtitleFile) -> Unit,
callback: (ExtractorLink) -> Unit,
): Boolean {

val video = app.get(data).document.select("mux-player").attr("src")

callback.invoke(
ExtractorLink(
name,
name,
video,
"",
Qualities.Unknown.value,
)
)

return true
}

private fun String.getPoster(): String? {
return fixUrlNull(
this.substringAfter("(")
.substringBefore(")"),
)
}

data class Index(
@JsonProperty("title") val title: String? = null,
@JsonProperty("permalink") val permalink: String? = null,
@JsonProperty("section") val section: String? = null,
)
}
14 changes: 14 additions & 0 deletions Raveeflix/src/main/kotlin/com/hexated/RaveeflixPlugin.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@

package com.hexated

import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
import com.lagradost.cloudstream3.plugins.Plugin
import android.content.Context

@CloudstreamPlugin
class RaveeflixPlugin: Plugin() {
override fun load(context: Context) {
// All providers should be added in this manner. Please don't edit the providers list directly.
registerMainAPI(Raveeflix())
}
}

0 comments on commit 15d9b59

Please sign in to comment.