import 'package:mangayomi/bridge_lib.dart'; import 'dart:convert'; class NineAnimeTv extends MProvider { NineAnimeTv({required this.source}); MSource source; final Client client = Client(); @override Future getPopular(int page) async { final res = (await client.get( Uri.parse("${source.baseUrl}/filter?sort=all&page=$page"), )).body; return parseAnimeList(res); } @override Future getLatestUpdates(int page) async { final res = (await client.get( Uri.parse( "${source.baseUrl}/filter?sort=recently_updated&page=$page", ), )).body; return parseAnimeList(res); } @override Future search(String query, int page, FilterList filterList) async { final filters = filterList.filters; String url = "${source.baseUrl}/filter?keyword=$query"; for (var filter in filters) { if (filter.type == "GenreFilter") { final genre = (filter.state as List).where((e) => e.state).toList(); url += "${ll(url)}genre="; if (genre.isNotEmpty) { for (var st in genre) { url += "${st.value}"; if (genre.length > 1) { url += "%2C"; } } if (genre.length > 1) { url = substringBeforeLast(url, '%2C'); } } } else if (filter.type == "SeasonFilter") { final season = (filter.state as List).where((e) => e.state).toList(); url += "${ll(url)}season="; if (season.isNotEmpty) { for (var st in season) { url += "${st.value}"; if (season.length > 1) { url += "%2C"; } } if (season.length > 1) { url = substringBeforeLast(url, '%2C'); } } } else if (filter.type == "YearFilter") { final year = (filter.state as List).where((e) => e.state).toList(); url += "${ll(url)}year="; if (year.isNotEmpty) { for (var st in year) { url += "${st.value}"; if (year.length > 1) { url += "%2C"; } } if (year.length > 1) { url = substringBeforeLast(url, '%2C'); } } } else if (filter.type == "TypeFilter") { final type = (filter.state as List).where((e) => e.state).toList(); url += "${ll(url)}type="; if (type.isNotEmpty) { for (var st in type) { url += "${st.value}"; if (type.length > 1) { url += "%2C"; } } if (type.length > 1) { url = substringBeforeLast(url, '%2C'); } } } else if (filter.type == "StatusFilter") { final status = filter.values[filter.state].value; url += "${ll(url)}status=$status"; } else if (filter.type == "LanguageFilter") { final language = (filter.state as List).where((e) => e.state).toList(); url += "${ll(url)}language="; if (language.isNotEmpty) { for (var st in language) { url += "${st.value}"; if (language.length > 1) { url += "%2C"; } } if (language.length > 1) { url = substringBeforeLast(url, '%2C'); } } } else if (filter.type == "SortFilter") { final sort = filter.values[filter.state].value; url += "${ll(url)}sort=$sort"; } } final res = (await client.get(Uri.parse("$url&page=$page"))).body; return parseAnimeList(res); } @override Future getDetail(String url) async { final statusList = [ {"Currently Airing": 0, "Finished Airing": 1}, ]; final res = (await client.get(Uri.parse("${source.baseUrl}$url"))).body; MManga anime = MManga(); final document = parseHtml(res); final infoElement = document.selectFirst("div.film-infor"); final status = infoElement.xpathFirst( '//div[contains(text(),"Status:")]/following-sibling::div/span/text()', ) ?? ""; anime.status = parseStatus(status, statusList); anime.description = infoElement.selectFirst("div.film-description > p")?.text ?? ""; anime.author = infoElement.xpathFirst( '//div[contains(text(),"Studios:")]/following-sibling::div/a/text()', ) ?? ""; anime.genre = infoElement.xpath( '//div[contains(text(),"Genre:")]/following-sibling::div/a/text()', ); final id = parseHtml(res).selectFirst("div[data-id]").attr("data-id"); final resEp = (await client.get( Uri.parse("${source.baseUrl}/ajax/episode/list/$id"), )).body; final html = json.decode(resEp)["html"]; List? episodesList = []; final epsElements = parseHtml(html).select("a"); for (var epElement in epsElements) { final id = epElement.attr('data-id'); final title = epElement.attr('title') ?? ""; final epNum = epElement.attr('data-number'); MChapter episode = MChapter(); episode.name = "Episode $epNum $title"; episode.url = id; episodesList.add(episode); } anime.chapters = episodesList.reversed.toList(); return anime; } @override Future> getVideoList(String url) async { final res = (await client.get( Uri.parse("${source.baseUrl}/ajax/episode/servers?episodeId=$url"), )).body; final html = json.decode(res)["html"]; final serverElements = parseHtml(html).select("div.server-item"); List videos = []; final hosterSelection = preferenceHosterSelection(source.id); final typeSelection = preferenceTypeSelection(source.id); for (var serverElement in serverElements) { final name = serverElement.text; final id = serverElement.attr("data-id"); final subDub = serverElement.attr("data-type"); final res = (await client.get( Uri.parse("${source.baseUrl}/ajax/episode/sources?id=$id"), )).body; final epUrl = json.decode(res)["link"]; List a = []; if (hosterSelection.contains(name) && typeSelection.contains(subDub)) { if (name.contains("Vidstreaming")) { a = await rapidCloudExtractor(epUrl, "Vidstreaming - $subDub"); } else if (name.contains("Vidcloud")) { a = await rapidCloudExtractor(epUrl, "Vidcloud - $subDub"); } videos.addAll(a); } } return sortVideos(videos, source.id); } MPages parseAnimeList(String res) { final elements = parseHtml(res).select("div.film_list-wrap > div"); List animeList = []; for (var element in elements) { MManga anime = MManga(); anime.name = element.selectFirst("div.film-detail > h3 > a").text; anime.imageUrl = element.selectFirst(" div.film-poster > img").getSrc; anime.link = element.selectFirst("div.film-detail > h3 > a").getHref; animeList.add(anime); } return MPages(animeList, true); } Future> rapidCloudExtractor(String url, String name) async { final serverUrl = ['https://megacloud.tv', 'https://rapid-cloud.co']; final serverType = url.startsWith('https://megacloud.tv') || url.startsWith('https://megacloud.club') ? 0 : 1; final sourceUrl = [ '/embed-2/ajax/e-1/getSources?id=', '/ajax/embed-6-v2/getSources?id=', ]; final sourceSpliter = ['/e-1/', '/embed-6-v2/']; final id = url.split(sourceSpliter[serverType]).last.split('?').first; final resServer = (await client.get( Uri.parse('${serverUrl[serverType]}${sourceUrl[serverType]}$id'), headers: {"X-Requested-With": "XMLHttpRequest"}, )).body; final encrypted = getMapValue(resServer, "encrypted"); String videoResJson = ""; List videos = []; if (encrypted == "true") { final ciphered = getMapValue(resServer, "sources"); List> indexPairs = await generateIndexPairs(serverType); var password = ''; String ciphertext = ciphered; int index = 0; for (List item in json.decode(json.encode(indexPairs))) { int start = item.first + index; int end = start + item.last; String passSubstr = ciphered.substring(start, end); password += passSubstr; ciphertext = ciphertext.replaceFirst(passSubstr, ""); index += item.last; } videoResJson = decryptAESCryptoJS(ciphertext, password); } else { videoResJson = json.encode( (json.decode(resServer)["sources"] as List>), ); } String masterUrl = ((json.decode(videoResJson) as List>) .first)['file']; String type = ((json.decode(videoResJson) as List>) .first)['type']; final tracks = (json.decode(resServer)['tracks'] as List) .where((e) => e['kind'] == 'captions' ? true : false) .toList(); List subtitles = []; for (var sub in tracks) { try { MTrack subtitle = MTrack(); subtitle ..label = sub["label"] ..file = sub["file"]; subtitles.add(subtitle); } catch (_) {} } if (type == "hls") { final masterPlaylistRes = (await client.get(Uri.parse(masterUrl))).body; for (var it in substringAfter( masterPlaylistRes, "#EXT-X-STREAM-INF:", ).split("#EXT-X-STREAM-INF:")) { final quality = "${substringBefore(substringBefore(substringAfter(substringAfter(it, "RESOLUTION="), "x"), ","), "\n")}p"; String videoUrl = substringBefore(substringAfter(it, "\n"), "\n"); if (!videoUrl.startsWith("http")) { videoUrl = "${masterUrl.split("/").sublist(0, masterUrl.split("/").length - 1).join("/")}/$videoUrl"; } MVideo video = MVideo(); video ..url = videoUrl ..originalUrl = videoUrl ..quality = "$name - $quality" ..subtitles = subtitles; videos.add(video); } } else { MVideo video = MVideo(); video ..url = masterUrl ..originalUrl = masterUrl ..quality = "$name - Default" ..subtitles = subtitles; videos.add(video); } return videos; } Future>> generateIndexPairs(int serverType) async { final jsPlayerUrl = [ "https://megacloud.tv/js/player/a/prod/e1-player.min.js", "https://rapid-cloud.co/js/player/prod/e6-player-v2.min.js", ]; final scriptText = (await client.get(Uri.parse(jsPlayerUrl[serverType]))).body; final switchCode = scriptText.substring( scriptText.lastIndexOf('switch'), scriptText.indexOf('=partKey'), ); List indexes = []; for (var variableMatch in RegExp(r'=(\w+)').allMatches(switchCode).toList()) { final regex = RegExp( ',${(variableMatch as RegExpMatch).group(1)}=((?:0x)?([0-9a-fA-F]+))', ); Match? match = regex.firstMatch(scriptText); if (match != null) { String value = match.group(1); if (value.contains("0x")) { indexes.add(int.parse(substringAfter(value, "0x"), radix: 16)); } else { indexes.add(int.parse(value)); } } } return chunked(indexes, 2); } List> chunked(List list, int size) { List> chunks = []; for (int i = 0; i < list.length; i += size) { int end = list.length; if (i + size < list.length) { end = i + size; } chunks.add(list.sublist(i, end)); } return chunks; } @override List getFilterList() { return [ GroupFilter("GenreFilter", "Genre", [ CheckBoxFilter("Action", "1"), CheckBoxFilter("Adventure", "2"), CheckBoxFilter("Cars", "3"), CheckBoxFilter("Comedy", "4"), CheckBoxFilter("Dementia", "5"), CheckBoxFilter("Demons", "6"), CheckBoxFilter("Drama", "8"), CheckBoxFilter("Ecchi", "9"), CheckBoxFilter("Fantasy", "10"), CheckBoxFilter("Game", "11"), CheckBoxFilter("Harem", "35"), CheckBoxFilter("Historical", "13"), CheckBoxFilter("Horror", "14"), CheckBoxFilter("Isekai", "44"), CheckBoxFilter("Josei", "43"), CheckBoxFilter("Kids", "15"), CheckBoxFilter("Magic", "16"), CheckBoxFilter("Martial Arts", "17"), CheckBoxFilter("Mecha", "18"), CheckBoxFilter("Military", "38"), CheckBoxFilter("Music", "19"), CheckBoxFilter("Mystery", "7"), CheckBoxFilter("Parody", "20"), CheckBoxFilter("Police", "39"), CheckBoxFilter("Psychological", "40"), CheckBoxFilter("Romance", "22"), CheckBoxFilter("Samurai", "21"), CheckBoxFilter("School", "23"), CheckBoxFilter("Sci-Fi", "24"), CheckBoxFilter("Seinen", "42"), CheckBoxFilter("Shoujo", "25"), CheckBoxFilter("Shoujo Ai", "26"), CheckBoxFilter("Shounen", "27"), CheckBoxFilter("Shounen Ai", "28"), CheckBoxFilter("Slice of Life", "36"), CheckBoxFilter("Space", "29"), CheckBoxFilter("Sports", "30"), CheckBoxFilter("Super Power", "31"), CheckBoxFilter("Supernatural", "37"), CheckBoxFilter("Thriller", "41"), CheckBoxFilter("Vampire", "32"), ]), GroupFilter("SeasonFilter", "Season", [ CheckBoxFilter("Fall", "3"), CheckBoxFilter("Summer", "2"), CheckBoxFilter("Spring", "1"), CheckBoxFilter("Winter", "4"), ]), GroupFilter("YearFilter", "Year", [ CheckBoxFilter("2024", "2024"), CheckBoxFilter("2023", "2023"), CheckBoxFilter("2022", "2022"), CheckBoxFilter("2021", "2021"), CheckBoxFilter("2020", "2020"), CheckBoxFilter("2019", "2019"), CheckBoxFilter("2018", "2018"), CheckBoxFilter("2017", "2017"), CheckBoxFilter("2016", "2016"), CheckBoxFilter("2015", "2015"), CheckBoxFilter("2014", "2014"), CheckBoxFilter("2013", "2013"), CheckBoxFilter("2012", "2012"), CheckBoxFilter("2011", "2011"), CheckBoxFilter("2010", "2010"), CheckBoxFilter("2009", "2009"), CheckBoxFilter("2008", "2008"), CheckBoxFilter("2007", "2007"), CheckBoxFilter("2006", "2006"), CheckBoxFilter("2005", "2005"), CheckBoxFilter("2004", "2004"), CheckBoxFilter("2003", "2003"), CheckBoxFilter("2002", "2002"), CheckBoxFilter("2001", "2001"), ]), SelectFilter("SortFilter", "Sort by", 0, [ SelectFilterOption("All", "all"), SelectFilterOption("Default", "default"), SelectFilterOption("Recently Added", "recently_added"), SelectFilterOption("Recently Updated", "recently_updated"), SelectFilterOption("Score", "score"), SelectFilterOption("Name A-Z", "name_az"), SelectFilterOption("Released Date", "released_date"), SelectFilterOption("Most Watched", "most_watched"), ]), GroupFilter("TypeFilter", "Type", [ CheckBoxFilter("Movie", "1"), CheckBoxFilter("TV Series", "2"), CheckBoxFilter("OVA", "3"), CheckBoxFilter("ONA", "4"), CheckBoxFilter("Special", "5"), CheckBoxFilter("Music", "6"), ]), SelectFilter("StatusFilter", "Status", 0, [ SelectFilterOption("All", "all"), SelectFilterOption("Finished Airing", "1"), SelectFilterOption("Currently Airing", "2"), SelectFilterOption("Not yet aired", "3"), ]), GroupFilter("LanguageFilter", "Language", [ CheckBoxFilter("Sub", "sub"), CheckBoxFilter("Dub", "dub"), ]), ]; } @override List getSourcePreferences() { return [ ListPreference( key: "preferred_quality", title: "Preferred Quality", summary: "", valueIndex: 1, entries: ["1080p", "720p", "480p", "360p"], entryValues: ["1080", "720", "480", "360"], ), ListPreference( key: "preferred_server", title: "Preferred server", summary: "", valueIndex: 0, entries: ["Vidstreaming", "VidCloud"], entryValues: ["Vidstreaming", "VidCloud"], ), ListPreference( key: "preferred_type", title: "Preferred Type", summary: "", valueIndex: 0, entries: ["Sub", "Dub"], entryValues: ["sub", "dub"], ), MultiSelectListPreference( key: "hoster_selection", title: "Enable/Disable Hosts", summary: "", entries: ["Vidstreaming", "VidCloud"], entryValues: ["Vidstreaming", "Vidcloud"], values: ["Vidstreaming", "Vidcloud"], ), MultiSelectListPreference( key: "type_selection", title: "Enable/Disable Types", summary: "", entries: ["Sub", "Dub"], entryValues: ["sub", "dub"], values: ["sub", "dub"], ), ]; } List sortVideos(List videos, int sourceId) { String quality = getPreferenceValue(sourceId, "preferred_quality"); String server = getPreferenceValue(sourceId, "preferred_server"); String type = getPreferenceValue(sourceId, "preferred_type"); videos.sort((MVideo a, MVideo b) { int qualityMatchA = 0; if (a.quality.contains(quality) && a.quality.toLowerCase().contains(type.toLowerCase()) && a.quality.toLowerCase().contains(server.toLowerCase())) { qualityMatchA = 1; } int qualityMatchB = 0; if (b.quality.contains(quality) && b.quality.toLowerCase().contains(type.toLowerCase()) && b.quality.toLowerCase().contains(server.toLowerCase())) { qualityMatchB = 1; } if (qualityMatchA != qualityMatchB) { return qualityMatchB - qualityMatchA; } final regex = RegExp(r'(\d+)p'); final matchA = regex.firstMatch(a.quality); final matchB = regex.firstMatch(b.quality); final int qualityNumA = int.tryParse(matchA?.group(1) ?? '0') ?? 0; final int qualityNumB = int.tryParse(matchB?.group(1) ?? '0') ?? 0; return qualityNumB - qualityNumA; }); return videos; } List preferenceHosterSelection(int sourceId) { return getPreferenceValue(sourceId, "hoster_selection"); } List preferenceTypeSelection(int sourceId) { return getPreferenceValue(sourceId, "type_selection"); } String ll(String url) { if (url.contains("?")) { return "&"; } return "?"; } } NineAnimeTv main(MSource source) { return NineAnimeTv(source: source); }