cloudstream/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultFragment.kt

510 lines
19 KiB
Kotlin
Raw Normal View History

2021-05-16 18:28:00 +00:00
package com.lagradost.cloudstream3.ui.result
2021-05-18 13:43:32 +00:00
import android.annotation.SuppressLint
import android.content.Context
2021-09-03 09:13:34 +00:00
import android.content.Intent
2021-07-17 15:56:26 +00:00
import android.content.Intent.*
2021-06-10 15:15:14 +00:00
import android.net.Uri
2021-05-16 18:28:00 +00:00
import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
2022-08-18 00:54:05 +00:00
import android.widget.ImageView
2021-05-18 13:43:32 +00:00
import androidx.appcompat.app.AlertDialog
import androidx.core.view.isGone
2021-09-19 22:36:32 +00:00
import androidx.core.view.isVisible
2022-01-07 19:27:25 +00:00
import androidx.lifecycle.ViewModelProvider
2021-09-19 22:36:32 +00:00
import androidx.preference.PreferenceManager
import androidx.recyclerview.widget.RecyclerView
import com.google.android.material.chip.Chip
import com.google.android.material.chip.ChipDrawable
2022-08-04 01:19:59 +00:00
import com.lagradost.cloudstream3.APIHolder.getApiDubstatusSettings
2022-08-18 00:54:05 +00:00
import com.lagradost.cloudstream3.APIHolder.getApiFromNameNull
import com.lagradost.cloudstream3.APIHolder.updateHasTrailers
2022-08-04 22:26:33 +00:00
import com.lagradost.cloudstream3.DubStatus
import com.lagradost.cloudstream3.MainActivity.Companion.afterPluginsLoadedEvent
2022-08-04 22:26:33 +00:00
import com.lagradost.cloudstream3.R
import com.lagradost.cloudstream3.SearchResponse
import com.lagradost.cloudstream3.TvType
2022-08-03 00:04:03 +00:00
import com.lagradost.cloudstream3.mvvm.*
import com.lagradost.cloudstream3.syncproviders.providers.Kitsu
2022-08-03 17:57:38 +00:00
import com.lagradost.cloudstream3.ui.download.DOWNLOAD_ACTION_DOWNLOAD
2021-07-24 20:50:57 +00:00
import com.lagradost.cloudstream3.ui.download.DownloadButtonSetup.handleDownloadClick
2023-07-18 20:18:14 +00:00
import com.lagradost.cloudstream3.ui.player.FullScreenPlayer
import com.lagradost.cloudstream3.ui.result.EpisodeAdapter.Companion.getPlayerAction
2022-03-29 21:50:07 +00:00
import com.lagradost.cloudstream3.ui.settings.SettingsFragment.Companion.isTrueTvSettings
2021-12-10 19:48:21 +00:00
import com.lagradost.cloudstream3.ui.settings.SettingsFragment.Companion.isTvSettings
2022-08-03 17:27:49 +00:00
import com.lagradost.cloudstream3.utils.*
import com.lagradost.cloudstream3.utils.AppUtils.getNameFull
2022-06-29 01:20:23 +00:00
import com.lagradost.cloudstream3.utils.AppUtils.html
2022-01-07 19:27:25 +00:00
import com.lagradost.cloudstream3.utils.AppUtils.loadCache
import com.lagradost.cloudstream3.utils.AppUtils.openBrowser
2022-08-03 17:27:49 +00:00
import com.lagradost.cloudstream3.utils.Coroutines.main
import com.lagradost.cloudstream3.utils.DataStoreHelper.getVideoWatchState
2021-06-15 16:07:20 +00:00
import com.lagradost.cloudstream3.utils.DataStoreHelper.getViewPos
import com.lagradost.cloudstream3.utils.UIHelper.colorFromAttribute
2022-08-03 00:04:03 +00:00
import com.lagradost.cloudstream3.utils.UIHelper.dismissSafe
import com.lagradost.cloudstream3.utils.UIHelper.hideKeyboard
2023-07-15 21:43:09 +00:00
import kotlinx.android.synthetic.main.fragment_result.download_button
2022-08-04 22:26:33 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_cast_items
import kotlinx.android.synthetic.main.fragment_result.result_cast_text
import kotlinx.android.synthetic.main.fragment_result.result_coming_soon
import kotlinx.android.synthetic.main.fragment_result.result_data_holder
import kotlinx.android.synthetic.main.fragment_result.result_description
2023-07-15 21:43:09 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_dub_select
2022-08-04 22:26:33 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_episode_loading
2023-07-15 21:43:09 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_episode_select
2022-08-04 22:26:33 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_episodes
import kotlinx.android.synthetic.main.fragment_result.result_error_text
import kotlinx.android.synthetic.main.fragment_result.result_finish_loading
import kotlinx.android.synthetic.main.fragment_result.result_info
import kotlinx.android.synthetic.main.fragment_result.result_loading
import kotlinx.android.synthetic.main.fragment_result.result_loading_error
import kotlinx.android.synthetic.main.fragment_result.result_meta_duration
import kotlinx.android.synthetic.main.fragment_result.result_meta_rating
import kotlinx.android.synthetic.main.fragment_result.result_meta_site
import kotlinx.android.synthetic.main.fragment_result.result_meta_type
import kotlinx.android.synthetic.main.fragment_result.result_meta_year
import kotlinx.android.synthetic.main.fragment_result.result_next_airing
import kotlinx.android.synthetic.main.fragment_result.result_next_airing_time
import kotlinx.android.synthetic.main.fragment_result.result_no_episodes
import kotlinx.android.synthetic.main.fragment_result.result_play_movie
2022-12-23 21:53:51 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_poster
2023-07-15 21:43:09 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_poster_background
2022-12-23 21:53:51 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_poster_holder
2022-08-04 22:26:33 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_reload_connection_open_in_browser
import kotlinx.android.synthetic.main.fragment_result.result_reload_connectionerror
import kotlinx.android.synthetic.main.fragment_result.result_resume_parent
import kotlinx.android.synthetic.main.fragment_result.result_resume_progress_holder
2023-07-15 21:43:09 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_resume_series_button
2022-08-04 22:26:33 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_resume_series_progress
import kotlinx.android.synthetic.main.fragment_result.result_resume_series_progress_text
import kotlinx.android.synthetic.main.fragment_result.result_resume_series_title
2023-07-15 21:43:09 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_season_button
2022-08-04 22:26:33 +00:00
import kotlinx.android.synthetic.main.fragment_result.result_tag
import kotlinx.android.synthetic.main.fragment_result.result_tag_holder
import kotlinx.android.synthetic.main.fragment_result.result_title
import kotlinx.android.synthetic.main.fragment_result.result_vpn
2023-07-18 20:18:14 +00:00
import kotlinx.android.synthetic.main.fragment_result_tv.result_resume_series_button_play
import kotlinx.android.synthetic.main.fragment_result_tv.temporary_no_focus
import kotlinx.coroutines.delay
2022-08-03 00:04:03 +00:00
import kotlinx.coroutines.runBlocking
2022-08-06 18:36:45 +00:00
2021-07-29 15:16:08 +00:00
const val START_ACTION_RESUME_LATEST = 1
2021-08-25 15:28:25 +00:00
const val START_ACTION_LOAD_EP = 2
/**
* Future proofed way to mark episodes as watched
**/
enum class VideoWatchState {
/** Default value when no key is set */
None,
Watched
}
2021-05-18 13:43:32 +00:00
data class ResultEpisode(
2022-01-07 19:27:25 +00:00
val headerName: String,
2021-05-18 13:43:32 +00:00
val name: String?,
2021-06-10 15:15:14 +00:00
val poster: String?,
2021-05-18 13:43:32 +00:00
val episode: Int,
2022-08-01 01:00:48 +00:00
val seasonIndex: Int?, // this is the "season" index used season names
val season: Int?, // this is the display
2021-06-14 16:58:43 +00:00
val data: String,
2021-05-18 13:43:32 +00:00
val apiName: String,
val id: Int,
2021-05-22 22:25:56 +00:00
val index: Int,
2021-06-15 16:07:20 +00:00
val position: Long, // time in MS
2021-06-10 15:15:14 +00:00
val duration: Long, // duration in MS
2021-06-26 19:32:50 +00:00
val rating: Int?,
val description: String?,
2021-09-19 22:36:32 +00:00
val isFiller: Boolean?,
2022-01-07 19:27:25 +00:00
val tvType: TvType,
val parentId: Int,
/**
* Conveys if the episode itself is marked as watched
**/
val videoWatchState: VideoWatchState
2021-05-18 13:43:32 +00:00
)
2021-05-16 18:28:00 +00:00
2021-06-15 16:07:20 +00:00
fun ResultEpisode.getRealPosition(): Long {
if (duration <= 0) return 0
val percentage = position * 100 / duration
if (percentage <= 5 || percentage >= 95) return 0
return position
}
2021-06-16 17:40:02 +00:00
fun ResultEpisode.getDisplayPosition(): Long {
if (duration <= 0) return 0
val percentage = position * 100 / duration
if (percentage <= 1) return 0
if (percentage <= 5) return 5 * duration / 100
if (percentage >= 95) return duration
return position
}
fun buildResultEpisode(
2022-01-07 19:27:25 +00:00
headerName: String,
name: String? = null,
poster: String? = null,
2021-06-15 16:07:20 +00:00
episode: Int,
2022-08-01 01:00:48 +00:00
seasonIndex: Int? = null,
season: Int? = null,
2021-06-15 16:07:20 +00:00
data: String,
apiName: String,
id: Int,
index: Int,
rating: Int? = null,
description: String? = null,
isFiller: Boolean? = null,
2022-01-07 19:27:25 +00:00
tvType: TvType,
parentId: Int,
2021-06-15 16:07:20 +00:00
): ResultEpisode {
val posDur = getViewPos(id)
val videoWatchState = getVideoWatchState(id) ?: VideoWatchState.None
2021-06-26 19:32:50 +00:00
return ResultEpisode(
2022-01-07 19:27:25 +00:00
headerName,
2021-06-26 19:32:50 +00:00
name,
2021-06-15 16:07:20 +00:00
poster,
episode,
2022-08-01 01:00:48 +00:00
seasonIndex,
2021-06-15 16:07:20 +00:00
season,
data,
apiName,
id,
index,
posDur?.position ?: 0,
2021-06-26 19:32:50 +00:00
posDur?.duration ?: 0,
rating,
2021-11-30 17:59:52 +00:00
description,
2022-01-07 19:27:25 +00:00
isFiller,
tvType,
parentId,
videoWatchState
2021-06-26 19:32:50 +00:00
)
2021-06-15 16:07:20 +00:00
}
2021-07-29 15:16:08 +00:00
/** 0f-1f */
2021-06-10 15:15:14 +00:00
fun ResultEpisode.getWatchProgress(): Float {
2021-07-29 15:16:08 +00:00
return (getDisplayPosition() / 1000).toFloat() / (duration / 1000).toFloat()
2021-06-10 15:15:14 +00:00
}
2023-07-18 20:18:14 +00:00
open class ResultFragment : FullScreenPlayer() {
2021-05-22 22:25:56 +00:00
companion object {
2022-05-02 21:32:28 +00:00
const val URL_BUNDLE = "url"
const val API_NAME_BUNDLE = "apiName"
const val SEASON_BUNDLE = "season"
const val EPISODE_BUNDLE = "episode"
const val START_ACTION_BUNDLE = "startAction"
const val START_VALUE_BUNDLE = "startValue"
const val RESTART_BUNDLE = "restart"
2022-08-04 22:26:33 +00:00
2022-05-02 21:32:28 +00:00
fun newInstance(
card: SearchResponse, startAction: Int = 0, startValue: Int? = null
): Bundle {
return Bundle().apply {
putString(URL_BUNDLE, card.url)
putString(API_NAME_BUNDLE, card.apiName)
if (card is DataStoreHelper.ResumeWatchingResult) {
if (card.season != null)
putInt(SEASON_BUNDLE, card.season)
if (card.episode != null)
putInt(EPISODE_BUNDLE, card.episode)
}
putInt(START_ACTION_BUNDLE, startAction)
if (startValue != null)
putInt(START_VALUE_BUNDLE, startValue)
2022-08-04 01:19:59 +00:00
2022-05-02 21:32:28 +00:00
putBoolean(RESTART_BUNDLE, true)
}
}
2021-12-17 12:42:25 +00:00
fun newInstance(
url: String,
apiName: String,
startAction: Int = 0,
startValue: Int = 0
): Bundle {
2021-09-20 21:11:36 +00:00
return Bundle().apply {
2022-05-02 21:32:28 +00:00
putString(URL_BUNDLE, url)
putString(API_NAME_BUNDLE, apiName)
putInt(START_ACTION_BUNDLE, startAction)
putInt(START_VALUE_BUNDLE, startValue)
putBoolean(RESTART_BUNDLE, true)
2021-05-18 13:43:32 +00:00
}
2021-09-20 21:11:36 +00:00
}
2022-01-18 14:10:01 +00:00
fun updateUI() {
updateUIListener?.invoke()
}
2022-01-31 20:47:59 +00:00
private var updateUIListener: (() -> Unit)? = null
2021-05-22 22:25:56 +00:00
}
2022-08-06 16:08:20 +00:00
open fun setTrailers(trailers: List<ExtractorLink>?) {}
2022-08-04 22:26:33 +00:00
protected lateinit var viewModel: ResultViewModel2 //by activityViewModels()
protected lateinit var syncModel: SyncViewModel
2023-07-18 20:18:14 +00:00
//protected open val resultLayout = R.layout.fragment_result_swipe
override var layout = R.layout.fragment_result_swipe
2021-05-16 18:28:00 +00:00
override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
2021-05-18 13:43:32 +00:00
savedInstanceState: Bundle?,
2021-05-16 18:28:00 +00:00
): View? {
2022-01-07 19:27:25 +00:00
viewModel =
2022-08-01 01:00:48 +00:00
ViewModelProvider(this)[ResultViewModel2::class.java]
2022-04-01 20:05:34 +00:00
syncModel =
ViewModelProvider(this)[SyncViewModel::class.java]
2023-07-18 20:18:14 +00:00
return super.onCreateView(inflater, container, savedInstanceState)
//return inflater.inflate(resultLayout, container, false)
2021-05-16 18:28:00 +00:00
}
2021-07-28 19:14:45 +00:00
override fun onDestroyView() {
updateUIListener = null
2021-07-28 19:14:45 +00:00
super.onDestroyView()
}
2021-06-16 00:15:07 +00:00
override fun onResume() {
afterPluginsLoadedEvent += ::reloadViewModel
2021-06-16 00:15:07 +00:00
super.onResume()
activity?.let {
it.window?.navigationBarColor =
2021-09-19 20:33:39 +00:00
it.colorFromAttribute(R.attr.primaryBlackBackground)
2021-06-16 00:15:07 +00:00
}
2021-05-22 22:25:56 +00:00
}
override fun onDestroy() {
afterPluginsLoadedEvent -= ::reloadViewModel
super.onDestroy()
}
2021-07-29 15:16:08 +00:00
2022-01-18 14:10:01 +00:00
private fun updateUI() {
2022-04-08 19:38:19 +00:00
syncModel.updateUserData()
2022-01-18 14:10:01 +00:00
viewModel.reloadEpisodes()
}
data class StoredData(
val url: String?,
val apiName: String,
val showFillers: Boolean,
val dubStatus: DubStatus,
val start: AutoResume?,
val playerAction: Int
)
2023-07-18 20:18:14 +00:00
fun getStoredData(context: Context): StoredData? {
val settingsManager = PreferenceManager.getDefaultSharedPreferences(context)
val url = arguments?.getString(URL_BUNDLE)
val apiName = arguments?.getString(API_NAME_BUNDLE) ?: return null
val showFillers =
settingsManager.getBoolean(context.getString(R.string.show_fillers_key), false)
val dubStatus = if (context.getApiDubstatusSettings()
.contains(DubStatus.Dubbed)
) DubStatus.Dubbed else DubStatus.Subbed
val startAction = arguments?.getInt(START_ACTION_BUNDLE)
val playerAction = getPlayerAction(context)
val start = startAction?.let { action ->
val startValue = arguments?.getInt(START_VALUE_BUNDLE)
val resumeEpisode = arguments?.getInt(EPISODE_BUNDLE)
val resumeSeason = arguments?.getInt(SEASON_BUNDLE)
arguments?.remove(START_VALUE_BUNDLE)
arguments?.remove(START_ACTION_BUNDLE)
AutoResume(
startAction = action,
id = startValue,
episode = resumeEpisode,
season = resumeSeason
)
}
return StoredData(url, apiName, showFillers, dubStatus, start, playerAction)
}
private fun reloadViewModel(forceReload: Boolean) {
if (!viewModel.hasLoaded() || forceReload) {
val storedData = getStoredData(activity ?: context ?: return) ?: return
viewModel.load(
activity,
storedData.url ?: return,
storedData.apiName,
storedData.showFillers,
storedData.dubStatus,
storedData.start
)
}
}
2021-05-18 13:43:32 +00:00
@SuppressLint("SetTextI18n")
2021-05-16 18:28:00 +00:00
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
result_cast_items?.layoutManager = object : LinearListLayout(view.context) {
override fun onRequestChildFocus(
parent: RecyclerView,
state: RecyclerView.State,
child: View,
focused: View?
): Boolean {
// Make the cast always focus the first visible item when focused
// from somewhere else. Otherwise it jumps to the last item.
return if (parent.focusedChild == null) {
scrollToPosition(this.findFirstCompletelyVisibleItemPosition())
true
} else {
super.onRequestChildFocus(parent, state, child, focused)
}
}
}.apply {
this.orientation = RecyclerView.HORIZONTAL
}
2022-07-24 02:49:15 +00:00
result_cast_items?.adapter = ActorAdaptor()
2022-06-16 23:58:55 +00:00
2022-01-18 14:10:01 +00:00
updateUIListener = ::updateUI
2022-05-02 21:32:28 +00:00
val restart = arguments?.getBoolean(RESTART_BUNDLE) ?: false
2021-09-20 21:11:36 +00:00
if (restart) {
2022-05-02 21:32:28 +00:00
arguments?.putBoolean(RESTART_BUNDLE, false)
2021-09-20 21:11:36 +00:00
}
activity?.window?.decorView?.clearFocus()
hideKeyboard()
context?.updateHasTrailers()
2022-01-07 19:27:25 +00:00
activity?.loadCache()
2021-09-19 20:33:39 +00:00
//activity?.fixPaddingStatusbar(result_barstatus)
2021-06-06 18:06:01 +00:00
/* val backParameter = result_back.layoutParams as FrameLayout.LayoutParams
backParameter.setMargins(
backParameter.leftMargin,
backParameter.topMargin + requireContext().getStatusBarHeight(),
backParameter.rightMargin,
backParameter.bottomMargin
)
result_back.layoutParams = backParameter*/
2021-06-16 16:54:07 +00:00
2021-05-20 15:22:28 +00:00
// activity?.fixPaddingStatusbar(result_toolbar)
2021-05-18 13:43:32 +00:00
val storedData = (activity ?: context)?.let {
getStoredData(it)
2022-08-04 01:19:59 +00:00
}
2021-06-15 23:25:58 +00:00
// This is to band-aid FireTV navigation
2022-08-28 23:52:15 +00:00
val isTv = isTvSettings()
2022-08-05 23:41:35 +00:00
result_season_button?.isFocusableInTouchMode = isTv
result_episode_select?.isFocusableInTouchMode = isTv
result_dub_select?.isFocusableInTouchMode = isTv
2022-04-01 20:05:34 +00:00
observeNullable(viewModel.resumeWatching) { resume ->
if (resume == null) {
result_resume_parent?.isVisible = false
return@observeNullable
}
result_resume_parent?.isVisible = true
resume.progress?.let { progress ->
result_resume_series_title?.apply {
isVisible = !resume.isMovie
text =
if (resume.isMovie) null else activity?.getNameFull(
resume.result.name,
resume.result.episode,
resume.result.season
2022-05-14 17:07:34 +00:00
)
2021-08-25 15:28:25 +00:00
}
result_resume_series_progress_text?.setText(progress.progressLeft)
result_resume_series_progress?.apply {
isVisible = true
this.max = progress.maxProgress
this.progress = progress.progress
2022-08-03 17:27:49 +00:00
}
result_resume_progress_holder?.isVisible = true
} ?: run {
result_resume_progress_holder?.isVisible = false
result_resume_series_progress?.isVisible = false
result_resume_series_title?.isVisible = false
result_resume_series_progress_text?.isVisible = false
2021-07-29 15:16:08 +00:00
}
result_resume_series_button?.isVisible = !resume.isMovie
result_resume_series_button_play?.isVisible = !resume.isMovie
val click = View.OnClickListener {
viewModel.handleAction(
activity,
EpisodeClickEvent(
storedData?.playerAction ?: ACTION_PLAY_EPISODE_IN_PLAYER,
resume.result
)
)
}
result_resume_series_button?.setOnClickListener(click)
result_resume_series_button_play?.setOnClickListener(click)
2021-07-29 15:16:08 +00:00
}
2022-08-03 17:27:49 +00:00
2021-09-19 22:36:32 +00:00
context?.let { ctx ->
2022-08-04 01:19:59 +00:00
2022-08-20 01:06:35 +00:00
//result_bookmark_button?.isVisible = ctx.isTvSettings()
2021-12-10 19:48:21 +00:00
2021-09-19 22:36:32 +00:00
val settingsManager = PreferenceManager.getDefaultSharedPreferences(ctx)
2021-06-16 16:54:07 +00:00
Kitsu.isEnabled =
settingsManager.getBoolean(ctx.getString(R.string.show_kitsu_posters_key), true)
if (storedData?.url != null) {
2021-09-19 22:36:32 +00:00
result_reload_connectionerror.setOnClickListener {
viewModel.load(
activity,
storedData.url,
storedData.apiName,
storedData.showFillers,
storedData.dubStatus,
storedData.start
)
2021-08-29 17:15:09 +00:00
}
2021-06-16 16:54:07 +00:00
result_reload_connection_open_in_browser?.setOnClickListener {
val i = Intent(ACTION_VIEW)
i.data = Uri.parse(storedData.url)
2022-03-14 12:44:54 +00:00
try {
startActivity(i)
} catch (e: Exception) {
2022-01-30 22:02:57 +00:00
logError(e)
}
}
2022-04-03 21:41:28 +00:00
// bloats the navigation on tv
2022-08-28 23:52:15 +00:00
if (!isTrueTvSettings()) {
2022-04-03 21:41:28 +00:00
result_meta_site?.setOnClickListener {
it.context?.openBrowser(storedData.url)
2022-04-03 21:41:28 +00:00
}
result_meta_site?.isFocusable = true
} else {
result_meta_site?.isFocusable = false
2021-09-19 22:36:32 +00:00
}
2022-08-01 02:46:43 +00:00
if (restart || !viewModel.hasLoaded()) {
2022-01-07 19:27:25 +00:00
//viewModel.clear()
viewModel.load(
activity,
storedData.url,
storedData.apiName,
storedData.showFillers,
storedData.dubStatus,
storedData.start
)
2021-09-12 15:57:07 +00:00
}
}
2021-06-16 16:54:07 +00:00
}
2022-02-04 20:49:35 +00:00
}
}