mirror of
https://github.com/recloudstream/cloudstream.git
synced 2024-08-15 01:53:11 +00:00
Merge branch 'master' into agp
This commit is contained in:
commit
f870248e61
39 changed files with 630 additions and 585 deletions
|
@ -164,7 +164,7 @@ dependencies {
|
||||||
androidTestImplementation("androidx.test.espresso:espresso-core:3.5.1")
|
androidTestImplementation("androidx.test.espresso:espresso-core:3.5.1")
|
||||||
|
|
||||||
// Android Core & Lifecycle
|
// Android Core & Lifecycle
|
||||||
implementation("androidx.core:core-ktx:1.12.0")
|
implementation("androidx.core:core-ktx:1.13.1")
|
||||||
implementation("androidx.appcompat:appcompat:1.6.1")
|
implementation("androidx.appcompat:appcompat:1.6.1")
|
||||||
implementation("androidx.navigation:navigation-ui-ktx:2.7.7")
|
implementation("androidx.navigation:navigation-ui-ktx:2.7.7")
|
||||||
implementation("androidx.lifecycle:lifecycle-livedata-ktx:2.7.0")
|
implementation("androidx.lifecycle:lifecycle-livedata-ktx:2.7.0")
|
||||||
|
@ -174,7 +174,7 @@ dependencies {
|
||||||
// Design & UI
|
// Design & UI
|
||||||
implementation("jp.wasabeef:glide-transformations:4.3.0")
|
implementation("jp.wasabeef:glide-transformations:4.3.0")
|
||||||
implementation("androidx.preference:preference-ktx:1.2.1")
|
implementation("androidx.preference:preference-ktx:1.2.1")
|
||||||
implementation("com.google.android.material:material:1.11.0")
|
implementation("com.google.android.material:material:1.12.0")
|
||||||
implementation("androidx.constraintlayout:constraintlayout:2.1.4")
|
implementation("androidx.constraintlayout:constraintlayout:2.1.4")
|
||||||
implementation("androidx.swiperefreshlayout:swiperefreshlayout:1.1.0")
|
implementation("androidx.swiperefreshlayout:swiperefreshlayout:1.1.0")
|
||||||
|
|
||||||
|
@ -185,7 +185,7 @@ dependencies {
|
||||||
|
|
||||||
// For KSP -> Official Annotation Processors are Not Yet Supported for KSP
|
// For KSP -> Official Annotation Processors are Not Yet Supported for KSP
|
||||||
ksp("dev.zacsweers.autoservice:auto-service-ksp:1.1.0")
|
ksp("dev.zacsweers.autoservice:auto-service-ksp:1.1.0")
|
||||||
implementation("com.google.guava:guava:32.1.3-android")
|
implementation("com.google.guava:guava:33.2.0-android")
|
||||||
implementation("dev.zacsweers.autoservice:auto-service-ksp:1.1.0")
|
implementation("dev.zacsweers.autoservice:auto-service-ksp:1.1.0")
|
||||||
|
|
||||||
// Media 3 (ExoPlayer)
|
// Media 3 (ExoPlayer)
|
||||||
|
@ -202,7 +202,7 @@ dependencies {
|
||||||
// PlayBack
|
// PlayBack
|
||||||
implementation("com.jaredrummler:colorpicker:1.1.0") // Subtitle Color Picker
|
implementation("com.jaredrummler:colorpicker:1.1.0") // Subtitle Color Picker
|
||||||
implementation("com.github.recloudstream:media-ffmpeg:1.1.0") // Custom FF-MPEG Lib for Audio Codecs
|
implementation("com.github.recloudstream:media-ffmpeg:1.1.0") // Custom FF-MPEG Lib for Audio Codecs
|
||||||
implementation("com.github.TeamNewPipe.NewPipeExtractor:NewPipeExtractor:6dc25f7b97") /* For Trailers
|
implementation("com.github.teamnewpipe:NewPipeExtractor:fafd471") /* For Trailers
|
||||||
^ Update to Latest Commits if Trailers Misbehave, github.com/TeamNewPipe/NewPipeExtractor/commits/dev */
|
^ Update to Latest Commits if Trailers Misbehave, github.com/TeamNewPipe/NewPipeExtractor/commits/dev */
|
||||||
implementation("com.github.albfernandez:juniversalchardet:2.4.0") // Subtitle Decoding
|
implementation("com.github.albfernandez:juniversalchardet:2.4.0") // Subtitle Decoding
|
||||||
|
|
||||||
|
@ -219,9 +219,7 @@ dependencies {
|
||||||
implementation("com.github.rubensousa:previewseekbar-media3:1.1.1.0") // SeekBar Preview
|
implementation("com.github.rubensousa:previewseekbar-media3:1.1.1.0") // SeekBar Preview
|
||||||
|
|
||||||
// Extensions & Other Libs
|
// Extensions & Other Libs
|
||||||
implementation("org.mozilla:rhino:1.7.13") /* run JavaScript
|
implementation("org.mozilla:rhino:1.7.15") // run JavaScript
|
||||||
^ Don't Bump RhinoJS to 1.7.14,`NoClassDefFoundError` Occurs and Trailers won't play (even with Desugaring)
|
|
||||||
NewPipeExtractor Issue */
|
|
||||||
implementation("me.xdrop:fuzzywuzzy:1.4.0") // Library/Ext Searching with Levenshtein Distance
|
implementation("me.xdrop:fuzzywuzzy:1.4.0") // Library/Ext Searching with Levenshtein Distance
|
||||||
implementation("com.github.LagradOst:SafeFile:0.0.6") // To Prevent the URI File Fu*kery
|
implementation("com.github.LagradOst:SafeFile:0.0.6") // To Prevent the URI File Fu*kery
|
||||||
implementation("org.conscrypt:conscrypt-android:2.5.2") // To Fix SSL Fu*kery on Android 9
|
implementation("org.conscrypt:conscrypt-android:2.5.2") // To Fix SSL Fu*kery on Android 9
|
||||||
|
|
|
@ -29,6 +29,7 @@ import com.google.android.material.chip.ChipGroup
|
||||||
import com.google.android.material.navigationrail.NavigationRailView
|
import com.google.android.material.navigationrail.NavigationRailView
|
||||||
import com.lagradost.cloudstream3.AcraApplication.Companion.getKey
|
import com.lagradost.cloudstream3.AcraApplication.Companion.getKey
|
||||||
import com.lagradost.cloudstream3.AcraApplication.Companion.removeKey
|
import com.lagradost.cloudstream3.AcraApplication.Companion.removeKey
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.resumeApps
|
||||||
import com.lagradost.cloudstream3.databinding.ToastBinding
|
import com.lagradost.cloudstream3.databinding.ToastBinding
|
||||||
import com.lagradost.cloudstream3.mvvm.logError
|
import com.lagradost.cloudstream3.mvvm.logError
|
||||||
import com.lagradost.cloudstream3.ui.player.PlayerEventType
|
import com.lagradost.cloudstream3.ui.player.PlayerEventType
|
||||||
|
|
|
@ -31,19 +31,16 @@ import java.text.SimpleDateFormat
|
||||||
import java.util.*
|
import java.util.*
|
||||||
import kotlin.math.absoluteValue
|
import kotlin.math.absoluteValue
|
||||||
|
|
||||||
const val USER_AGENT =
|
|
||||||
"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/116.0.0.0 Safari/537.36"
|
|
||||||
|
|
||||||
//val baseHeader = mapOf("User-Agent" to USER_AGENT)
|
|
||||||
val mapper = JsonMapper.builder().addModule(kotlinModule())
|
|
||||||
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false).build()!!
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the constant for the all languages preference, if this is set then it is
|
* Defines the constant for the all languages preference, if this is set then it is
|
||||||
* the equivalent of all languages being set
|
* the equivalent of all languages being set
|
||||||
**/
|
**/
|
||||||
const val AllLanguagesName = "universal"
|
const val AllLanguagesName = "universal"
|
||||||
|
|
||||||
|
//val baseHeader = mapOf("User-Agent" to USER_AGENT)
|
||||||
|
val mapper = JsonMapper.builder().addModule(kotlinModule())
|
||||||
|
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false).build()!!
|
||||||
|
|
||||||
object APIHolder {
|
object APIHolder {
|
||||||
val unixTime: Long
|
val unixTime: Long
|
||||||
get() = System.currentTimeMillis() / 1000L
|
get() = System.currentTimeMillis() / 1000L
|
||||||
|
@ -121,7 +118,8 @@ object APIHolder {
|
||||||
|
|
||||||
fun LoadResponse.getId(): Int {
|
fun LoadResponse.getId(): Int {
|
||||||
// this fixes an issue with outdated api as getLoadResponseIdFromUrl might be fucked
|
// this fixes an issue with outdated api as getLoadResponseIdFromUrl might be fucked
|
||||||
return (if (this is ResultViewModel2.LoadResponseFromSearch) this.id else null) ?: getLoadResponseIdFromUrl(url, apiName)
|
return (if (this is ResultViewModel2.LoadResponseFromSearch) this.id else null)
|
||||||
|
?: getLoadResponseIdFromUrl(url, apiName)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -222,10 +220,15 @@ object APIHolder {
|
||||||
} ?: false
|
} ?: false
|
||||||
|
|
||||||
val matchingTypes = types?.any { it.name.equals(media.format, true) } == true
|
val matchingTypes = types?.any { it.name.equals(media.format, true) } == true
|
||||||
if(lessAccurate) matchingTitles || matchingTypes && matchingYears else matchingTitles && matchingTypes && matchingYears
|
if (lessAccurate) matchingTitles || matchingTypes && matchingYears else matchingTitles && matchingTypes && matchingYears
|
||||||
} ?: return null
|
} ?: return null
|
||||||
|
|
||||||
Tracker(res.idMal, res.id.toString(), res.coverImage?.extraLarge ?: res.coverImage?.large, res.bannerImage)
|
Tracker(
|
||||||
|
res.idMal,
|
||||||
|
res.id.toString(),
|
||||||
|
res.coverImage?.extraLarge ?: res.coverImage?.large,
|
||||||
|
res.bannerImage
|
||||||
|
)
|
||||||
} catch (t: Throwable) {
|
} catch (t: Throwable) {
|
||||||
logError(t)
|
logError(t)
|
||||||
null
|
null
|
||||||
|
@ -866,6 +869,7 @@ enum class TvType(value: Int?) {
|
||||||
Others(12),
|
Others(12),
|
||||||
Music(13),
|
Music(13),
|
||||||
AudioBook(14),
|
AudioBook(14),
|
||||||
|
|
||||||
/** Wont load the built in player, make your own interaction */
|
/** Wont load the built in player, make your own interaction */
|
||||||
CustomMedia(15),
|
CustomMedia(15),
|
||||||
}
|
}
|
||||||
|
@ -1253,13 +1257,15 @@ interface LoadResponse {
|
||||||
|
|
||||||
fun LoadResponse.getImdbId(): String? {
|
fun LoadResponse.getImdbId(): String? {
|
||||||
return normalSafeApiCall {
|
return normalSafeApiCall {
|
||||||
SimklApi.readIdFromString(this.syncData[simklIdPrefix])?.get(SimklApi.Companion.SyncServices.Imdb)
|
SimklApi.readIdFromString(this.syncData[simklIdPrefix])
|
||||||
|
?.get(SimklApi.Companion.SyncServices.Imdb)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun LoadResponse.getTMDbId(): String? {
|
fun LoadResponse.getTMDbId(): String? {
|
||||||
return normalSafeApiCall {
|
return normalSafeApiCall {
|
||||||
SimklApi.readIdFromString(this.syncData[simklIdPrefix])?.get(SimklApi.Companion.SyncServices.Tmdb)
|
SimklApi.readIdFromString(this.syncData[simklIdPrefix])
|
||||||
|
?.get(SimklApi.Companion.SyncServices.Tmdb)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1556,8 +1562,26 @@ data class TorrentLoadResponse(
|
||||||
posterHeaders: Map<String, String>? = null,
|
posterHeaders: Map<String, String>? = null,
|
||||||
backgroundPosterUrl: String? = null,
|
backgroundPosterUrl: String? = null,
|
||||||
) : this(
|
) : this(
|
||||||
name, url, apiName, magnet, torrent, plot, type, posterUrl, year, rating, tags, duration, trailers,
|
name,
|
||||||
recommendations, actors, comingSoon, syncData, posterHeaders, backgroundPosterUrl, null
|
url,
|
||||||
|
apiName,
|
||||||
|
magnet,
|
||||||
|
torrent,
|
||||||
|
plot,
|
||||||
|
type,
|
||||||
|
posterUrl,
|
||||||
|
year,
|
||||||
|
rating,
|
||||||
|
tags,
|
||||||
|
duration,
|
||||||
|
trailers,
|
||||||
|
recommendations,
|
||||||
|
actors,
|
||||||
|
comingSoon,
|
||||||
|
syncData,
|
||||||
|
posterHeaders,
|
||||||
|
backgroundPosterUrl,
|
||||||
|
null
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1609,7 +1633,8 @@ data class AnimeLoadResponse(
|
||||||
return this.episodes.maxOf { (_, episodes) ->
|
return this.episodes.maxOf { (_, episodes) ->
|
||||||
episodes.count { episodeData ->
|
episodes.count { episodeData ->
|
||||||
// Prioritize display season as actual season may be something random to fit multiple seasons into one.
|
// Prioritize display season as actual season may be something random to fit multiple seasons into one.
|
||||||
val episodeSeason = displayMap[episodeData.season] ?: episodeData.season ?: Int.MIN_VALUE
|
val episodeSeason =
|
||||||
|
displayMap[episodeData.season] ?: episodeData.season ?: Int.MIN_VALUE
|
||||||
// Count all episodes from season 1 to below the current season.
|
// Count all episodes from season 1 to below the current season.
|
||||||
episodeSeason in 1..<season
|
episodeSeason in 1..<season
|
||||||
}
|
}
|
||||||
|
@ -1646,9 +1671,31 @@ data class AnimeLoadResponse(
|
||||||
seasonNames: List<SeasonData>? = null,
|
seasonNames: List<SeasonData>? = null,
|
||||||
backgroundPosterUrl: String? = null,
|
backgroundPosterUrl: String? = null,
|
||||||
) : this(
|
) : this(
|
||||||
engName, japName, name, url, apiName, type, posterUrl, year, episodes, showStatus, plot, tags,
|
engName,
|
||||||
synonyms, rating, duration, trailers, recommendations, actors, comingSoon, syncData, posterHeaders,
|
japName,
|
||||||
nextAiring, seasonNames, backgroundPosterUrl, null
|
name,
|
||||||
|
url,
|
||||||
|
apiName,
|
||||||
|
type,
|
||||||
|
posterUrl,
|
||||||
|
year,
|
||||||
|
episodes,
|
||||||
|
showStatus,
|
||||||
|
plot,
|
||||||
|
tags,
|
||||||
|
synonyms,
|
||||||
|
rating,
|
||||||
|
duration,
|
||||||
|
trailers,
|
||||||
|
recommendations,
|
||||||
|
actors,
|
||||||
|
comingSoon,
|
||||||
|
syncData,
|
||||||
|
posterHeaders,
|
||||||
|
nextAiring,
|
||||||
|
seasonNames,
|
||||||
|
backgroundPosterUrl,
|
||||||
|
null
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1780,7 +1827,7 @@ data class MovieLoadResponse(
|
||||||
backgroundPosterUrl: String? = null,
|
backgroundPosterUrl: String? = null,
|
||||||
) : this(
|
) : this(
|
||||||
name, url, apiName, type, dataUrl, posterUrl, year, plot, rating, tags, duration, trailers,
|
name, url, apiName, type, dataUrl, posterUrl, year, plot, rating, tags, duration, trailers,
|
||||||
recommendations, actors, comingSoon, syncData, posterHeaders, backgroundPosterUrl,null
|
recommendations, actors, comingSoon, syncData, posterHeaders, backgroundPosterUrl, null
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1923,7 +1970,8 @@ data class TvSeriesLoadResponse(
|
||||||
|
|
||||||
return episodes.count { episodeData ->
|
return episodes.count { episodeData ->
|
||||||
// Prioritize display season as actual season may be something random to fit multiple seasons into one.
|
// Prioritize display season as actual season may be something random to fit multiple seasons into one.
|
||||||
val episodeSeason = displayMap[episodeData.season] ?: episodeData.season ?: Int.MIN_VALUE
|
val episodeSeason =
|
||||||
|
displayMap[episodeData.season] ?: episodeData.season ?: Int.MIN_VALUE
|
||||||
// Count all episodes from season 1 to below the current season.
|
// Count all episodes from season 1 to below the current season.
|
||||||
episodeSeason in 1..<season
|
episodeSeason in 1..<season
|
||||||
} + episode
|
} + episode
|
||||||
|
@ -1956,9 +2004,28 @@ data class TvSeriesLoadResponse(
|
||||||
seasonNames: List<SeasonData>? = null,
|
seasonNames: List<SeasonData>? = null,
|
||||||
backgroundPosterUrl: String? = null,
|
backgroundPosterUrl: String? = null,
|
||||||
) : this(
|
) : this(
|
||||||
name, url, apiName, type, episodes, posterUrl, year, plot, showStatus, rating, tags, duration,
|
name,
|
||||||
trailers, recommendations, actors, comingSoon, syncData, posterHeaders, nextAiring, seasonNames,
|
url,
|
||||||
backgroundPosterUrl, null
|
apiName,
|
||||||
|
type,
|
||||||
|
episodes,
|
||||||
|
posterUrl,
|
||||||
|
year,
|
||||||
|
plot,
|
||||||
|
showStatus,
|
||||||
|
rating,
|
||||||
|
tags,
|
||||||
|
duration,
|
||||||
|
trailers,
|
||||||
|
recommendations,
|
||||||
|
actors,
|
||||||
|
comingSoon,
|
||||||
|
syncData,
|
||||||
|
posterHeaders,
|
||||||
|
nextAiring,
|
||||||
|
seasonNames,
|
||||||
|
backgroundPosterUrl,
|
||||||
|
null
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2022,6 +2089,7 @@ data class AniSearch(
|
||||||
@JsonProperty("extraLarge") var extraLarge: String? = null,
|
@JsonProperty("extraLarge") var extraLarge: String? = null,
|
||||||
@JsonProperty("large") var large: String? = null,
|
@JsonProperty("large") var large: String? = null,
|
||||||
)
|
)
|
||||||
|
|
||||||
data class Title(
|
data class Title(
|
||||||
@JsonProperty("romaji") var romaji: String? = null,
|
@JsonProperty("romaji") var romaji: String? = null,
|
||||||
@JsonProperty("english") var english: String? = null,
|
@JsonProperty("english") var english: String? = null,
|
||||||
|
|
|
@ -174,7 +174,6 @@ import java.net.URLDecoder
|
||||||
import java.nio.charset.Charset
|
import java.nio.charset.Charset
|
||||||
import kotlin.math.abs
|
import kotlin.math.abs
|
||||||
import kotlin.math.absoluteValue
|
import kotlin.math.absoluteValue
|
||||||
import kotlin.reflect.KClass
|
|
||||||
import kotlin.system.exitProcess
|
import kotlin.system.exitProcess
|
||||||
|
|
||||||
//https://github.com/videolan/vlc-android/blob/3706c4be2da6800b3d26344fc04fab03ffa4b860/application/vlc-android/src/org/videolan/vlc/gui/video/VideoPlayerActivity.kt#L1898
|
//https://github.com/videolan/vlc-android/blob/3706c4be2da6800b3d26344fc04fab03ffa4b860/application/vlc-android/src/org/videolan/vlc/gui/video/VideoPlayerActivity.kt#L1898
|
||||||
|
@ -187,117 +186,93 @@ import kotlin.system.exitProcess
|
||||||
|
|
||||||
//https://github.com/jellyfin/jellyfin-android/blob/6cbf0edf84a3da82347c8d59b5d5590749da81a9/app/src/main/java/org/jellyfin/mobile/bridge/ExternalPlayer.kt#L225
|
//https://github.com/jellyfin/jellyfin-android/blob/6cbf0edf84a3da82347c8d59b5d5590749da81a9/app/src/main/java/org/jellyfin/mobile/bridge/ExternalPlayer.kt#L225
|
||||||
|
|
||||||
const val VLC_PACKAGE = "org.videolan.vlc"
|
|
||||||
const val MPV_PACKAGE = "is.xyz.mpv"
|
|
||||||
const val WEB_VIDEO_CAST_PACKAGE = "com.instantbits.cast.webvideo"
|
|
||||||
|
|
||||||
val VLC_COMPONENT = ComponentName(VLC_PACKAGE, "$VLC_PACKAGE.gui.video.VideoPlayerActivity")
|
|
||||||
val MPV_COMPONENT = ComponentName(MPV_PACKAGE, "$MPV_PACKAGE.MPVActivity")
|
|
||||||
|
|
||||||
//TODO REFACTOR AF
|
|
||||||
open class ResultResume(
|
|
||||||
val packageString: String,
|
|
||||||
val action: String = Intent.ACTION_VIEW,
|
|
||||||
val position: String? = null,
|
|
||||||
val duration: String? = null,
|
|
||||||
var launcher: ActivityResultLauncher<Intent>? = null,
|
|
||||||
) {
|
|
||||||
val defaultTime = -1L
|
|
||||||
|
|
||||||
val lastId get() = "${packageString}_last_open_id"
|
|
||||||
suspend fun launch(id: Int?, callback: suspend Intent.() -> Unit) {
|
|
||||||
val intent = Intent(action)
|
|
||||||
|
|
||||||
if (id != null)
|
|
||||||
setKey(lastId, id)
|
|
||||||
else
|
|
||||||
removeKey(lastId)
|
|
||||||
|
|
||||||
intent.setPackage(packageString)
|
|
||||||
callback.invoke(intent)
|
|
||||||
launcher?.launch(intent)
|
|
||||||
}
|
|
||||||
|
|
||||||
open fun getPosition(intent: Intent?): Long {
|
|
||||||
return defaultTime
|
|
||||||
}
|
|
||||||
|
|
||||||
open fun getDuration(intent: Intent?): Long {
|
|
||||||
return defaultTime
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val VLC = object : ResultResume(
|
|
||||||
VLC_PACKAGE,
|
|
||||||
// Android 13 intent restrictions fucks up specifically launching the VLC player
|
|
||||||
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) {
|
|
||||||
"org.videolan.vlc.player.result"
|
|
||||||
} else {
|
|
||||||
Intent.ACTION_VIEW
|
|
||||||
},
|
|
||||||
"extra_position",
|
|
||||||
"extra_duration",
|
|
||||||
) {
|
|
||||||
override fun getPosition(intent: Intent?): Long {
|
|
||||||
return intent?.getLongExtra(this.position, defaultTime) ?: defaultTime
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun getDuration(intent: Intent?): Long {
|
|
||||||
return intent?.getLongExtra(this.duration, defaultTime) ?: defaultTime
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val MPV = object : ResultResume(
|
|
||||||
MPV_PACKAGE,
|
|
||||||
//"is.xyz.mpv.MPVActivity.result", // resume not working :pensive:
|
|
||||||
position = "position",
|
|
||||||
duration = "duration",
|
|
||||||
) {
|
|
||||||
override fun getPosition(intent: Intent?): Long {
|
|
||||||
return intent?.getIntExtra(this.position, defaultTime.toInt())?.toLong() ?: defaultTime
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun getDuration(intent: Intent?): Long {
|
|
||||||
return intent?.getIntExtra(this.duration, defaultTime.toInt())?.toLong() ?: defaultTime
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val WEB_VIDEO = ResultResume(WEB_VIDEO_CAST_PACKAGE)
|
|
||||||
|
|
||||||
val resumeApps = arrayOf(
|
|
||||||
VLC, MPV, WEB_VIDEO
|
|
||||||
)
|
|
||||||
|
|
||||||
// Short name for requests client to make it nicer to use
|
|
||||||
|
|
||||||
var app = Requests(responseParser = object : ResponseParser {
|
|
||||||
val mapper: ObjectMapper = jacksonObjectMapper().configure(
|
|
||||||
DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES,
|
|
||||||
false
|
|
||||||
)
|
|
||||||
|
|
||||||
override fun <T : Any> parse(text: String, kClass: KClass<T>): T {
|
|
||||||
return mapper.readValue(text, kClass.java)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun <T : Any> parseSafe(text: String, kClass: KClass<T>): T? {
|
|
||||||
return try {
|
|
||||||
mapper.readValue(text, kClass.java)
|
|
||||||
} catch (e: Exception) {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun writeValueAsString(obj: Any): String {
|
|
||||||
return mapper.writeValueAsString(obj)
|
|
||||||
}
|
|
||||||
}).apply {
|
|
||||||
defaultHeaders = mapOf("user-agent" to USER_AGENT)
|
|
||||||
}
|
|
||||||
|
|
||||||
class MainActivity : AppCompatActivity(), ColorPickerDialogListener,
|
class MainActivity : AppCompatActivity(), ColorPickerDialogListener,
|
||||||
BiometricAuthenticator.BiometricAuthCallback {
|
BiometricAuthenticator.BiometricAuthCallback {
|
||||||
companion object {
|
companion object {
|
||||||
|
const val VLC_PACKAGE = "org.videolan.vlc"
|
||||||
|
const val MPV_PACKAGE = "is.xyz.mpv"
|
||||||
|
const val WEB_VIDEO_CAST_PACKAGE = "com.instantbits.cast.webvideo"
|
||||||
|
|
||||||
|
val VLC_COMPONENT = ComponentName(VLC_PACKAGE, "$VLC_PACKAGE.gui.video.VideoPlayerActivity")
|
||||||
|
val MPV_COMPONENT = ComponentName(MPV_PACKAGE, "$MPV_PACKAGE.MPVActivity")
|
||||||
|
|
||||||
|
//TODO REFACTOR AF
|
||||||
|
open class ResultResume(
|
||||||
|
val packageString: String,
|
||||||
|
val action: String = Intent.ACTION_VIEW,
|
||||||
|
val position: String? = null,
|
||||||
|
val duration: String? = null,
|
||||||
|
var launcher: ActivityResultLauncher<Intent>? = null,
|
||||||
|
) {
|
||||||
|
val defaultTime = -1L
|
||||||
|
|
||||||
|
val lastId get() = "${packageString}_last_open_id"
|
||||||
|
suspend fun launch(id: Int?, callback: suspend Intent.() -> Unit) {
|
||||||
|
val intent = Intent(action)
|
||||||
|
|
||||||
|
if (id != null)
|
||||||
|
setKey(lastId, id)
|
||||||
|
else
|
||||||
|
removeKey(lastId)
|
||||||
|
|
||||||
|
intent.setPackage(packageString)
|
||||||
|
callback.invoke(intent)
|
||||||
|
launcher?.launch(intent)
|
||||||
|
}
|
||||||
|
|
||||||
|
open fun getPosition(intent: Intent?): Long {
|
||||||
|
return defaultTime
|
||||||
|
}
|
||||||
|
|
||||||
|
open fun getDuration(intent: Intent?): Long {
|
||||||
|
return defaultTime
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val VLC = object : ResultResume(
|
||||||
|
VLC_PACKAGE,
|
||||||
|
// Android 13 intent restrictions fucks up specifically launching the VLC player
|
||||||
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) {
|
||||||
|
"org.videolan.vlc.player.result"
|
||||||
|
} else {
|
||||||
|
Intent.ACTION_VIEW
|
||||||
|
},
|
||||||
|
"extra_position",
|
||||||
|
"extra_duration",
|
||||||
|
) {
|
||||||
|
override fun getPosition(intent: Intent?): Long {
|
||||||
|
return intent?.getLongExtra(this.position, defaultTime) ?: defaultTime
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getDuration(intent: Intent?): Long {
|
||||||
|
return intent?.getLongExtra(this.duration, defaultTime) ?: defaultTime
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val MPV = object : ResultResume(
|
||||||
|
MPV_PACKAGE,
|
||||||
|
//"is.xyz.mpv.MPVActivity.result", // resume not working :pensive:
|
||||||
|
position = "position",
|
||||||
|
duration = "duration",
|
||||||
|
) {
|
||||||
|
override fun getPosition(intent: Intent?): Long {
|
||||||
|
return intent?.getIntExtra(this.position, defaultTime.toInt())?.toLong()
|
||||||
|
?: defaultTime
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getDuration(intent: Intent?): Long {
|
||||||
|
return intent?.getIntExtra(this.duration, defaultTime.toInt())?.toLong()
|
||||||
|
?: defaultTime
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val WEB_VIDEO = ResultResume(WEB_VIDEO_CAST_PACKAGE)
|
||||||
|
|
||||||
|
val resumeApps = arrayOf(
|
||||||
|
VLC, MPV, WEB_VIDEO
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
const val TAG = "MAINACT"
|
const val TAG = "MAINACT"
|
||||||
const val ANIMATED_OUTLINE: Boolean = false
|
const val ANIMATED_OUTLINE: Boolean = false
|
||||||
var lastError: String? = null
|
var lastError: String? = null
|
||||||
|
@ -677,7 +652,7 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun dispatchKeyEvent(event: KeyEvent?): Boolean {
|
override fun dispatchKeyEvent(event: KeyEvent): Boolean {
|
||||||
val response = CommonActivity.dispatchKeyEvent(this, event)
|
val response = CommonActivity.dispatchKeyEvent(this, event)
|
||||||
if (response != null)
|
if (response != null)
|
||||||
return response
|
return response
|
||||||
|
@ -1403,7 +1378,7 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
observe(viewModel.watchStatus,::setWatchStatus)
|
observe(viewModel.watchStatus, ::setWatchStatus)
|
||||||
observe(syncViewModel.userData, ::setUserData)
|
observe(syncViewModel.userData, ::setUserData)
|
||||||
observeNullable(viewModel.subscribeStatus, ::setSubscribeStatus)
|
observeNullable(viewModel.subscribeStatus, ::setSubscribeStatus)
|
||||||
|
|
||||||
|
@ -1831,7 +1806,7 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener,
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onAuthenticationError() {
|
override fun onAuthenticationError() {
|
||||||
finish()
|
finish()
|
||||||
}
|
}
|
||||||
|
|
||||||
private var backPressedCallback: OnBackPressedCallback? = null
|
private var backPressedCallback: OnBackPressedCallback? = null
|
||||||
|
|
|
@ -21,3 +21,8 @@ class FourPlayRu : ContentX() {
|
||||||
override var name = "FourPlayRu"
|
override var name = "FourPlayRu"
|
||||||
override var mainUrl = "https://four.playru.net"
|
override var mainUrl = "https://four.playru.net"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class FourPichive : ContentX() {
|
||||||
|
override var name = "FourPichive"
|
||||||
|
override var mainUrl = "https://four.pichive.online"
|
||||||
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ import android.net.Uri
|
||||||
import com.lagradost.cloudstream3.*
|
import com.lagradost.cloudstream3.*
|
||||||
import com.fasterxml.jackson.annotation.JsonAlias
|
import com.fasterxml.jackson.annotation.JsonAlias
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty
|
import com.fasterxml.jackson.annotation.JsonProperty
|
||||||
|
import com.lagradost.cloudstream3.APIHolder.unixTimeMS
|
||||||
import com.lagradost.cloudstream3.LoadResponse.Companion.addImdbId
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addImdbId
|
||||||
import com.lagradost.cloudstream3.LoadResponse.Companion.addTMDbId
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addTMDbId
|
||||||
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
||||||
|
@ -166,6 +167,7 @@ open class TraktProvider : MainAPI() {
|
||||||
val episodes = mutableListOf<Episode>()
|
val episodes = mutableListOf<Episode>()
|
||||||
val seasons = parseJson<List<Seasons>>(resSeasons)
|
val seasons = parseJson<List<Seasons>>(resSeasons)
|
||||||
val seasonsNames = mutableListOf<SeasonData>()
|
val seasonsNames = mutableListOf<SeasonData>()
|
||||||
|
var nextAir: NextAiring? = null
|
||||||
|
|
||||||
seasons.forEach { season ->
|
seasons.forEach { season ->
|
||||||
|
|
||||||
|
@ -215,6 +217,13 @@ open class TraktProvider : MainAPI() {
|
||||||
description = episode.overview,
|
description = episode.overview,
|
||||||
).apply {
|
).apply {
|
||||||
this.addDate(episode.firstAired)
|
this.addDate(episode.firstAired)
|
||||||
|
if (nextAir == null && this.date != null && this.date!! > unixTimeMS) {
|
||||||
|
nextAir = NextAiring(
|
||||||
|
episode = this.episode!!,
|
||||||
|
unixTime = this.date!!.div(1000L),
|
||||||
|
season = if (this.season == 1) null else this.season,
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -240,6 +249,7 @@ open class TraktProvider : MainAPI() {
|
||||||
this.actors = actors
|
this.actors = actors
|
||||||
this.comingSoon = isUpcoming(mediaDetails.released)
|
this.comingSoon = isUpcoming(mediaDetails.released)
|
||||||
//posterHeaders
|
//posterHeaders
|
||||||
|
this.nextAiring = nextAir
|
||||||
this.seasonNames = seasonsNames
|
this.seasonNames = seasonsNames
|
||||||
this.backgroundPosterUrl = getOriginalWidthImageUrl(backDropUrl)
|
this.backgroundPosterUrl = getOriginalWidthImageUrl(backDropUrl)
|
||||||
this.contentRating = mediaDetails.certification
|
this.contentRating = mediaDetails.certification
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
package com.lagradost.cloudstream3.subtitles
|
package com.lagradost.cloudstream3.subtitles
|
||||||
|
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse
|
||||||
import com.lagradost.cloudstream3.TvType
|
import com.lagradost.cloudstream3.TvType
|
||||||
|
|
||||||
class AbstractSubtitleEntities {
|
class AbstractSubtitleEntities {
|
||||||
|
@ -19,8 +20,11 @@ class AbstractSubtitleEntities {
|
||||||
|
|
||||||
data class SubtitleSearch(
|
data class SubtitleSearch(
|
||||||
var query: String = "",
|
var query: String = "",
|
||||||
var imdb: Long? = null,
|
|
||||||
var lang: String? = null,
|
var lang: String? = null,
|
||||||
|
var imdbId: String? = null,
|
||||||
|
var tmdbId: Int? = null,
|
||||||
|
var malId: Int? = null,
|
||||||
|
var aniListId: Int? = null,
|
||||||
var epNumber: Int? = null,
|
var epNumber: Int? = null,
|
||||||
var seasonNumber: Int? = null,
|
var seasonNumber: Int? = null,
|
||||||
var year: Int? = null
|
var year: Int? = null
|
||||||
|
|
|
@ -3,7 +3,6 @@ package com.lagradost.cloudstream3.syncproviders
|
||||||
import com.lagradost.cloudstream3.AcraApplication.Companion.getKey
|
import com.lagradost.cloudstream3.AcraApplication.Companion.getKey
|
||||||
import com.lagradost.cloudstream3.AcraApplication.Companion.removeKeys
|
import com.lagradost.cloudstream3.AcraApplication.Companion.removeKeys
|
||||||
import com.lagradost.cloudstream3.AcraApplication.Companion.setKey
|
import com.lagradost.cloudstream3.AcraApplication.Companion.setKey
|
||||||
import com.lagradost.cloudstream3.syncproviders.providers.SubScene
|
|
||||||
import com.lagradost.cloudstream3.syncproviders.providers.*
|
import com.lagradost.cloudstream3.syncproviders.providers.*
|
||||||
import java.util.concurrent.TimeUnit
|
import java.util.concurrent.TimeUnit
|
||||||
|
|
||||||
|
@ -13,9 +12,8 @@ abstract class AccountManager(private val defIndex: Int) : AuthAPI {
|
||||||
val aniListApi = AniListApi(0)
|
val aniListApi = AniListApi(0)
|
||||||
val openSubtitlesApi = OpenSubtitlesApi(0)
|
val openSubtitlesApi = OpenSubtitlesApi(0)
|
||||||
val simklApi = SimklApi(0)
|
val simklApi = SimklApi(0)
|
||||||
val indexSubtitlesApi = IndexSubtitleApi()
|
|
||||||
val addic7ed = Addic7ed()
|
val addic7ed = Addic7ed()
|
||||||
val subScene = SubScene()
|
val subDlApi = SubDlApi(0)
|
||||||
val localListApi = LocalList()
|
val localListApi = LocalList()
|
||||||
|
|
||||||
// used to login via app intent
|
// used to login via app intent
|
||||||
|
@ -27,7 +25,7 @@ abstract class AccountManager(private val defIndex: Int) : AuthAPI {
|
||||||
// this needs init with context and can be accessed in settings
|
// this needs init with context and can be accessed in settings
|
||||||
val accountManagers
|
val accountManagers
|
||||||
get() = listOf(
|
get() = listOf(
|
||||||
malApi, aniListApi, openSubtitlesApi, simklApi //nginxApi
|
malApi, aniListApi, openSubtitlesApi, subDlApi, simklApi //nginxApi
|
||||||
)
|
)
|
||||||
|
|
||||||
// used for active syncing
|
// used for active syncing
|
||||||
|
@ -37,14 +35,16 @@ abstract class AccountManager(private val defIndex: Int) : AuthAPI {
|
||||||
)
|
)
|
||||||
|
|
||||||
val inAppAuths
|
val inAppAuths
|
||||||
get() = listOf(openSubtitlesApi)//, nginxApi)
|
get() = listOf<InAppAuthAPIManager>(
|
||||||
|
openSubtitlesApi,
|
||||||
|
subDlApi
|
||||||
|
)//, nginxApi)
|
||||||
|
|
||||||
val subtitleProviders
|
val subtitleProviders
|
||||||
get() = listOf(
|
get() = listOf(
|
||||||
openSubtitlesApi,
|
openSubtitlesApi,
|
||||||
indexSubtitlesApi, // they got anti scraping measures in place :(
|
|
||||||
addic7ed,
|
addic7ed,
|
||||||
subScene
|
subDlApi
|
||||||
)
|
)
|
||||||
|
|
||||||
const val appString = "cloudstreamapp"
|
const val appString = "cloudstreamapp"
|
||||||
|
|
|
@ -1,265 +0,0 @@
|
||||||
package com.lagradost.cloudstream3.syncproviders.providers
|
|
||||||
|
|
||||||
import android.util.Log
|
|
||||||
import com.lagradost.cloudstream3.TvType
|
|
||||||
import com.lagradost.cloudstream3.app
|
|
||||||
import com.lagradost.cloudstream3.imdbUrlToIdNullable
|
|
||||||
import com.lagradost.cloudstream3.subtitles.AbstractSubApi
|
|
||||||
import com.lagradost.cloudstream3.subtitles.AbstractSubtitleEntities
|
|
||||||
import com.lagradost.cloudstream3.utils.SubtitleHelper
|
|
||||||
|
|
||||||
class IndexSubtitleApi : AbstractSubApi {
|
|
||||||
override val name = "IndexSubtitle"
|
|
||||||
override val idPrefix = "indexsubtitle"
|
|
||||||
override val requiresLogin = false
|
|
||||||
override val icon: Nothing? = null
|
|
||||||
override val createAccountUrl: Nothing? = null
|
|
||||||
|
|
||||||
override fun loginInfo(): Nothing? = null
|
|
||||||
|
|
||||||
override fun logOut() {}
|
|
||||||
|
|
||||||
|
|
||||||
companion object {
|
|
||||||
const val host = "https://indexsubtitle.com"
|
|
||||||
const val TAG = "INDEXSUBS"
|
|
||||||
|
|
||||||
fun getOrdinal(num: Int?): String? {
|
|
||||||
return when (num) {
|
|
||||||
1 -> "First"
|
|
||||||
2 -> "Second"
|
|
||||||
3 -> "Third"
|
|
||||||
4 -> "Fourth"
|
|
||||||
5 -> "Fifth"
|
|
||||||
6 -> "Sixth"
|
|
||||||
7 -> "Seventh"
|
|
||||||
8 -> "Eighth"
|
|
||||||
9 -> "Ninth"
|
|
||||||
10 -> "Tenth"
|
|
||||||
11 -> "Eleventh"
|
|
||||||
12 -> "Twelfth"
|
|
||||||
13 -> "Thirteenth"
|
|
||||||
14 -> "Fourteenth"
|
|
||||||
15 -> "Fifteenth"
|
|
||||||
16 -> "Sixteenth"
|
|
||||||
17 -> "Seventeenth"
|
|
||||||
18 -> "Eighteenth"
|
|
||||||
19 -> "Nineteenth"
|
|
||||||
20 -> "Twentieth"
|
|
||||||
21 -> "Twenty-First"
|
|
||||||
22 -> "Twenty-Second"
|
|
||||||
23 -> "Twenty-Third"
|
|
||||||
24 -> "Twenty-Fourth"
|
|
||||||
25 -> "Twenty-Fifth"
|
|
||||||
26 -> "Twenty-Sixth"
|
|
||||||
27 -> "Twenty-Seventh"
|
|
||||||
28 -> "Twenty-Eighth"
|
|
||||||
29 -> "Twenty-Ninth"
|
|
||||||
30 -> "Thirtieth"
|
|
||||||
31 -> "Thirty-First"
|
|
||||||
32 -> "Thirty-Second"
|
|
||||||
33 -> "Thirty-Third"
|
|
||||||
34 -> "Thirty-Fourth"
|
|
||||||
35 -> "Thirty-Fifth"
|
|
||||||
else -> null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun fixUrl(url: String): String {
|
|
||||||
if (url.startsWith("http")) {
|
|
||||||
return url
|
|
||||||
}
|
|
||||||
if (url.isEmpty()) {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
val startsWithNoHttp = url.startsWith("//")
|
|
||||||
if (startsWithNoHttp) {
|
|
||||||
return "https:$url"
|
|
||||||
} else {
|
|
||||||
if (url.startsWith('/')) {
|
|
||||||
return host + url
|
|
||||||
}
|
|
||||||
return "$host/$url"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun isRightEps(text: String, seasonNum: Int?, epNum: Int?): Boolean {
|
|
||||||
val FILTER_EPS_REGEX =
|
|
||||||
Regex("(?i)((Chapter\\s?0?${epNum})|((Season)?\\s?0?${seasonNum}?\\s?(Episode)\\s?0?${epNum}[^0-9]))|(?i)((S?0?${seasonNum}?E0?${epNum}[^0-9])|(0?${seasonNum}[a-z]0?${epNum}[^0-9]))")
|
|
||||||
return text.contains(FILTER_EPS_REGEX)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun haveEps(text: String): Boolean {
|
|
||||||
val HAVE_EPS_REGEX =
|
|
||||||
Regex("(?i)((Chapter\\s?0?\\d)|((Season)?\\s?0?\\d?\\s?(Episode)\\s?0?\\d))|(?i)((S?0?\\d?E0?\\d)|(0?\\d[a-z]0?\\d))")
|
|
||||||
return text.contains(HAVE_EPS_REGEX)
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun search(query: AbstractSubtitleEntities.SubtitleSearch): List<AbstractSubtitleEntities.SubtitleEntity> {
|
|
||||||
val imdbId = query.imdb ?: 0
|
|
||||||
val lang = query.lang
|
|
||||||
val queryLang = SubtitleHelper.fromTwoLettersToLanguage(lang.toString())
|
|
||||||
val queryText = query.query
|
|
||||||
val epNum = query.epNumber ?: 0
|
|
||||||
val seasonNum = query.seasonNumber ?: 0
|
|
||||||
val yearNum = query.year ?: 0
|
|
||||||
|
|
||||||
val urlItems = ArrayList<String>()
|
|
||||||
|
|
||||||
fun cleanResources(
|
|
||||||
results: MutableList<AbstractSubtitleEntities.SubtitleEntity>,
|
|
||||||
name: String,
|
|
||||||
link: String
|
|
||||||
) {
|
|
||||||
results.add(
|
|
||||||
AbstractSubtitleEntities.SubtitleEntity(
|
|
||||||
idPrefix = idPrefix,
|
|
||||||
name = name,
|
|
||||||
lang = queryLang.toString(),
|
|
||||||
data = link,
|
|
||||||
source = this.name,
|
|
||||||
type = if (seasonNum > 0) TvType.TvSeries else TvType.Movie,
|
|
||||||
epNumber = epNum,
|
|
||||||
seasonNumber = seasonNum,
|
|
||||||
year = yearNum,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
val document = app.get("$host/?search=$queryText").document
|
|
||||||
|
|
||||||
document.select("div.my-3.p-3 div.media").map { block ->
|
|
||||||
if (seasonNum > 0) {
|
|
||||||
val name = block.select("strong.text-primary, strong.text-info").text().trim()
|
|
||||||
val season = getOrdinal(seasonNum)
|
|
||||||
if ((block.selectFirst("a")?.attr("href")
|
|
||||||
?.contains(
|
|
||||||
"$season",
|
|
||||||
ignoreCase = true
|
|
||||||
)!! || name.contains(
|
|
||||||
"$season",
|
|
||||||
ignoreCase = true
|
|
||||||
)) && name.contains(queryText, ignoreCase = true)
|
|
||||||
) {
|
|
||||||
block.select("div.media").mapNotNull {
|
|
||||||
urlItems.add(
|
|
||||||
fixUrl(
|
|
||||||
it.selectFirst("a")!!.attr("href")
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (block.selectFirst("strong")!!.text().trim()
|
|
||||||
.matches(Regex("(?i)^$queryText\$"))
|
|
||||||
) {
|
|
||||||
if (block.select("span[title=Release]").isNullOrEmpty()) {
|
|
||||||
block.select("div.media").mapNotNull {
|
|
||||||
val urlItem = fixUrl(
|
|
||||||
it.selectFirst("a")!!.attr("href")
|
|
||||||
)
|
|
||||||
val itemDoc = app.get(urlItem).document
|
|
||||||
val id = imdbUrlToIdNullable(
|
|
||||||
itemDoc.selectFirst("div.d-flex span.badge.badge-primary")?.parent()
|
|
||||||
?.attr("href")
|
|
||||||
)?.toLongOrNull()
|
|
||||||
val year = itemDoc.selectFirst("div.d-flex span.badge.badge-success")
|
|
||||||
?.ownText()
|
|
||||||
?.trim().toString()
|
|
||||||
Log.i(TAG, "id => $id \nyear => $year||$yearNum")
|
|
||||||
if (imdbId > 0) {
|
|
||||||
if (id == imdbId) {
|
|
||||||
urlItems.add(urlItem)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (year.contains("$yearNum")) {
|
|
||||||
urlItems.add(urlItem)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (block.select("span[title=Release]").text().trim()
|
|
||||||
.contains("$yearNum")
|
|
||||||
) {
|
|
||||||
block.select("div.media").mapNotNull {
|
|
||||||
urlItems.add(
|
|
||||||
fixUrl(
|
|
||||||
it.selectFirst("a")!!.attr("href")
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Log.i(TAG, "urlItems => $urlItems")
|
|
||||||
val results = mutableListOf<AbstractSubtitleEntities.SubtitleEntity>()
|
|
||||||
|
|
||||||
urlItems.forEach { url ->
|
|
||||||
val request = app.get(url)
|
|
||||||
if (request.isSuccessful) {
|
|
||||||
request.document.select("div.my-3.p-3 div.media").map { block ->
|
|
||||||
if (block.select("span.d-block span[data-original-title=Language]").text()
|
|
||||||
.trim()
|
|
||||||
.contains("$queryLang")
|
|
||||||
) {
|
|
||||||
var name = block.select("strong.text-primary, strong.text-info").text().trim()
|
|
||||||
val link = fixUrl(block.selectFirst("a")!!.attr("href"))
|
|
||||||
if (seasonNum > 0) {
|
|
||||||
when {
|
|
||||||
isRightEps(name, seasonNum, epNum) -> {
|
|
||||||
cleanResources(results, name, link)
|
|
||||||
}
|
|
||||||
!(haveEps(name)) -> {
|
|
||||||
name = "$name (S${seasonNum}:E${epNum})"
|
|
||||||
cleanResources(results, name, link)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
cleanResources(results, name, link)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return results
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun load(data: AbstractSubtitleEntities.SubtitleEntity): String? {
|
|
||||||
val seasonNum = data.seasonNumber
|
|
||||||
val epNum = data.epNumber
|
|
||||||
|
|
||||||
val req = app.get(data.data)
|
|
||||||
|
|
||||||
if (req.isSuccessful) {
|
|
||||||
val document = req.document
|
|
||||||
val link = if (document.select("div.my-3.p-3 div.media").size == 1) {
|
|
||||||
fixUrl(
|
|
||||||
document.selectFirst("div.my-3.p-3 div.media a")!!.attr("href")
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
document.select("div.my-3.p-3 div.media").firstNotNullOf { block ->
|
|
||||||
val name =
|
|
||||||
block.selectFirst("strong.d-block")?.text()?.trim().toString()
|
|
||||||
if (seasonNum!! > 0) {
|
|
||||||
if (isRightEps(name, seasonNum, epNum)) {
|
|
||||||
fixUrl(block.selectFirst("a")!!.attr("href"))
|
|
||||||
} else {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
fixUrl(block.selectFirst("a")!!.attr("href"))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return link
|
|
||||||
}
|
|
||||||
|
|
||||||
return null
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -185,7 +185,7 @@ class OpenSubtitlesApi(index: Int) : InAppAuthAPIManager(index), AbstractSubApi
|
||||||
throwIfCantDoRequest()
|
throwIfCantDoRequest()
|
||||||
val fixedLang = fixLanguage(query.lang)
|
val fixedLang = fixLanguage(query.lang)
|
||||||
|
|
||||||
val imdbId = query.imdb ?: 0
|
val imdbId = query.imdbId?.replace("tt", "")?.toInt() ?: 0
|
||||||
val queryText = query.query
|
val queryText = query.query
|
||||||
val epNum = query.epNumber ?: 0
|
val epNum = query.epNumber ?: 0
|
||||||
val seasonNum = query.seasonNumber ?: 0
|
val seasonNum = query.seasonNumber ?: 0
|
||||||
|
|
|
@ -1,118 +0,0 @@
|
||||||
package com.lagradost.cloudstream3.syncproviders.providers
|
|
||||||
|
|
||||||
import com.lagradost.cloudstream3.app
|
|
||||||
import com.lagradost.cloudstream3.mvvm.debugPrint
|
|
||||||
import com.lagradost.cloudstream3.subtitles.AbstractSubProvider
|
|
||||||
import com.lagradost.cloudstream3.subtitles.AbstractSubtitleEntities
|
|
||||||
import com.lagradost.cloudstream3.subtitles.SubtitleResource
|
|
||||||
import com.lagradost.cloudstream3.syncproviders.providers.IndexSubtitleApi.Companion.getOrdinal
|
|
||||||
import com.lagradost.cloudstream3.utils.SubtitleHelper
|
|
||||||
|
|
||||||
class SubScene : AbstractSubProvider {
|
|
||||||
val mainUrl = "https://subscene.com"
|
|
||||||
val name = "Subscene"
|
|
||||||
override val idPrefix = "subscene"
|
|
||||||
|
|
||||||
override suspend fun search(query: AbstractSubtitleEntities.SubtitleSearch): List<AbstractSubtitleEntities.SubtitleEntity>? {
|
|
||||||
val seasonName =
|
|
||||||
query.seasonNumber?.let { number ->
|
|
||||||
// Need to translate "7" to "Seventh Season"
|
|
||||||
getOrdinal(number)?.let { words -> " - $words Season" }
|
|
||||||
} ?: ""
|
|
||||||
|
|
||||||
val fullQuery = query.query + seasonName
|
|
||||||
|
|
||||||
val doc = app.post(
|
|
||||||
"$mainUrl/subtitles/searchbytitle",
|
|
||||||
data = mapOf("query" to fullQuery, "l" to "")
|
|
||||||
).document
|
|
||||||
|
|
||||||
return doc.select("div.title a").map { element ->
|
|
||||||
val href = "$mainUrl${element.attr("href")}"
|
|
||||||
val title = element.text()
|
|
||||||
|
|
||||||
AbstractSubtitleEntities.SubtitleEntity(
|
|
||||||
idPrefix = idPrefix,
|
|
||||||
name = title,
|
|
||||||
source = name,
|
|
||||||
data = href,
|
|
||||||
lang = query.lang ?: "en",
|
|
||||||
epNumber = query.epNumber
|
|
||||||
)
|
|
||||||
}.distinctBy { it.data }
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun SubtitleResource.getResources(data: AbstractSubtitleEntities.SubtitleEntity) {
|
|
||||||
val resultDoc = app.get(data.data).document
|
|
||||||
val queryLanguage = SubtitleHelper.fromTwoLettersToLanguage(data.lang) ?: "English"
|
|
||||||
|
|
||||||
val results = resultDoc.select("table tbody tr").mapNotNull { element ->
|
|
||||||
val anchor = element.select("a")
|
|
||||||
val href = anchor.attr("href") ?: return@mapNotNull null
|
|
||||||
val fixedHref = "$mainUrl${href}"
|
|
||||||
val spans = anchor.select("span")
|
|
||||||
val language = spans.firstOrNull()?.text()
|
|
||||||
val title = spans.getOrNull(1)?.text()
|
|
||||||
val isPositive = anchor.select("span.positive-icon").isNotEmpty()
|
|
||||||
|
|
||||||
TableElement(title, language, fixedHref, isPositive)
|
|
||||||
}.sortedBy {
|
|
||||||
it.getScore(queryLanguage, data.epNumber)
|
|
||||||
}
|
|
||||||
|
|
||||||
debugPrint { "$name found subtitles: ${results.takeLast(3)}" }
|
|
||||||
// Last = highest score
|
|
||||||
val selectedResult = results.lastOrNull() ?: return
|
|
||||||
|
|
||||||
val subtitleDocument = app.get(selectedResult.href).document
|
|
||||||
val subtitleDownloadUrl =
|
|
||||||
"$mainUrl${subtitleDocument.select("div.download a").attr("href")}"
|
|
||||||
|
|
||||||
this.addZipUrl(subtitleDownloadUrl) { name, _ ->
|
|
||||||
name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Class to manage the various different subtitle results and rank them.
|
|
||||||
*/
|
|
||||||
data class TableElement(
|
|
||||||
val title: String?,
|
|
||||||
val language: String?,
|
|
||||||
val href: String,
|
|
||||||
val isPositive: Boolean
|
|
||||||
) {
|
|
||||||
private fun matchesLanguage(other: String): Boolean {
|
|
||||||
return language != null && (language.contains(other, ignoreCase = true) ||
|
|
||||||
other.contains(language, ignoreCase = true))
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Scores in this order:
|
|
||||||
* Preferred Language > Episode number > Positive rating > English Language
|
|
||||||
*/
|
|
||||||
fun getScore(queryLanguage: String, episodeNum: Int?): Int {
|
|
||||||
var score = 0
|
|
||||||
if (this.matchesLanguage(queryLanguage)) {
|
|
||||||
score += 8
|
|
||||||
}
|
|
||||||
// Matches Episode 7 using "E07" with any number of leading zeroes
|
|
||||||
if (episodeNum != null && title != null && title.contains(
|
|
||||||
Regex(
|
|
||||||
"""E0*${episodeNum}""",
|
|
||||||
RegexOption.IGNORE_CASE
|
|
||||||
)
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
score += 4
|
|
||||||
}
|
|
||||||
if (isPositive) {
|
|
||||||
score += 2
|
|
||||||
}
|
|
||||||
if (this.matchesLanguage("English")) {
|
|
||||||
score += 1
|
|
||||||
}
|
|
||||||
return score
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,247 @@
|
||||||
|
package com.lagradost.cloudstream3.syncproviders.providers
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty
|
||||||
|
import com.lagradost.cloudstream3.AcraApplication.Companion.getKey
|
||||||
|
import com.lagradost.cloudstream3.AcraApplication.Companion.removeKey
|
||||||
|
import com.lagradost.cloudstream3.AcraApplication.Companion.setKey
|
||||||
|
import com.lagradost.cloudstream3.ErrorLoadingException
|
||||||
|
import com.lagradost.cloudstream3.R
|
||||||
|
import com.lagradost.cloudstream3.TvType
|
||||||
|
import com.lagradost.cloudstream3.app
|
||||||
|
import com.lagradost.cloudstream3.mvvm.logError
|
||||||
|
import com.lagradost.cloudstream3.subtitles.AbstractSubApi
|
||||||
|
import com.lagradost.cloudstream3.subtitles.AbstractSubtitleEntities
|
||||||
|
import com.lagradost.cloudstream3.subtitles.SubtitleResource
|
||||||
|
import com.lagradost.cloudstream3.syncproviders.AuthAPI.LoginInfo
|
||||||
|
import com.lagradost.cloudstream3.syncproviders.InAppAuthAPI
|
||||||
|
import com.lagradost.cloudstream3.syncproviders.InAppAuthAPIManager
|
||||||
|
|
||||||
|
class SubDlApi(index: Int) : InAppAuthAPIManager(index), AbstractSubApi {
|
||||||
|
override val idPrefix = "subdl"
|
||||||
|
override val name = "SubDL"
|
||||||
|
override val icon = R.drawable.subdl_logo_big
|
||||||
|
override val requiresPassword = true
|
||||||
|
override val requiresEmail = true
|
||||||
|
override val createAccountUrl = "https://subdl.com/login"
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
const val APIURL = "https://api.subdl.com"
|
||||||
|
const val APIENDPOINT = "$APIURL/api/v1/subtitles"
|
||||||
|
const val DOWNLOADENDPOINT = "https://dl.subdl.com"
|
||||||
|
const val SUBDL_SUBTITLES_USER_KEY: String = "subdl_user"
|
||||||
|
var currentSession: SubtitleOAuthEntity? = null
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun initialize() {
|
||||||
|
currentSession = getAuthKey()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun logOut() {
|
||||||
|
setAuthKey(null)
|
||||||
|
removeAccountKeys()
|
||||||
|
currentSession = getAuthKey()
|
||||||
|
}
|
||||||
|
override suspend fun login(data: InAppAuthAPI.LoginData): Boolean {
|
||||||
|
val email = data.email ?: throw ErrorLoadingException("Requires Email")
|
||||||
|
val password = data.password ?: throw ErrorLoadingException("Requires Password")
|
||||||
|
switchToNewAccount()
|
||||||
|
try {
|
||||||
|
if (initLogin(email, password)) {
|
||||||
|
registerAccount()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
logError(e)
|
||||||
|
switchToOldAccount()
|
||||||
|
}
|
||||||
|
switchToOldAccount()
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getLatestLoginData(): InAppAuthAPI.LoginData? {
|
||||||
|
val current = getAuthKey() ?: return null
|
||||||
|
return InAppAuthAPI.LoginData(
|
||||||
|
email = current.userEmail,
|
||||||
|
password = current.pass
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun loginInfo(): LoginInfo? {
|
||||||
|
getAuthKey()?.let { user ->
|
||||||
|
return LoginInfo(
|
||||||
|
profilePicture = null,
|
||||||
|
name = user.name ?: user.userEmail,
|
||||||
|
accountIndex = accountIndex
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun search(query: AbstractSubtitleEntities.SubtitleSearch): List<AbstractSubtitleEntities.SubtitleEntity>? {
|
||||||
|
|
||||||
|
val queryText = query.query
|
||||||
|
val epNum = query.epNumber ?: 0
|
||||||
|
val seasonNum = query.seasonNumber ?: 0
|
||||||
|
val yearNum = query.year ?: 0
|
||||||
|
|
||||||
|
val idQuery = when {
|
||||||
|
query.imdbId != null -> "&imdb_id=${query.imdbId}"
|
||||||
|
query.tmdbId != null -> "&tmdb_id=${query.tmdbId}"
|
||||||
|
else -> null
|
||||||
|
}
|
||||||
|
|
||||||
|
val epQuery = if (epNum > 0) "&episode_number=$epNum" else ""
|
||||||
|
val seasonQuery = if (seasonNum > 0) "&season_number=$seasonNum" else ""
|
||||||
|
val yearQuery = if (yearNum > 0) "&year=$yearNum" else ""
|
||||||
|
|
||||||
|
val searchQueryUrl = when (idQuery) {
|
||||||
|
//Use imdb/tmdb id to search if its valid
|
||||||
|
null -> "$APIENDPOINT?api_key=${currentSession?.apiKey}&film_name=$queryText&languages=${query.lang}$epQuery$seasonQuery$yearQuery"
|
||||||
|
else -> "$APIENDPOINT?api_key=${currentSession?.apiKey}$idQuery&languages=${query.lang}$epQuery$seasonQuery$yearQuery"
|
||||||
|
}
|
||||||
|
|
||||||
|
val req = app.get(
|
||||||
|
url = searchQueryUrl,
|
||||||
|
headers = mapOf(
|
||||||
|
"Accept" to "application/json"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return req.parsedSafe<ApiResponse>()?.subtitles?.map { subtitle ->
|
||||||
|
|
||||||
|
val lang = subtitle.lang.replaceFirstChar { it.uppercase() }
|
||||||
|
val resEpNum = subtitle.episode ?: query.epNumber
|
||||||
|
val resSeasonNum = subtitle.season ?: query.seasonNumber
|
||||||
|
val type = if ((resSeasonNum ?: 0) > 0) TvType.TvSeries else TvType.Movie
|
||||||
|
|
||||||
|
AbstractSubtitleEntities.SubtitleEntity(
|
||||||
|
idPrefix = this.idPrefix,
|
||||||
|
name = subtitle.releaseName,
|
||||||
|
lang = lang,
|
||||||
|
data = "${DOWNLOADENDPOINT}${subtitle.url}",
|
||||||
|
type = type,
|
||||||
|
source = this.name,
|
||||||
|
epNumber = resEpNum,
|
||||||
|
seasonNumber = resSeasonNum,
|
||||||
|
isHearingImpaired = subtitle.hearingImpaired ?: false,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun SubtitleResource.getResources(data: AbstractSubtitleEntities.SubtitleEntity) {
|
||||||
|
this.addZipUrl(data.data) { name, _ ->
|
||||||
|
name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private suspend fun initLogin(useremail: String, password: String): Boolean {
|
||||||
|
|
||||||
|
val tokenResponse = app.post(
|
||||||
|
url = "$APIURL/login",
|
||||||
|
data = mapOf(
|
||||||
|
"email" to useremail,
|
||||||
|
"password" to password
|
||||||
|
)
|
||||||
|
).parsedSafe<OAuthTokenResponse>()
|
||||||
|
|
||||||
|
if (tokenResponse?.token == null) return false
|
||||||
|
|
||||||
|
val apiResponse = app.get(
|
||||||
|
url = "$APIURL/user/userApi",
|
||||||
|
headers = mapOf(
|
||||||
|
"Authorization" to "Bearer ${tokenResponse.token}"
|
||||||
|
)
|
||||||
|
).parsedSafe<ApiKeyResponse>()
|
||||||
|
|
||||||
|
if (apiResponse?.ok == false) return false
|
||||||
|
|
||||||
|
setAuthKey(
|
||||||
|
SubtitleOAuthEntity(
|
||||||
|
userEmail = useremail,
|
||||||
|
pass = password,
|
||||||
|
name = tokenResponse.userData?.username ?: tokenResponse.userData?.name,
|
||||||
|
accessToken = tokenResponse.token,
|
||||||
|
apiKey = apiResponse?.apiKey
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getAuthKey(): SubtitleOAuthEntity? {
|
||||||
|
return getKey(accountId, SUBDL_SUBTITLES_USER_KEY)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setAuthKey(data: SubtitleOAuthEntity?) {
|
||||||
|
if (data == null) removeKey(
|
||||||
|
accountId,
|
||||||
|
SUBDL_SUBTITLES_USER_KEY
|
||||||
|
)
|
||||||
|
currentSession = data
|
||||||
|
setKey(accountId, SUBDL_SUBTITLES_USER_KEY, data)
|
||||||
|
}
|
||||||
|
|
||||||
|
data class SubtitleOAuthEntity(
|
||||||
|
@JsonProperty("userEmail") var userEmail: String,
|
||||||
|
@JsonProperty("pass") var pass: String,
|
||||||
|
@JsonProperty("name") var name: String? = null,
|
||||||
|
@JsonProperty("accessToken") var accessToken: String? = null,
|
||||||
|
@JsonProperty("apiKey") var apiKey: String? = null,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class OAuthTokenResponse(
|
||||||
|
@JsonProperty("token") val token: String? = null,
|
||||||
|
@JsonProperty("userData") val userData: UserData? = null,
|
||||||
|
@JsonProperty("status") val status: Boolean? = null,
|
||||||
|
@JsonProperty("message") val message: String? = null,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class UserData(
|
||||||
|
@JsonProperty("email") val email: String,
|
||||||
|
@JsonProperty("name") val name: String,
|
||||||
|
@JsonProperty("country") val country: String,
|
||||||
|
@JsonProperty("scStepCode") val scStepCode: String,
|
||||||
|
@JsonProperty("scVerified") val scVerified: Boolean,
|
||||||
|
@JsonProperty("username") val username: String? = null,
|
||||||
|
@JsonProperty("scUsername") val scUsername: String,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class ApiKeyResponse(
|
||||||
|
@JsonProperty("ok") val ok: Boolean? = false,
|
||||||
|
@JsonProperty("api_key") val apiKey: String? = null,
|
||||||
|
@JsonProperty("usage") val usage: Usage? = null,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class Usage(
|
||||||
|
@JsonProperty("total") val total: Long? = 0,
|
||||||
|
@JsonProperty("today") val today: Long? = 0,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class ApiResponse(
|
||||||
|
@JsonProperty("status") val status: Boolean? = null,
|
||||||
|
@JsonProperty("results") val results: List<Result>? = null,
|
||||||
|
@JsonProperty("subtitles") val subtitles: List<Subtitle>? = null,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class Result(
|
||||||
|
@JsonProperty("sd_id") val sdId: Int? = null,
|
||||||
|
@JsonProperty("type") val type: String? = null,
|
||||||
|
@JsonProperty("name") val name: String? = null,
|
||||||
|
@JsonProperty("imdb_id") val imdbId: String? = null,
|
||||||
|
@JsonProperty("tmdb_id") val tmdbId: Long? = null,
|
||||||
|
@JsonProperty("first_air_date") val firstAirDate: String? = null,
|
||||||
|
@JsonProperty("year") val year: Int? = null,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class Subtitle(
|
||||||
|
@JsonProperty("release_name") val releaseName: String,
|
||||||
|
@JsonProperty("name") val name: String,
|
||||||
|
@JsonProperty("lang") val lang: String,
|
||||||
|
@JsonProperty("author") val author: String? = null,
|
||||||
|
@JsonProperty("url") val url: String? = null,
|
||||||
|
@JsonProperty("subtitlePage") val subtitlePage: String? = null,
|
||||||
|
@JsonProperty("season") val season: Int? = null,
|
||||||
|
@JsonProperty("episode") val episode: Int? = null,
|
||||||
|
@JsonProperty("language") val language: String? = null,
|
||||||
|
@JsonProperty("hi") val hearingImpaired: Boolean? = null,
|
||||||
|
)
|
||||||
|
}
|
|
@ -17,7 +17,7 @@ import com.lagradost.safefile.SafeFile
|
||||||
const val DTAG = "PlayerActivity"
|
const val DTAG = "PlayerActivity"
|
||||||
|
|
||||||
class DownloadedPlayerActivity : AppCompatActivity() {
|
class DownloadedPlayerActivity : AppCompatActivity() {
|
||||||
override fun dispatchKeyEvent(event: KeyEvent?): Boolean {
|
override fun dispatchKeyEvent(event: KeyEvent): Boolean {
|
||||||
CommonActivity.dispatchKeyEvent(this, event)?.let {
|
CommonActivity.dispatchKeyEvent(this, event)?.let {
|
||||||
return it
|
return it
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,6 +32,7 @@ import com.lagradost.cloudstream3.CommonActivity.keyEventListener
|
||||||
import com.lagradost.cloudstream3.CommonActivity.playerEventListener
|
import com.lagradost.cloudstream3.CommonActivity.playerEventListener
|
||||||
import com.lagradost.cloudstream3.CommonActivity.screenHeight
|
import com.lagradost.cloudstream3.CommonActivity.screenHeight
|
||||||
import com.lagradost.cloudstream3.CommonActivity.screenWidth
|
import com.lagradost.cloudstream3.CommonActivity.screenWidth
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse
|
||||||
import com.lagradost.cloudstream3.R
|
import com.lagradost.cloudstream3.R
|
||||||
import com.lagradost.cloudstream3.databinding.PlayerCustomLayoutBinding
|
import com.lagradost.cloudstream3.databinding.PlayerCustomLayoutBinding
|
||||||
import com.lagradost.cloudstream3.databinding.SubtitleOffsetBinding
|
import com.lagradost.cloudstream3.databinding.SubtitleOffsetBinding
|
||||||
|
@ -177,7 +178,7 @@ open class FullScreenPlayer : AbstractPlayerFragment() {
|
||||||
|
|
||||||
open fun openOnlineSubPicker(
|
open fun openOnlineSubPicker(
|
||||||
context: Context,
|
context: Context,
|
||||||
imdbId: Long?,
|
loadResponse: LoadResponse?,
|
||||||
dismissCallback: (() -> Unit)
|
dismissCallback: (() -> Unit)
|
||||||
) {
|
) {
|
||||||
throw NotImplementedError()
|
throw NotImplementedError()
|
||||||
|
|
|
@ -25,6 +25,10 @@ import com.lagradost.cloudstream3.*
|
||||||
import com.lagradost.cloudstream3.APIHolder.getApiFromNameNull
|
import com.lagradost.cloudstream3.APIHolder.getApiFromNameNull
|
||||||
import com.lagradost.cloudstream3.AcraApplication.Companion.setKey
|
import com.lagradost.cloudstream3.AcraApplication.Companion.setKey
|
||||||
import com.lagradost.cloudstream3.CommonActivity.showToast
|
import com.lagradost.cloudstream3.CommonActivity.showToast
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse.Companion.getAniListId
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse.Companion.getImdbId
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse.Companion.getMalId
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse.Companion.getTMDbId
|
||||||
import com.lagradost.cloudstream3.databinding.DialogOnlineSubtitlesBinding
|
import com.lagradost.cloudstream3.databinding.DialogOnlineSubtitlesBinding
|
||||||
import com.lagradost.cloudstream3.databinding.FragmentPlayerBinding
|
import com.lagradost.cloudstream3.databinding.FragmentPlayerBinding
|
||||||
import com.lagradost.cloudstream3.databinding.PlayerSelectSourceAndSubsBinding
|
import com.lagradost.cloudstream3.databinding.PlayerSelectSourceAndSubsBinding
|
||||||
|
@ -39,7 +43,6 @@ import com.lagradost.cloudstream3.ui.player.PlayerSubtitleHelper.Companion.toSub
|
||||||
import com.lagradost.cloudstream3.ui.player.source_priority.QualityDataHelper
|
import com.lagradost.cloudstream3.ui.player.source_priority.QualityDataHelper
|
||||||
import com.lagradost.cloudstream3.ui.player.source_priority.QualityProfileDialog
|
import com.lagradost.cloudstream3.ui.player.source_priority.QualityProfileDialog
|
||||||
import com.lagradost.cloudstream3.ui.result.*
|
import com.lagradost.cloudstream3.ui.result.*
|
||||||
import com.lagradost.cloudstream3.ui.settings.Globals
|
|
||||||
import com.lagradost.cloudstream3.ui.settings.Globals.EMULATOR
|
import com.lagradost.cloudstream3.ui.settings.Globals.EMULATOR
|
||||||
import com.lagradost.cloudstream3.ui.settings.Globals.TV
|
import com.lagradost.cloudstream3.ui.settings.Globals.TV
|
||||||
import com.lagradost.cloudstream3.ui.settings.Globals.isLayout
|
import com.lagradost.cloudstream3.ui.settings.Globals.isLayout
|
||||||
|
@ -258,6 +261,7 @@ class GeneratorPlayer : FullScreenPlayer() {
|
||||||
var episode: Int? = null,
|
var episode: Int? = null,
|
||||||
var season: Int? = null,
|
var season: Int? = null,
|
||||||
var name: String? = null,
|
var name: String? = null,
|
||||||
|
var imdbId: String? = null,
|
||||||
)
|
)
|
||||||
|
|
||||||
private fun getMetaData(): TempMetaData {
|
private fun getMetaData(): TempMetaData {
|
||||||
|
@ -284,7 +288,7 @@ class GeneratorPlayer : FullScreenPlayer() {
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun openOnlineSubPicker(
|
override fun openOnlineSubPicker(
|
||||||
context: Context, imdbId: Long?, dismissCallback: (() -> Unit)
|
context: Context, loadResponse: LoadResponse?, dismissCallback: (() -> Unit)
|
||||||
) {
|
) {
|
||||||
val providers = subsProviders
|
val providers = subsProviders
|
||||||
val isSingleProvider = subsProviders.size == 1
|
val isSingleProvider = subsProviders.size == 1
|
||||||
|
@ -377,6 +381,7 @@ class GeneratorPlayer : FullScreenPlayer() {
|
||||||
}
|
}
|
||||||
|
|
||||||
val currentTempMeta = getMetaData()
|
val currentTempMeta = getMetaData()
|
||||||
|
|
||||||
// bruh idk why it is not correct
|
// bruh idk why it is not correct
|
||||||
val color = ColorStateList.valueOf(context.colorFromAttribute(R.attr.colorAccent))
|
val color = ColorStateList.valueOf(context.colorFromAttribute(R.attr.colorAccent))
|
||||||
binding.searchLoadingBar.progressTintList = color
|
binding.searchLoadingBar.progressTintList = color
|
||||||
|
@ -424,7 +429,10 @@ class GeneratorPlayer : FullScreenPlayer() {
|
||||||
val search =
|
val search =
|
||||||
AbstractSubtitleEntities.SubtitleSearch(
|
AbstractSubtitleEntities.SubtitleSearch(
|
||||||
query = query ?: return@ioSafe,
|
query = query ?: return@ioSafe,
|
||||||
imdb = imdbId,
|
imdbId = loadResponse?.getImdbId(),
|
||||||
|
tmdbId = loadResponse?.getTMDbId()?.toInt(),
|
||||||
|
malId = loadResponse?.getMalId()?.toInt(),
|
||||||
|
aniListId = loadResponse?.getAniListId()?.toInt(),
|
||||||
epNumber = currentTempMeta.episode,
|
epNumber = currentTempMeta.episode,
|
||||||
seasonNumber = currentTempMeta.season,
|
seasonNumber = currentTempMeta.season,
|
||||||
lang = currentLanguageTwoLetters.ifBlank { null },
|
lang = currentLanguageTwoLetters.ifBlank { null },
|
||||||
|
@ -633,6 +641,8 @@ class GeneratorPlayer : FullScreenPlayer() {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (subsProvidersIsActive) {
|
if (subsProvidersIsActive) {
|
||||||
|
val currentLoadResponse = viewModel.getLoadResponse()
|
||||||
|
|
||||||
val loadFromOpenSubsFooter: TextView = layoutInflater.inflate(
|
val loadFromOpenSubsFooter: TextView = layoutInflater.inflate(
|
||||||
R.layout.sort_bottom_footer_add_choice, null
|
R.layout.sort_bottom_footer_add_choice, null
|
||||||
) as TextView
|
) as TextView
|
||||||
|
@ -643,7 +653,7 @@ class GeneratorPlayer : FullScreenPlayer() {
|
||||||
loadFromOpenSubsFooter.setOnClickListener {
|
loadFromOpenSubsFooter.setOnClickListener {
|
||||||
shouldDismiss = false
|
shouldDismiss = false
|
||||||
sourceDialog.dismissSafe(activity)
|
sourceDialog.dismissSafe(activity)
|
||||||
openOnlineSubPicker(it.context, null) {
|
openOnlineSubPicker(it.context, currentLoadResponse) {
|
||||||
dismiss()
|
dismiss()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ import androidx.lifecycle.LiveData
|
||||||
import androidx.lifecycle.MutableLiveData
|
import androidx.lifecycle.MutableLiveData
|
||||||
import androidx.lifecycle.ViewModel
|
import androidx.lifecycle.ViewModel
|
||||||
import androidx.lifecycle.viewModelScope
|
import androidx.lifecycle.viewModelScope
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse
|
||||||
import com.lagradost.cloudstream3.mvvm.Resource
|
import com.lagradost.cloudstream3.mvvm.Resource
|
||||||
import com.lagradost.cloudstream3.mvvm.launchSafe
|
import com.lagradost.cloudstream3.mvvm.launchSafe
|
||||||
import com.lagradost.cloudstream3.mvvm.logError
|
import com.lagradost.cloudstream3.mvvm.logError
|
||||||
|
@ -111,6 +112,9 @@ class PlayerGeneratorViewModel : ViewModel() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
fun getLoadResponse(): LoadResponse? {
|
||||||
|
return normalSafeApiCall { (generator as? RepoLinkGenerator?)?.page }
|
||||||
|
}
|
||||||
|
|
||||||
fun getMeta(): Any? {
|
fun getMeta(): Any? {
|
||||||
return normalSafeApiCall { generator?.getCurrent() }
|
return normalSafeApiCall { generator?.getCurrent() }
|
||||||
|
|
|
@ -12,6 +12,7 @@ import androidx.core.view.isGone
|
||||||
import androidx.core.view.isVisible
|
import androidx.core.view.isVisible
|
||||||
import com.lagradost.cloudstream3.CommonActivity.screenHeight
|
import com.lagradost.cloudstream3.CommonActivity.screenHeight
|
||||||
import com.lagradost.cloudstream3.CommonActivity.screenWidth
|
import com.lagradost.cloudstream3.CommonActivity.screenWidth
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse
|
||||||
import com.lagradost.cloudstream3.R
|
import com.lagradost.cloudstream3.R
|
||||||
import com.lagradost.cloudstream3.ui.player.CSPlayerEvent
|
import com.lagradost.cloudstream3.ui.player.CSPlayerEvent
|
||||||
import com.lagradost.cloudstream3.ui.player.PlayerEventSource
|
import com.lagradost.cloudstream3.ui.player.PlayerEventSource
|
||||||
|
@ -110,7 +111,7 @@ open class ResultTrailerPlayer : ResultFragmentPhone() {
|
||||||
|
|
||||||
override fun openOnlineSubPicker(
|
override fun openOnlineSubPicker(
|
||||||
context: Context,
|
context: Context,
|
||||||
imdbId: Long?,
|
loadResponse: LoadResponse?,
|
||||||
dismissCallback: () -> Unit
|
dismissCallback: () -> Unit
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,8 +27,17 @@ import com.lagradost.cloudstream3.CommonActivity.getCastSession
|
||||||
import com.lagradost.cloudstream3.CommonActivity.showToast
|
import com.lagradost.cloudstream3.CommonActivity.showToast
|
||||||
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
||||||
import com.lagradost.cloudstream3.LoadResponse.Companion.getAniListId
|
import com.lagradost.cloudstream3.LoadResponse.Companion.getAniListId
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse.Companion.getImdbId
|
||||||
import com.lagradost.cloudstream3.LoadResponse.Companion.getMalId
|
import com.lagradost.cloudstream3.LoadResponse.Companion.getMalId
|
||||||
import com.lagradost.cloudstream3.LoadResponse.Companion.isMovie
|
import com.lagradost.cloudstream3.LoadResponse.Companion.isMovie
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.MPV
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.MPV_COMPONENT
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.MPV_PACKAGE
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.VLC
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.VLC_COMPONENT
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.VLC_PACKAGE
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.WEB_VIDEO
|
||||||
|
import com.lagradost.cloudstream3.MainActivity.Companion.WEB_VIDEO_CAST_PACKAGE
|
||||||
import com.lagradost.cloudstream3.metaproviders.SyncRedirector
|
import com.lagradost.cloudstream3.metaproviders.SyncRedirector
|
||||||
import com.lagradost.cloudstream3.mvvm.*
|
import com.lagradost.cloudstream3.mvvm.*
|
||||||
import com.lagradost.cloudstream3.syncproviders.AccountManager
|
import com.lagradost.cloudstream3.syncproviders.AccountManager
|
||||||
|
@ -1354,7 +1363,7 @@ class ResultViewModel2 : ViewModel() {
|
||||||
|
|
||||||
private fun launchActivity(
|
private fun launchActivity(
|
||||||
activity: Activity?,
|
activity: Activity?,
|
||||||
resumeApp: ResultResume,
|
resumeApp: MainActivity.Companion.ResultResume,
|
||||||
id: Int? = null,
|
id: Int? = null,
|
||||||
work: suspend (Intent.(Activity) -> Unit)
|
work: suspend (Intent.(Activity) -> Unit)
|
||||||
): Job? {
|
): Job? {
|
||||||
|
@ -1719,7 +1728,7 @@ class ResultViewModel2 : ViewModel() {
|
||||||
txt(R.string.episode_action_cast_mirror)
|
txt(R.string.episode_action_cast_mirror)
|
||||||
) { (result, index) ->
|
) { (result, index) ->
|
||||||
val host = device?.host ?: return@acquireSingleLink
|
val host = device?.host ?: return@acquireSingleLink
|
||||||
val link = result.links.firstOrNull() ?: return@acquireSingleLink
|
val link = result.links.getOrNull(index) ?: return@acquireSingleLink
|
||||||
|
|
||||||
FcastSession(host).use { session ->
|
FcastSession(host).use { session ->
|
||||||
session.sendMessage(
|
session.sendMessage(
|
||||||
|
@ -2409,7 +2418,7 @@ class ResultViewModel2 : ViewModel() {
|
||||||
null,
|
null,
|
||||||
loadResponse.type,
|
loadResponse.type,
|
||||||
mainId,
|
mainId,
|
||||||
null
|
null,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,6 +27,7 @@ import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.aniList
|
||||||
import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.malApi
|
import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.malApi
|
||||||
import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.openSubtitlesApi
|
import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.openSubtitlesApi
|
||||||
import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.simklApi
|
import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.simklApi
|
||||||
|
import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.subDlApi
|
||||||
import com.lagradost.cloudstream3.syncproviders.AuthAPI
|
import com.lagradost.cloudstream3.syncproviders.AuthAPI
|
||||||
import com.lagradost.cloudstream3.syncproviders.InAppAuthAPI
|
import com.lagradost.cloudstream3.syncproviders.InAppAuthAPI
|
||||||
import com.lagradost.cloudstream3.syncproviders.OAuth2API
|
import com.lagradost.cloudstream3.syncproviders.OAuth2API
|
||||||
|
@ -324,6 +325,7 @@ class SettingsAccount : PreferenceFragmentCompat(), BiometricAuthenticator.Biome
|
||||||
R.string.anilist_key to aniListApi,
|
R.string.anilist_key to aniListApi,
|
||||||
R.string.simkl_key to simklApi,
|
R.string.simkl_key to simklApi,
|
||||||
R.string.opensubtitles_key to openSubtitlesApi,
|
R.string.opensubtitles_key to openSubtitlesApi,
|
||||||
|
R.string.subdl_key to subDlApi,
|
||||||
)
|
)
|
||||||
|
|
||||||
for ((key, api) in syncApis) {
|
for ((key, api) in syncApis) {
|
||||||
|
|
|
@ -9,6 +9,7 @@ import androidx.lifecycle.MutableLiveData
|
||||||
import androidx.lifecycle.ViewModel
|
import androidx.lifecycle.ViewModel
|
||||||
import androidx.lifecycle.viewModelScope
|
import androidx.lifecycle.viewModelScope
|
||||||
import com.lagradost.cloudstream3.CommonActivity.showToast
|
import com.lagradost.cloudstream3.CommonActivity.showToast
|
||||||
|
import com.lagradost.cloudstream3.MainAPI.Companion.settingsForProvider
|
||||||
import com.lagradost.cloudstream3.PROVIDER_STATUS_DOWN
|
import com.lagradost.cloudstream3.PROVIDER_STATUS_DOWN
|
||||||
import com.lagradost.cloudstream3.R
|
import com.lagradost.cloudstream3.R
|
||||||
import com.lagradost.cloudstream3.amap
|
import com.lagradost.cloudstream3.amap
|
||||||
|
@ -181,8 +182,11 @@ class PluginsViewModel : ViewModel() {
|
||||||
}
|
}
|
||||||
|
|
||||||
private suspend fun updatePluginListPrivate(context: Context, repositoryUrl: String) {
|
private suspend fun updatePluginListPrivate(context: Context, repositoryUrl: String) {
|
||||||
|
val isAdult = settingsForProvider.enableAdult
|
||||||
val plugins = getPlugins(repositoryUrl)
|
val plugins = getPlugins(repositoryUrl)
|
||||||
val list = plugins.map { plugin ->
|
val list = plugins.filter {
|
||||||
|
return@filter !(it.second.tvTypes?.contains("NSFW") == true && !isAdult)
|
||||||
|
}.map { plugin ->
|
||||||
PluginViewData(plugin, isDownloaded(context, plugin.second.internalName, plugin.first))
|
PluginViewData(plugin, isDownloaded(context, plugin.second.internalName, plugin.first))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,6 +26,7 @@ import com.lagradost.cloudstream3.syncproviders.providers.MALApi.Companion.MAL_T
|
||||||
import com.lagradost.cloudstream3.syncproviders.providers.MALApi.Companion.MAL_UNIXTIME_KEY
|
import com.lagradost.cloudstream3.syncproviders.providers.MALApi.Companion.MAL_UNIXTIME_KEY
|
||||||
import com.lagradost.cloudstream3.syncproviders.providers.MALApi.Companion.MAL_USER_KEY
|
import com.lagradost.cloudstream3.syncproviders.providers.MALApi.Companion.MAL_USER_KEY
|
||||||
import com.lagradost.cloudstream3.syncproviders.providers.OpenSubtitlesApi.Companion.OPEN_SUBTITLES_USER_KEY
|
import com.lagradost.cloudstream3.syncproviders.providers.OpenSubtitlesApi.Companion.OPEN_SUBTITLES_USER_KEY
|
||||||
|
import com.lagradost.cloudstream3.syncproviders.providers.SubDlApi.Companion.SUBDL_SUBTITLES_USER_KEY
|
||||||
import com.lagradost.cloudstream3.ui.result.txt
|
import com.lagradost.cloudstream3.ui.result.txt
|
||||||
import com.lagradost.cloudstream3.utils.Coroutines.ioSafe
|
import com.lagradost.cloudstream3.utils.Coroutines.ioSafe
|
||||||
import com.lagradost.cloudstream3.utils.Coroutines.main
|
import com.lagradost.cloudstream3.utils.Coroutines.main
|
||||||
|
@ -64,6 +65,7 @@ object BackupUtils {
|
||||||
PLUGINS_KEY_LOCAL,
|
PLUGINS_KEY_LOCAL,
|
||||||
|
|
||||||
OPEN_SUBTITLES_USER_KEY,
|
OPEN_SUBTITLES_USER_KEY,
|
||||||
|
SUBDL_SUBTITLES_USER_KEY,
|
||||||
|
|
||||||
DOWNLOAD_EPISODE_CACHE,
|
DOWNLOAD_EPISODE_CACHE,
|
||||||
|
|
||||||
|
|
|
@ -110,6 +110,7 @@ import com.lagradost.cloudstream3.extractors.Hotlinger
|
||||||
import com.lagradost.cloudstream3.extractors.FourCX
|
import com.lagradost.cloudstream3.extractors.FourCX
|
||||||
import com.lagradost.cloudstream3.extractors.PlayRu
|
import com.lagradost.cloudstream3.extractors.PlayRu
|
||||||
import com.lagradost.cloudstream3.extractors.FourPlayRu
|
import com.lagradost.cloudstream3.extractors.FourPlayRu
|
||||||
|
import com.lagradost.cloudstream3.extractors.FourPichive
|
||||||
import com.lagradost.cloudstream3.extractors.HDMomPlayer
|
import com.lagradost.cloudstream3.extractors.HDMomPlayer
|
||||||
import com.lagradost.cloudstream3.extractors.HDPlayerSystem
|
import com.lagradost.cloudstream3.extractors.HDPlayerSystem
|
||||||
import com.lagradost.cloudstream3.extractors.VideoSeyred
|
import com.lagradost.cloudstream3.extractors.VideoSeyred
|
||||||
|
@ -748,6 +749,7 @@ val extractorApis: MutableList<ExtractorApi> = arrayListOf(
|
||||||
FourCX(),
|
FourCX(),
|
||||||
PlayRu(),
|
PlayRu(),
|
||||||
FourPlayRu(),
|
FourPlayRu(),
|
||||||
|
FourPichive(),
|
||||||
HDMomPlayer(),
|
HDMomPlayer(),
|
||||||
HDPlayerSystem(),
|
HDPlayerSystem(),
|
||||||
VideoSeyred(),
|
VideoSeyred(),
|
||||||
|
|
10
app/src/main/res/drawable/subdl_logo_big.xml
Normal file
10
app/src/main/res/drawable/subdl_logo_big.xml
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:height="20dp"
|
||||||
|
android:viewportHeight="320"
|
||||||
|
android:viewportWidth="320"
|
||||||
|
android:width="20dp">
|
||||||
|
|
||||||
|
<path android:fillColor="@color/white"
|
||||||
|
android:pathData="m107.87,39.3l-8.44,8.59l0,35.68l0,35.83l18.95,22.5c10.36,12.44 30.35,36.27 44.41,53l25.46,30.5l0,12.14l0,12.29l-24.43,-0l-24.43,-0l0,-11.84l0,-11.84l-19.99,-0l-19.99,-0l0,23.24l0,23.24l8.44,8.59l8.44,8.59l48.26,-0l48.26,-0l7.7,-7.85l7.7,-7.85l0,-36.86l-0.15,-37.01l-23.98,-28.28c-13.18,-15.54 -33.16,-39.23 -44.26,-52.55l-20.43,-24.13l0,-12.29l0,-12.29l24.43,-0l24.43,-0l0,12.58l0,12.58l19.99,-0l19.99,-0l0,-24.87l0,-24.87l-7.85,-7.7l-7.85,-7.7l-48.11,-0l-48.11,-0l-8.44,8.59z"/>
|
||||||
|
|
||||||
|
</vector>
|
|
@ -62,14 +62,16 @@
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/account_switch_account"
|
android:id="@+id/account_switch_account"
|
||||||
android:text="@string/switch_account"
|
android:text="@string/switch_account"
|
||||||
style="@style/SettingsItem" />
|
style="@style/SettingsItem"
|
||||||
|
android:focusable="true"/>
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/account_logout"
|
android:id="@+id/account_logout"
|
||||||
android:text="@string/logout"
|
android:text="@string/logout"
|
||||||
style="@style/SettingsItem">
|
style="@style/SettingsItem"
|
||||||
|
android:focusable="true">
|
||||||
|
|
||||||
<requestFocus />
|
<requestFocus />
|
||||||
</TextView>
|
</TextView>
|
||||||
|
|
|
@ -1,10 +1,11 @@
|
||||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:tools="http://schemas.android.com/tools"
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
android:foreground="?android:attr/selectableItemBackgroundBorderless"
|
android:foreground="?android:attr/selectableItemBackgroundBorderless"
|
||||||
android:orientation="horizontal"
|
android:orientation="horizontal"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_width="match_parent">
|
android:layout_width="match_parent"
|
||||||
|
android:focusable="true">
|
||||||
|
|
||||||
<androidx.cardview.widget.CardView
|
<androidx.cardview.widget.CardView
|
||||||
android:id="@+id/account_profile_picture_holder"
|
android:id="@+id/account_profile_picture_holder"
|
||||||
|
@ -15,16 +16,16 @@
|
||||||
android:layout_height="30dp">
|
android:layout_height="30dp">
|
||||||
|
|
||||||
<ImageView
|
<ImageView
|
||||||
android:id="@+id/account_profile_picture"
|
android:id="@+id/account_profile_picture"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
tools:ignore="ContentDescription" />
|
tools:ignore="ContentDescription" />
|
||||||
</androidx.cardview.widget.CardView>
|
</androidx.cardview.widget.CardView>
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:foreground="@null"
|
android:foreground="@null"
|
||||||
android:id="@+id/account_name"
|
android:id="@+id/account_name"
|
||||||
tools:text="Account 1"
|
tools:text="Account 1"
|
||||||
style="@style/SettingsItem" />
|
style="@style/SettingsItem" />
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
|
|
|
@ -7,18 +7,20 @@
|
||||||
android:layout_height="match_parent">
|
android:layout_height="match_parent">
|
||||||
|
|
||||||
<androidx.recyclerview.widget.RecyclerView
|
<androidx.recyclerview.widget.RecyclerView
|
||||||
android:id="@+id/account_list"
|
android:id="@+id/account_list"
|
||||||
app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager"
|
app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager"
|
||||||
android:background="?attr/primaryBlackBackground"
|
android:background="?attr/primaryBlackBackground"
|
||||||
tools:listitem="@layout/account_single"
|
tools:listitem="@layout/account_single"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_rowWeight="1"
|
android:layout_rowWeight="1"
|
||||||
android:layout_height="wrap_content" />
|
android:layout_height="wrap_content"
|
||||||
|
android:focusable="true"/>
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/account_add"
|
android:id="@+id/account_add"
|
||||||
android:text="@string/add_account"
|
android:text="@string/add_account"
|
||||||
style="@style/SettingsItem">
|
style="@style/SettingsItem"
|
||||||
|
android:focusable="true">
|
||||||
|
|
||||||
<requestFocus />
|
<requestFocus />
|
||||||
</TextView>
|
</TextView>
|
||||||
|
|
|
@ -40,7 +40,7 @@
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="30dp"
|
android:layout_height="30dp"
|
||||||
android:layout_gravity="center_vertical"
|
android:layout_gravity="center_vertical"
|
||||||
android:layout_marginEnd="30dp">
|
android:layout_marginEnd="40dp">
|
||||||
|
|
||||||
<androidx.appcompat.widget.SearchView
|
<androidx.appcompat.widget.SearchView
|
||||||
android:id="@+id/subtitles_search"
|
android:id="@+id/subtitles_search"
|
||||||
|
@ -91,7 +91,8 @@
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:layout_gravity="center_vertical|end"
|
android:layout_gravity="center_vertical|end"
|
||||||
android:layout_marginEnd="20dp"
|
android:nextFocusLeft="@id/subtitles_search"
|
||||||
|
android:nextFocusRight="@id/search_filter"
|
||||||
android:text="@string/none"
|
android:text="@string/none"
|
||||||
/>
|
/>
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
@ -106,7 +107,7 @@
|
||||||
android:layout_margin="10dp"
|
android:layout_margin="10dp"
|
||||||
android:background="?selectableItemBackgroundBorderless"
|
android:background="?selectableItemBackgroundBorderless"
|
||||||
android:contentDescription="@string/change_providers_img_des"
|
android:contentDescription="@string/change_providers_img_des"
|
||||||
android:nextFocusLeft="@id/main_search"
|
android:nextFocusLeft="@id/year_btt"
|
||||||
android:nextFocusRight="@id/main_search"
|
android:nextFocusRight="@id/main_search"
|
||||||
android:nextFocusUp="@id/nav_rail_view"
|
android:nextFocusUp="@id/nav_rail_view"
|
||||||
android:nextFocusDown="@id/search_autofit_results"
|
android:nextFocusDown="@id/search_autofit_results"
|
||||||
|
|
|
@ -471,6 +471,7 @@
|
||||||
<string name="simkl_key" translatable="false">simkl_key</string>
|
<string name="simkl_key" translatable="false">simkl_key</string>
|
||||||
<string name="mal_key" translatable="false">mal_key</string>
|
<string name="mal_key" translatable="false">mal_key</string>
|
||||||
<string name="opensubtitles_key" translatable="false">opensubtitles_key</string>
|
<string name="opensubtitles_key" translatable="false">opensubtitles_key</string>
|
||||||
|
<string name="subdl_key" translatable="false">subdl_key</string>
|
||||||
<string name="nginx_key" translatable="false">nginx_key</string>
|
<string name="nginx_key" translatable="false">nginx_key</string>
|
||||||
<string name="example_password">password123</string>
|
<string name="example_password">password123</string>
|
||||||
<string name="example_username">Username</string>
|
<string name="example_username">Username</string>
|
||||||
|
@ -773,4 +774,5 @@
|
||||||
<string name="audio_book_singular">Audio Book</string>
|
<string name="audio_book_singular">Audio Book</string>
|
||||||
<string name="custom_media_singluar">Media</string>
|
<string name="custom_media_singluar">Media</string>
|
||||||
<string name="reset_btn">Reset</string>
|
<string name="reset_btn">Reset</string>
|
||||||
|
<string name="cs3wiki">CloudStream Wiki</string>
|
||||||
</resources>
|
</resources>
|
|
@ -17,6 +17,10 @@
|
||||||
android:icon="@drawable/open_subtitles_icon"
|
android:icon="@drawable/open_subtitles_icon"
|
||||||
android:key="@string/opensubtitles_key" />
|
android:key="@string/opensubtitles_key" />
|
||||||
|
|
||||||
|
<Preference
|
||||||
|
android:icon="@drawable/subdl_logo_big"
|
||||||
|
android:key="@string/subdl_key" />
|
||||||
|
|
||||||
<SwitchPreference
|
<SwitchPreference
|
||||||
android:defaultValue="false"
|
android:defaultValue="false"
|
||||||
android:icon="@drawable/ic_outline_account_circle_24"
|
android:icon="@drawable/ic_outline_account_circle_24"
|
||||||
|
|
|
@ -86,6 +86,14 @@
|
||||||
android:action="android.intent.action.VIEW"
|
android:action="android.intent.action.VIEW"
|
||||||
android:data="https://discord.gg/5Hus6fM" />
|
android:data="https://discord.gg/5Hus6fM" />
|
||||||
</Preference>
|
</Preference>
|
||||||
|
<Preference
|
||||||
|
android:title="@string/cs3wiki"
|
||||||
|
android:icon="@drawable/baseline_description_24"
|
||||||
|
app:summary="https://cloudstream.miraheze.org/">
|
||||||
|
<intent
|
||||||
|
android:action="android.intent.action.VIEW"
|
||||||
|
android:data="https://cloudstream.miraheze.org/" />
|
||||||
|
</Preference>
|
||||||
|
|
||||||
</PreferenceCategory>
|
</PreferenceCategory>
|
||||||
</PreferenceScreen>
|
</PreferenceScreen>
|
|
@ -5,8 +5,13 @@ buildscript {
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
|
<<<<<<< agp
|
||||||
classpath("com.android.tools.build:gradle:8.4.0")
|
classpath("com.android.tools.build:gradle:8.4.0")
|
||||||
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.9.22")
|
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.9.22")
|
||||||
|
=======
|
||||||
|
classpath("com.android.tools.build:gradle:8.2.2")
|
||||||
|
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.9.23")
|
||||||
|
>>>>>>> master
|
||||||
classpath("org.jetbrains.dokka:dokka-gradle-plugin:1.9.10")
|
classpath("org.jetbrains.dokka:dokka-gradle-plugin:1.9.10")
|
||||||
// Universal build config
|
// Universal build config
|
||||||
classpath("com.codingfeline.buildkonfig:buildkonfig-gradle-plugin:0.15.1")
|
classpath("com.codingfeline.buildkonfig:buildkonfig-gradle-plugin:0.15.1")
|
||||||
|
|
|
@ -0,0 +1,11 @@
|
||||||
|
package com.lagradost.cloudstream3.utils
|
||||||
|
|
||||||
|
import android.os.Handler
|
||||||
|
import android.os.Looper
|
||||||
|
|
||||||
|
actual fun runOnMainThreadNative(work: () -> Unit) {
|
||||||
|
val mainHandler = Handler(Looper.getMainLooper())
|
||||||
|
mainHandler.post {
|
||||||
|
work()
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,35 @@
|
||||||
|
package com.lagradost.cloudstream3
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.DeserializationFeature
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
|
||||||
|
import com.lagradost.nicehttp.Requests
|
||||||
|
import com.lagradost.nicehttp.ResponseParser
|
||||||
|
import kotlin.reflect.KClass
|
||||||
|
|
||||||
|
// Short name for requests client to make it nicer to use
|
||||||
|
|
||||||
|
var app = Requests(responseParser = object : ResponseParser {
|
||||||
|
val mapper: ObjectMapper = jacksonObjectMapper().configure(
|
||||||
|
DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
|
||||||
|
override fun <T : Any> parse(text: String, kClass: KClass<T>): T {
|
||||||
|
return mapper.readValue(text, kClass.java)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun <T : Any> parseSafe(text: String, kClass: KClass<T>): T? {
|
||||||
|
return try {
|
||||||
|
mapper.readValue(text, kClass.java)
|
||||||
|
} catch (e: Exception) {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun writeValueAsString(obj: Any): String {
|
||||||
|
return mapper.writeValueAsString(obj)
|
||||||
|
}
|
||||||
|
}).apply {
|
||||||
|
defaultHeaders = mapOf("user-agent" to USER_AGENT)
|
||||||
|
}
|
|
@ -1,3 +1,6 @@
|
||||||
package com.lagradost.cloudstream3
|
package com.lagradost.cloudstream3
|
||||||
|
|
||||||
|
const val USER_AGENT =
|
||||||
|
"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/116.0.0.0 Safari/537.36"
|
||||||
|
|
||||||
class ErrorLoadingException(message: String? = null) : Exception(message)
|
class ErrorLoadingException(message: String? = null) : Exception(message)
|
|
@ -1,12 +1,11 @@
|
||||||
package com.lagradost.cloudstream3.utils
|
package com.lagradost.cloudstream3.utils
|
||||||
|
|
||||||
import android.os.Handler
|
|
||||||
import android.os.Looper
|
|
||||||
import com.lagradost.cloudstream3.mvvm.launchSafe
|
import com.lagradost.cloudstream3.mvvm.launchSafe
|
||||||
import com.lagradost.cloudstream3.mvvm.logError
|
import com.lagradost.cloudstream3.mvvm.logError
|
||||||
import kotlinx.coroutines.*
|
import kotlinx.coroutines.*
|
||||||
import java.util.Collections.synchronizedList
|
import java.util.Collections.synchronizedList
|
||||||
|
|
||||||
|
expect fun runOnMainThreadNative(work: (() -> Unit))
|
||||||
object Coroutines {
|
object Coroutines {
|
||||||
fun <T> T.main(work: suspend ((T) -> Unit)): Job {
|
fun <T> T.main(work: suspend ((T) -> Unit)): Job {
|
||||||
val value = this
|
val value = this
|
||||||
|
@ -50,10 +49,7 @@ object Coroutines {
|
||||||
}
|
}
|
||||||
|
|
||||||
fun runOnMainThread(work: (() -> Unit)) {
|
fun runOnMainThread(work: (() -> Unit)) {
|
||||||
val mainHandler = Handler(Looper.getMainLooper())
|
runOnMainThreadNative(work)
|
||||||
mainHandler.post {
|
|
||||||
work()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
|
@ -0,0 +1,5 @@
|
||||||
|
package com.lagradost.cloudstream3.utils
|
||||||
|
|
||||||
|
actual fun runOnMainThreadNative(work: () -> Unit) {
|
||||||
|
work.invoke()
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue