mirror of
https://github.com/recloudstream/cloudstream-extensions-multilingual.git
synced 2024-08-15 03:15:14 +00:00
MesFilms French Provider (#15)
* Added MesFilmsProvider * Added MesFilmsProvider * fixed additional info bs * finally fixed mesfilms after 30 days lol * Fixed French Stream Providers * Update FrenchStreamProviderPlugin.kt * Added Vido Provider to base cloudstream * Fixed issues (finally lol) Removed tv shows from search * Update MesFilmsProvider.kt --------- Co-authored-by: sarlay
This commit is contained in:
parent
55b4086082
commit
7e4e4e57ae
7 changed files with 467 additions and 157 deletions
25
MesFilmsProvider/build.gradle.kts
Normal file
25
MesFilmsProvider/build.gradle.kts
Normal file
|
@ -0,0 +1,25 @@
|
|||
// use an integer for version numbers
|
||||
version = 1
|
||||
|
||||
|
||||
cloudstream {
|
||||
language = "fr"
|
||||
// All of these properties are optional, you can safely remove them
|
||||
|
||||
// description = "Lorem Ipsum"
|
||||
authors = listOf("Sarlay")
|
||||
|
||||
/**
|
||||
* Status int as the following:
|
||||
* 0: Down
|
||||
* 1: Ok
|
||||
* 2: Slow
|
||||
* 3: Beta only
|
||||
* */
|
||||
status = 1 // will be 3 if unspecified
|
||||
tvTypes = listOf(
|
||||
"Movie",
|
||||
)
|
||||
|
||||
iconUrl = "https://www.google.com/s2/favicons?domain=mesfilms.lol&sz=%size%"
|
||||
}
|
2
MesFilmsProvider/src/main/AndroidManifest.xml
Normal file
2
MesFilmsProvider/src/main/AndroidManifest.xml
Normal file
|
@ -0,0 +1,2 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest package="com.lagradost"/>
|
|
@ -0,0 +1,301 @@
|
|||
package com.lagradost
|
||||
|
||||
import com.fasterxml.jackson.annotation.JsonProperty
|
||||
import com.lagradost.cloudstream3.*
|
||||
import com.lagradost.cloudstream3.LoadResponse.Companion.addRating
|
||||
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
||||
import com.lagradost.cloudstream3.metaproviders.CrossTmdbProvider
|
||||
import com.lagradost.cloudstream3.utils.*
|
||||
import com.lagradost.cloudstream3.utils.AppUtils.parseJson
|
||||
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
|
||||
import org.jsoup.Jsoup
|
||||
|
||||
|
||||
class MesFilmsProvider : MainAPI() {
|
||||
override var mainUrl = "https://mesfilms.lol"
|
||||
override var name = "Mes Films"
|
||||
override val hasQuickSearch = false // recherche rapide (optionel, pas vraimet utile)
|
||||
override val hasMainPage = true // page d'accueil (optionel mais encoragé)
|
||||
override var lang = "fr" // fournisseur est en francais
|
||||
override val supportedTypes = setOf(TvType.Movie) // ici on ne supporte que les films
|
||||
// override val supportedTypes = setOf(TvType.Movie, TvType.TvSeries) // films et series
|
||||
// liste des types: https://recloudstream.github.io/dokka/app/com.lagradost.cloudstream3/-tv-type/index.html
|
||||
|
||||
/**
|
||||
Cherche le site pour un titre spécifique
|
||||
|
||||
La recherche retourne une SearchResponse, qui peut être des classes suivants: AnimeSearchResponse, MovieSearchResponse, TorrentSearchResponse, TvSeriesSearchResponse
|
||||
Chaque classes nécessite des données différentes, mais a en commun le nom, le poster et l'url
|
||||
**/
|
||||
override suspend fun search(query: String): List<SearchResponse> {
|
||||
val link = "$mainUrl/?s=$query"
|
||||
val document = app.get(link).document // on convertit le html en un document
|
||||
return document.select("div.search-page > div.result-item > article") // on séléctione tous les éléments 'enfant' du type articles
|
||||
.filter { article -> // on filtre la liste obtenue de tous les articles
|
||||
val type = article?.selectFirst("> div.image > div.thumbnail > a > span")?.text()
|
||||
?.replace("\t", "")?.replace("\n", "")
|
||||
|
||||
type != "Épisode" // enlève tous les éléments qui sont des épisodes de série
|
||||
}.mapNotNull { div -> // apmap crée une liste des éléments (ici newMovieSearchResponse et newTvSeriesSearchResponse)
|
||||
val posterContainer = div.selectFirst("> div.image > div.thumbnail > a") // selectione le premier élément correspondant à ces critères
|
||||
val type = posterContainer?.selectFirst("> span")?.text()?.replace("\t", "")?.replace("\n", "")
|
||||
// replace enlève tous les '\t' et '\n' du titre
|
||||
val mediaPoster = posterContainer?.selectFirst("> img")?.attr("src") // récupère le texte de l'attribut src de l'élément
|
||||
|
||||
val href = posterContainer?.attr("href") ?: return@mapNotNull null // renvoie une erreur si il n'y a pas de lien vers le média
|
||||
// val test1 = stringVariable ?: "valeur par défault"
|
||||
// val test2 = stringVariable ?: doSomething() // si stringVariable est null, on appelle la fonction doSomething
|
||||
// '?:' est appelé Elvis Operator, si la variable stringVariable est null, alors on utilise la "valeur par défault"
|
||||
// https://stackoverflow.com/questions/48253107/what-does-do-in-kotlin-elvis-operator
|
||||
// val details = div.select("> div.details > div.meta")
|
||||
//val rating = details.select("> span.rating").text()
|
||||
// val year = details.select("> span.year").text().toIntOrNull()
|
||||
|
||||
val title = div.selectFirst("> div.details > div.title > a")?.text().toString()
|
||||
|
||||
when (type) {
|
||||
"FILM" -> (
|
||||
newMovieSearchResponse( // réponse du film qui sera ajoutée à la liste map qui sera ensuite return
|
||||
title,
|
||||
href,
|
||||
TvType.Movie,
|
||||
false
|
||||
) {
|
||||
this.posterUrl = mediaPoster
|
||||
// this.year = year
|
||||
// this.rating = rating
|
||||
}
|
||||
)
|
||||
else -> {
|
||||
return@mapNotNull null // le type n'est pas reconnu ==> enlever
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private data class EmbedUrlClass(
|
||||
@JsonProperty("embed_url") val url: String?,
|
||||
)
|
||||
|
||||
/**
|
||||
* charge la page d'informations, il ya toutes les donées, les épisodes, le résumé etc ...
|
||||
* Il faut retourner soit: AnimeLoadResponse, MovieLoadResponse, TorrentLoadResponse, TvSeriesLoadResponse.
|
||||
*/
|
||||
|
||||
data class EpisodeData(
|
||||
@JsonProperty("url") val url: String,
|
||||
@JsonProperty("episodeNumber") val episodeNumber: String?,
|
||||
)
|
||||
|
||||
override suspend fun load(url: String): LoadResponse {
|
||||
// url est le lien retourné par la fonction search (la variable href) ou la fonction getMainPage
|
||||
val document = app.get(url).document // convertit en document
|
||||
|
||||
val meta = document.selectFirst("div.sheader")
|
||||
val poster = meta?.select("div.poster > img")?.attr("data-src") // récupere le texte de l'attribut 'data-src'
|
||||
|
||||
val title = meta?.select("div.data > h1")?.text() ?: throw ErrorLoadingException("Invalid title")
|
||||
|
||||
val data = meta.select("div.data")
|
||||
val extra = data.select("div.extra")
|
||||
|
||||
val description = extra.select("span.tagline").first()?.text() // first() selectione le premier élément de la liste
|
||||
|
||||
val rating1 = document.select("div.custom_fields > span.valor > b#repimdb > strong").text()
|
||||
val rating2 = document.select("div.custom_fields > span.valor > strong").text()
|
||||
|
||||
val rating = when {
|
||||
!rating1.isNullOrBlank() -> rating1
|
||||
!rating2.isNullOrBlank() -> rating2
|
||||
else -> null
|
||||
}
|
||||
|
||||
|
||||
val date = extra.select("span.date").first()?.text()?.takeLast(4) // prends les 4 dernier chiffres
|
||||
|
||||
val tags = data.select("div.sgeneros > a").apmap {it.text()} // séléctione tous les tags et les ajoutes à une liste
|
||||
|
||||
val postId = document.select("#report-video-button-field > input[name=postID]").first()?.attr("value") // élémennt spécifique à ce site
|
||||
|
||||
val mediaType = if(url.contains("/film/")) {
|
||||
"movie"
|
||||
} else {
|
||||
"TV"
|
||||
}
|
||||
// un api est disponible sur ce site pour récupérer le trailer (lien vers youtube)
|
||||
val trailerUrl = postId?.let{
|
||||
try {
|
||||
val payloadRequest = mapOf(
|
||||
"action" to "doo_player_ajax",
|
||||
"post" to postId,
|
||||
"nume" to "trailer",
|
||||
"type" to mediaType
|
||||
) // on crée les donées de la requetes ici
|
||||
val getTrailer =
|
||||
app.post(
|
||||
"$mainUrl/wp-admin/admin-ajax.php",
|
||||
headers = mapOf("Content-Type" to "application/x-www-form-urlencoded; charset=UTF-8"),
|
||||
data = payloadRequest
|
||||
).text
|
||||
parseJson<EmbedUrlClass>(getTrailer).url
|
||||
// parseJson lit le contenu de la réponse (la variable getTrailer) et cherche la valeur d'embed_url dans cette réponse
|
||||
|
||||
} catch (e: Exception){
|
||||
null
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
if (mediaType == "movie") {
|
||||
return newMovieLoadResponse(
|
||||
title,
|
||||
url,
|
||||
TvType.Movie,
|
||||
url
|
||||
) { // retourne les informations du film
|
||||
this.posterUrl = poster
|
||||
addRating(rating)
|
||||
this.year = date?.toIntOrNull()
|
||||
this.tags = tags
|
||||
this.plot = description
|
||||
trailerUrl?.let{addTrailer(trailerUrl)}
|
||||
}
|
||||
} else // a tv serie
|
||||
{
|
||||
throw ErrorLoadingException("Nothing besides movies are implemented for this provider")
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// récupere les liens .mp4 ou m3u8 directement à partir du paramètre data généré avec la fonction load()
|
||||
override suspend fun loadLinks(
|
||||
data: String, // fournit par load()
|
||||
isCasting: Boolean,
|
||||
subtitleCallback: (SubtitleFile) -> Unit,
|
||||
callback: (ExtractorLink) -> Unit,
|
||||
): Boolean {
|
||||
val document = app.get(data).document
|
||||
|
||||
document.select("ul#playeroptionsul > li:not(#player-option-trailer)").apmap { li -> // séléctione tous les players sauf celui avec l'id player-option-trailer
|
||||
// https://jsoup.org/cookbook/extracting-data/selector-syntax
|
||||
val quality = li.selectFirst("span.title")?.text()
|
||||
val server = li.selectFirst("> span.server")?.text()
|
||||
val languageInfo =
|
||||
|
||||
li.selectFirst("span.flag > img")?.attr("data-src")
|
||||
?.substringAfterLast("/") // séléctione la partie de la chaine de caractère apr_s le dernier /
|
||||
?.replace(".png", "") ?: ""
|
||||
|
||||
val postId = li.attr("data-post")
|
||||
|
||||
val indexOfPlayer = li.attr("data-nume")
|
||||
|
||||
// un api est disponible sur le site pour récupéré les liens vers des embed (iframe) type uqload
|
||||
val payloadRequest = mapOf(
|
||||
"action" to "doo_player_ajax",
|
||||
"post" to postId,
|
||||
"nume" to indexOfPlayer,
|
||||
"type" to "movie"
|
||||
)
|
||||
|
||||
|
||||
val getPlayerEmbed =
|
||||
app.post(
|
||||
"$mainUrl/wp-admin/admin-ajax.php",
|
||||
headers = mapOf("Content-Type" to "application/x-www-form-urlencoded; charset=UTF-8"),
|
||||
data = payloadRequest
|
||||
).text
|
||||
val playerUrl = parseJson<EmbedUrlClass>(getPlayerEmbed).url // récupère l'url de l'embed en lisant le json
|
||||
|
||||
if (playerUrl != null)
|
||||
loadExtractor(
|
||||
httpsify(playerUrl),
|
||||
playerUrl,
|
||||
subtitleCallback
|
||||
) { link -> // charge un extracteur d'extraire le lien direct .mp4
|
||||
callback.invoke(ExtractorLink( // ici je modifie le callback pour ajouter des informations, normalement ce n'est pas nécessaire
|
||||
link.source,
|
||||
link.name + " $languageInfo",
|
||||
link.url,
|
||||
link.referer,
|
||||
getQualityFromName(quality),
|
||||
link.isM3u8,
|
||||
link.headers,
|
||||
link.extractorData
|
||||
))
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
|
||||
override val mainPage = mainPageOf(
|
||||
Pair("$mainUrl/film/", "Récemment ajouté"),
|
||||
Pair("$mainUrl/tendance/?get=movies", "Tendance"),
|
||||
Pair("$mainUrl/evaluations/?get=movies", "Les plus notés"),
|
||||
Pair("$mainUrl/films-classiques/", "Quelques classiques"),
|
||||
)
|
||||
|
||||
override suspend fun getMainPage(
|
||||
page: Int,
|
||||
request: MainPageRequest
|
||||
): HomePageResponse {
|
||||
val document = app.get(request.data).document
|
||||
val movies = document.select("div.items > article.movies")
|
||||
val categoryTitle = request.name
|
||||
val returnList = movies.mapNotNull { article ->
|
||||
// ici si un élément est null, il sera automatiquement enlevé de la liste
|
||||
val poster = article.select("div.poster")
|
||||
val posterUrl = poster.select("> img").attr("data-src")
|
||||
val qualityExtracted = poster.select("> div.mepo > span.quality").text().uppercase()
|
||||
|
||||
val quality = getQualityFromString(
|
||||
when (!qualityExtracted.isNullOrBlank()) {
|
||||
qualityExtracted.contains("WEBRIP") -> {
|
||||
"WebRip"
|
||||
}
|
||||
qualityExtracted.contains("HDRIP") -> {
|
||||
"HD"
|
||||
}
|
||||
qualityExtracted.contains("HDLIGHT") -> {
|
||||
"HD"
|
||||
}
|
||||
qualityExtracted.contains("BDRIP") -> {
|
||||
"BlueRay"
|
||||
}
|
||||
qualityExtracted.contains("DVD") -> {
|
||||
"DVD"
|
||||
}
|
||||
qualityExtracted.contains("CAM") -> {
|
||||
"Cam"
|
||||
}
|
||||
|
||||
else -> {
|
||||
null
|
||||
}
|
||||
}
|
||||
)
|
||||
|
||||
val data = article.select("div.data")
|
||||
val title = data.select("> h3 > a").text()
|
||||
val link = data.select("> h3 > a").attr("href")
|
||||
newMovieSearchResponse(
|
||||
title,
|
||||
link,
|
||||
TvType.Movie,
|
||||
false,
|
||||
) {
|
||||
this.posterUrl = posterUrl
|
||||
this.quality = quality
|
||||
}
|
||||
}
|
||||
if (returnList.isEmpty()) throw ErrorLoadingException("No movies")
|
||||
return HomePageResponse(
|
||||
listOf(
|
||||
HomePageList(categoryTitle, returnList)
|
||||
)
|
||||
)
|
||||
}
|
||||
}
|
|
@ -0,0 +1,14 @@
|
|||
|
||||
package com.lagradost
|
||||
|
||||
import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
|
||||
import com.lagradost.cloudstream3.plugins.Plugin
|
||||
import android.content.Context
|
||||
|
||||
@CloudstreamPlugin
|
||||
class MesFilmsProviderPlugin: Plugin() {
|
||||
override fun load(context: Context) {
|
||||
// All providers should be added in this manner. Please don't edit the providers list directly.
|
||||
registerMainAPI(MesFilmsProvider())
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue