diff --git a/src/de/moflixstream/build.gradle b/src/de/moflixstream/build.gradle index 8f53d83f70..a3cda7d280 100644 --- a/src/de/moflixstream/build.gradle +++ b/src/de/moflixstream/build.gradle @@ -1,12 +1,14 @@ -apply plugin: 'com.android.application' -apply plugin: 'kotlin-android' -apply plugin: 'kotlinx-serialization' +plugins { + alias(libs.plugins.android.application) + alias(libs.plugins.kotlin.android) + alias(libs.plugins.kotlin.serialization) +} ext { extName = 'Moflix-Stream' pkgNameSuffix = 'de.moflixstream' extClass = '.MoflixStream' - extVersionCode = 2 + extVersionCode = 3 libVersion = '13' } diff --git a/src/de/moflixstream/src/eu/kanade/tachiyomi/animeextension/de/moflixstream/MoflixStream.kt b/src/de/moflixstream/src/eu/kanade/tachiyomi/animeextension/de/moflixstream/MoflixStream.kt index 75c096f8ad..17e331e96a 100644 --- a/src/de/moflixstream/src/eu/kanade/tachiyomi/animeextension/de/moflixstream/MoflixStream.kt +++ b/src/de/moflixstream/src/eu/kanade/tachiyomi/animeextension/de/moflixstream/MoflixStream.kt @@ -1,10 +1,17 @@ package eu.kanade.tachiyomi.animeextension.de.moflixstream import android.app.Application -import android.content.SharedPreferences import androidx.preference.ListPreference import androidx.preference.MultiSelectListPreference import androidx.preference.PreferenceScreen +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.AnimeDetailsDto +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.EpisodeListDto +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.EpisodePageDto +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.ItemInfo +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.PopularPaginationDto +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.SearchDto +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.SeasonPaginationDto +import eu.kanade.tachiyomi.animeextension.de.moflixstream.dto.VideoResponseDto import eu.kanade.tachiyomi.animeextension.de.moflixstream.extractors.UnpackerExtractor import eu.kanade.tachiyomi.animeextension.de.moflixstream.extractors.VidGuardExtractor import eu.kanade.tachiyomi.animesource.ConfigurableAnimeSource @@ -19,17 +26,11 @@ import eu.kanade.tachiyomi.lib.streamvidextractor.StreamVidExtractor import eu.kanade.tachiyomi.lib.streamwishextractor.StreamWishExtractor import eu.kanade.tachiyomi.network.GET import kotlinx.serialization.json.Json -import kotlinx.serialization.json.JsonObject -import kotlinx.serialization.json.int -import kotlinx.serialization.json.jsonArray -import kotlinx.serialization.json.jsonObject -import kotlinx.serialization.json.jsonPrimitive import okhttp3.Headers -import okhttp3.OkHttpClient -import okhttp3.Request import okhttp3.Response import uy.kohesive.injekt.Injekt import uy.kohesive.injekt.api.get +import uy.kohesive.injekt.injectLazy import kotlin.Exception class MoflixStream : ConfigurableAnimeSource, AnimeHttpSource() { @@ -42,303 +43,169 @@ class MoflixStream : ConfigurableAnimeSource, AnimeHttpSource() { override val supportsLatest = false - override val client: OkHttpClient = network.cloudflareClient + override val client = network.cloudflareClient - private val preferences: SharedPreferences by lazy { + override fun headersBuilder() = super.headersBuilder().add("Referer", "$baseUrl/") + + private val preferences by lazy { Injekt.get().getSharedPreferences("source_$id", 0x0000) } - private val json = Json { - isLenient = true - ignoreUnknownKeys = true - } + private val json: Json by injectLazy() + + private val apiUrl = "$baseUrl/api/v1" - override fun popularAnimeRequest(page: Int): Request = GET( - "$baseUrl/api/v1/channel/345?returnContentOnly=true&restriction=&order=rating:desc&paginate=simple&perPage=50&query=&page=$page", + // ============================== Popular =============================== + override fun popularAnimeRequest(page: Int) = GET( + "$apiUrl/channel/345?returnContentOnly=true&restriction=&order=rating:desc&paginate=simple&perPage=50&query=&page=$page", headers = Headers.headersOf("referer", "$baseUrl/movies?order=rating%3Adesc"), ) override fun popularAnimeParse(response: Response): AnimesPage { - val responseString = response.body.string() - return parsePopularAnimeJson(responseString) - } + val pagination = response.parseAs().pagination - private fun parsePopularAnimeJson(jsonLine: String?): AnimesPage { - val jsonData = jsonLine ?: return AnimesPage(emptyList(), false) - val jObject = json.decodeFromString(jsonData) - val jO = jObject.jsonObject["pagination"]!!.jsonObject - val nextPage = jO.jsonObject["next_page_url"]!!.jsonPrimitive.content - .substringAfter("page=").toInt() - val page = jO.jsonObject["current_page"]!!.jsonPrimitive.int - val hasNextPage = page < nextPage - val array = jO["data"]!!.jsonArray - val animeList = mutableListOf() - for (item in array) { - val anime = SAnime.create() - anime.title = item.jsonObject["name"]!!.jsonPrimitive.content - val animeId = item.jsonObject["id"]!!.jsonPrimitive.content - anime.setUrlWithoutDomain("$baseUrl/api/v1/titles/$animeId?load=images,genres,productionCountries,keywords,videos,primaryVideo,seasons,compactCredits") - anime.thumbnail_url = item.jsonObject["poster"]?.jsonPrimitive?.content ?: item.jsonObject["backdrop"]?.jsonPrimitive?.content - animeList.add(anime) - } + val animeList = pagination.data.parseItems() + val hasNextPage = pagination.current_page < (pagination.next_page ?: 1) return AnimesPage(animeList, hasNextPage) } - // episodes + // =============================== Latest =============================== + override fun latestUpdatesParse(response: Response) = throw Exception("not Used") - override fun episodeListRequest(anime: SAnime): Request = GET(baseUrl + anime.url, headers = Headers.headersOf("referer", baseUrl)) + override fun latestUpdatesRequest(page: Int) = throw Exception("Not used") - override fun episodeListParse(response: Response): List { - val responseString = response.body.string() - val url = response.request.url.toString() - return parseEpisodeAnimeJson(responseString, url) - } + // =============================== Search =============================== + override fun searchAnimeRequest(page: Int, query: String, filters: AnimeFilterList) = GET( + "$apiUrl/search/$query?query=$query", + headers = Headers.headersOf("referer", "$baseUrl/search/$query"), + ) - private fun parseEpisodeAnimeJson(jsonLine: String?, url: String): List { - val jsonData = jsonLine ?: return emptyList() - val jObject = json.decodeFromString(jsonData) - val episodeList = mutableListOf() - val mId = jObject.jsonObject["title"]!!.jsonObject["id"]!!.jsonPrimitive.content - val season = jObject.jsonObject["seasons"]?.jsonObject - if (season != null) { - val dataArray = season.jsonObject["data"]!!.jsonArray - val next = season.jsonObject["next_page"]?.jsonPrimitive?.content - if (next != null) { - val seNextJsonData = client.newCall(GET("$baseUrl/api/v1/titles/$mId/seasons?perPage=8&query=&page=$next", headers = Headers.headersOf("referer", baseUrl))).execute().body.string() - val seNextJObject = json.decodeFromString(seNextJsonData) - val seasonNext = seNextJObject.jsonObject["pagination"]!!.jsonObject - val dataNextArray = seasonNext.jsonObject["data"]!!.jsonArray - val dataAllArray = dataArray.plus(dataNextArray) - for (item in dataAllArray) { - val id = item.jsonObject["title_id"]!!.jsonPrimitive.content - val num = item.jsonObject["number"]!!.jsonPrimitive.content - val seUrl = "$baseUrl/api/v1/titles/$id/seasons/$num?load=episodes,primaryVideo" - val seJsonData = client.newCall(GET(seUrl, headers = Headers.headersOf("referer", baseUrl))).execute().body.string() - val seJObject = json.decodeFromString(seJsonData) - val epObject = seJObject.jsonObject["episodes"]!!.jsonObject - val epDataArray = epObject.jsonObject["data"]!!.jsonArray.reversed() - for (epItem in epDataArray) { - val episode = SEpisode.create() - val seNum = epItem.jsonObject["season_number"]!!.jsonPrimitive.content - val epNum = epItem.jsonObject["episode_number"]!!.jsonPrimitive.content - episode.name = "Staffel $seNum Folge $epNum : " + epItem.jsonObject["name"]!!.jsonPrimitive.content - episode.episode_number = epNum.toFloat() - val epId = epItem.jsonObject["title_id"]!!.jsonPrimitive.content - episode.setUrlWithoutDomain("$baseUrl/api/v1/titles/$epId/seasons/$seNum/episodes/$epNum?load=videos,compactCredits,primaryVideo") - episodeList.add(episode) - } - } - } else { - for (item in dataArray) { - val id = item.jsonObject["title_id"]!!.jsonPrimitive.content - val num = item.jsonObject["number"]!!.jsonPrimitive.content - val seUrl = "$baseUrl/api/v1/titles/$id/seasons/$num?load=episodes,primaryVideo" - val seJsonData = client.newCall(GET(seUrl, headers = Headers.headersOf("referer", baseUrl))).execute().body.string() - val seJObject = json.decodeFromString(seJsonData) - val epObject = seJObject.jsonObject["episodes"]!!.jsonObject - val epDataArray = epObject.jsonObject["data"]!!.jsonArray.reversed() - for (epItem in epDataArray) { - val episode = SEpisode.create() - val seNum = epItem.jsonObject["season_number"]!!.jsonPrimitive.content - val epNum = epItem.jsonObject["episode_number"]!!.jsonPrimitive.content - episode.name = "Staffel $seNum Folge $epNum : " + epItem.jsonObject["name"]!!.jsonPrimitive.content - episode.episode_number = epNum.toFloat() - val epId = epItem.jsonObject["title_id"]!!.jsonPrimitive.content - episode.setUrlWithoutDomain("$baseUrl/api/v1/titles/$epId/seasons/$seNum/episodes/$epNum?load=videos,compactCredits,primaryVideo") - episodeList.add(episode) - } - } - } - } else { - val episode = SEpisode.create() - episode.episode_number = 1F - episode.name = "Film" - episode.setUrlWithoutDomain(url) - episodeList.add(episode) - } - return episodeList + override fun searchAnimeParse(response: Response): AnimesPage { + val data = response.parseAs() + val animeList = data.results.parseItems() + return AnimesPage(animeList, false) } - // Video Extractor + // =========================== Anime Details ============================ + override fun animeDetailsParse(response: Response) = SAnime.create().apply { + val data = response.parseAs().title - override fun videoListRequest(episode: SEpisode): Request { - return GET(baseUrl + episode.url, headers = Headers.headersOf("referer", baseUrl)) + setUrlWithoutDomain("$apiUrl/titles/${data.id}?$ANIME_URL_QUERIES") + title = data.name + thumbnail_url = data.thumbnail + genre = data.genres.joinToString { it.name } + description = data.description } - override fun videoListParse(response: Response): List