Skip to content

Commit ce7b31d

Browse files
committed
Add still watching feature
1 parent 98a96ca commit ce7b31d

23 files changed

+709
-183
lines changed

app/src/main/java/org/jellyfin/androidtv/di/AppModule.kt

+5-2
Original file line numberDiff line numberDiff line change
@@ -26,13 +26,14 @@ import org.jellyfin.androidtv.data.repository.UserViewsRepository
2626
import org.jellyfin.androidtv.data.repository.UserViewsRepositoryImpl
2727
import org.jellyfin.androidtv.data.service.BackgroundService
2828
import org.jellyfin.androidtv.integration.dream.DreamViewModel
29-
import org.jellyfin.androidtv.ui.ScreensaverViewModel
29+
import org.jellyfin.androidtv.ui.InteractionTrackerViewModel
3030
import org.jellyfin.androidtv.ui.itemhandling.ItemLauncher
3131
import org.jellyfin.androidtv.ui.navigation.Destinations
3232
import org.jellyfin.androidtv.ui.navigation.NavigationRepository
3333
import org.jellyfin.androidtv.ui.navigation.NavigationRepositoryImpl
3434
import org.jellyfin.androidtv.ui.picture.PictureViewerViewModel
3535
import org.jellyfin.androidtv.ui.playback.PlaybackControllerContainer
36+
import org.jellyfin.androidtv.ui.playback.stillwatching.StillWatchingViewModel
3637
import org.jellyfin.androidtv.ui.playback.nextup.NextUpViewModel
3738
import org.jellyfin.androidtv.ui.playback.segment.MediaSegmentRepository
3839
import org.jellyfin.androidtv.ui.playback.segment.MediaSegmentRepositoryImpl
@@ -106,6 +107,8 @@ val appModule = module {
106107
// Non API related
107108
single { DataRefreshService() }
108109
single { PlaybackControllerContainer() }
110+
single { InteractionTrackerViewModel() }
111+
109112

110113
single<UserRepository> { UserRepositoryImpl() }
111114
single<UserViewsRepository> { UserViewsRepositoryImpl(get()) }
@@ -120,8 +123,8 @@ val appModule = module {
120123
viewModel { UserLoginViewModel(get(), get(), get(), get(defaultDeviceInfo)) }
121124
viewModel { ServerAddViewModel(get()) }
122125
viewModel { NextUpViewModel(get(), get(), get(), get()) }
126+
viewModel { StillWatchingViewModel(get(), get(), get()) }
123127
viewModel { PictureViewerViewModel(get()) }
124-
viewModel { ScreensaverViewModel(get()) }
125128
viewModel { SearchViewModel(get()) }
126129
viewModel { DreamViewModel(get(), get(), get(), get(), get()) }
127130

app/src/main/java/org/jellyfin/androidtv/preference/UserPreferences.kt

+6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import org.jellyfin.androidtv.preference.constant.ClockBehavior
99
import org.jellyfin.androidtv.preference.constant.NextUpBehavior
1010
import org.jellyfin.androidtv.preference.constant.RatingType
1111
import org.jellyfin.androidtv.preference.constant.RefreshRateSwitchingBehavior
12+
import org.jellyfin.androidtv.preference.constant.StillWatchingBehavior
1213
import org.jellyfin.androidtv.preference.constant.WatchedIndicatorBehavior
1314
import org.jellyfin.androidtv.preference.constant.ZoomMode
1415
import org.jellyfin.androidtv.ui.playback.segment.MediaSegmentAction
@@ -86,6 +87,11 @@ class UserPreferences(context: Context) : SharedPreferenceStore(
8687
*/
8788
var cinemaModeEnabled = booleanPreference("pref_enable_cinema_mode", true)
8889

90+
/**
91+
* Enable still watching
92+
*/
93+
var stillWatchingBehavior = enumPreference("pref_enable_still_watching", StillWatchingBehavior.DEFAULT)
94+
8995
/* Playback - Video */
9096
/**
9197
* Whether to use an external playback application or not.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
package org.jellyfin.androidtv.preference.constant
2+
3+
import org.jellyfin.androidtv.R
4+
import org.jellyfin.preference.PreferenceEnum
5+
6+
enum class StillWatchingBehavior(
7+
override val nameRes: Int
8+
) : PreferenceEnum {
9+
TEST_EPISODE_COUNT(R.string.lbl_still_watching_test_episode_count),
10+
TEST_MIN_MINUTES(R.string.lbl_still_watching_test_min_minutes),
11+
/**
12+
* Takes shorter than Netflix implementation to show still watching screen.
13+
*/
14+
SHORT(R.string.lbl_still_watching_short),
15+
/**
16+
* Default behavior for still watching. Matches Netflix implementation
17+
*/
18+
DEFAULT(R.string.lbl_still_watching_default),
19+
/**
20+
* Takes longer than Netflix implementation to show still watching screen.
21+
*/
22+
LONG(R.string.lbl_still_watching_long),
23+
/**
24+
* Takes longer than Netflix implementation to show still watching screen.
25+
*/
26+
VERY_LONG(R.string.lbl_still_watching_very_long),
27+
/**
28+
* Disables still watching screen.
29+
*/
30+
DISABLED(R.string.state_disabled)
31+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,193 @@
1+
package org.jellyfin.androidtv.ui
2+
3+
import androidx.lifecycle.Lifecycle
4+
import androidx.lifecycle.LifecycleEventObserver
5+
import androidx.lifecycle.LifecycleOwner
6+
import androidx.lifecycle.ViewModel
7+
import androidx.lifecycle.viewModelScope
8+
import kotlinx.coroutines.Job
9+
import kotlinx.coroutines.delay
10+
import kotlinx.coroutines.flow.MutableStateFlow
11+
import kotlinx.coroutines.flow.asStateFlow
12+
import kotlinx.coroutines.launch
13+
import org.jellyfin.androidtv.preference.UserPreferences
14+
import org.jellyfin.androidtv.ui.playback.PlaybackController
15+
import org.jellyfin.androidtv.ui.playback.PlaybackControllerContainer
16+
import org.jellyfin.androidtv.ui.playback.stillwatching.StillWatchingStates
17+
import org.jellyfin.androidtv.util.TimeUtils.MILLIS_PER_MIN
18+
import org.jellyfin.sdk.model.api.BaseItemDto
19+
import org.jellyfin.sdk.model.api.BaseItemKind
20+
import org.koin.core.component.KoinComponent
21+
import org.koin.core.component.inject
22+
import kotlin.time.Duration.Companion.milliseconds
23+
24+
class InteractionTrackerViewModel : ViewModel(), KoinComponent {
25+
// Screensaver vars
26+
private var timer: Job? = null
27+
private var locks = 0
28+
29+
// Still Watching vars
30+
private var isWatchingEpisodes = false
31+
private var episodeCount = 0
32+
private var watchTime = 0L
33+
private var episodeInteractMs = 0L
34+
private var episodeWasInterrupted: Boolean = false
35+
private var showStillWatching: Boolean = false
36+
37+
// Injects
38+
private val userPreferences by inject<UserPreferences>()
39+
private val playbackControllerContainer by inject<PlaybackControllerContainer>()
40+
41+
// Preferences
42+
43+
private val inAppEnabled get() = userPreferences[UserPreferences.screensaverInAppEnabled]
44+
private val timeout get() = userPreferences[UserPreferences.screensaverInAppTimeout].milliseconds
45+
46+
// State
47+
48+
private val _screensaverVisible = MutableStateFlow(false)
49+
val visible get() = _screensaverVisible.asStateFlow()
50+
51+
private val _keepScreenOn = MutableStateFlow(inAppEnabled)
52+
val keepScreenOn get() = _keepScreenOn.asStateFlow()
53+
54+
var activityPaused: Boolean = false
55+
set(value) {
56+
field = value
57+
notifyInteraction(canCancel = true, userInitiated = false)
58+
}
59+
60+
init {
61+
notifyInteraction(canCancel = true, userInitiated = false)
62+
}
63+
64+
fun getShowStillWatching(): Boolean {
65+
return showStillWatching
66+
}
67+
68+
fun notifyStartSession(item: BaseItemDto, items: List<BaseItemDto>) {
69+
// No need to track when only watching 1 episode
70+
if (itemIsEpisode(item) && items.size > 1) {
71+
resetSession()
72+
episodeWasInterrupted = false
73+
showStillWatching = false
74+
}
75+
}
76+
77+
fun onEpisodeWatched() {
78+
if (!episodeWasInterrupted) episodeCount++
79+
calculateWatchTime()
80+
episodeWasInterrupted = false
81+
checkStillWatchingStatus()
82+
}
83+
84+
fun notifyStart(item: BaseItemDto) {
85+
if (itemIsEpisode(item)) {
86+
isWatchingEpisodes = true
87+
}
88+
}
89+
90+
private fun checkStillWatchingStatus() {
91+
val stillWatchingSetting = StillWatchingStates.getSetting(userPreferences[UserPreferences.stillWatchingBehavior].toString())
92+
93+
val minMinutesInMs = stillWatchingSetting.minMinutes.toLong() * MILLIS_PER_MIN
94+
95+
val episodeRequirementMet = episodeCount == stillWatchingSetting.episodeCount
96+
val watchTimeRequirementMet = watchTime >= minMinutesInMs
97+
98+
if (episodeRequirementMet || watchTimeRequirementMet) {
99+
showStillWatching = true
100+
}
101+
}
102+
103+
fun notifyInteraction(canCancel: Boolean, userInitiated: Boolean) {
104+
// Cancel pending screensaver timer (if any)
105+
timer?.cancel()
106+
107+
// If watching episodes, reset episode count and watch time
108+
if (isWatchingEpisodes && userInitiated) {
109+
resetSession()
110+
111+
val playbackController: PlaybackController = playbackControllerContainer.playbackController!!
112+
113+
episodeInteractMs = playbackController.currentPosition
114+
episodeWasInterrupted = true
115+
}
116+
117+
// Hide screensaver when interacted with allowed cancellation or when disabled
118+
if (_screensaverVisible.value && (canCancel || !inAppEnabled || activityPaused)) {
119+
_screensaverVisible.value = false
120+
}
121+
122+
// Create new timer to show screensaver when enabled
123+
if (inAppEnabled && !activityPaused && locks == 0) {
124+
timer = viewModelScope.launch {
125+
delay(timeout)
126+
_screensaverVisible.value = true
127+
}
128+
}
129+
130+
// Update KEEP_SCREEN_ON flag value
131+
_keepScreenOn.value = inAppEnabled || locks > 0
132+
}
133+
134+
/**
135+
* Create a lock that prevents the screensaver for running until the returned function is called
136+
* or the lifecycle is destroyed.
137+
*
138+
* @return Function to cancel the lock
139+
*/
140+
fun addLifecycleLock(lifecycle: Lifecycle): () -> Unit {
141+
if (lifecycle.currentState == Lifecycle.State.DESTROYED) return {}
142+
143+
val lock = ScreensaverLock(lifecycle)
144+
lock.activate()
145+
return lock::cancel
146+
}
147+
148+
private inner class ScreensaverLock(private val lifecycle: Lifecycle) : LifecycleEventObserver {
149+
private var active: Boolean = false
150+
151+
override fun onStateChanged(source: LifecycleOwner, event: Lifecycle.Event) {
152+
if (event == Lifecycle.Event.ON_DESTROY) cancel()
153+
}
154+
155+
fun activate() {
156+
if (active) return
157+
lifecycle.addObserver(this)
158+
locks++
159+
notifyInteraction(canCancel = true, userInitiated = false)
160+
active = true
161+
}
162+
163+
fun cancel() {
164+
if (!active) return
165+
locks--
166+
lifecycle.removeObserver(this)
167+
notifyInteraction(canCancel = false, userInitiated = false)
168+
active = false
169+
}
170+
}
171+
172+
private fun itemIsEpisode(item: BaseItemDto? = null): Boolean {
173+
if (item != null) {
174+
return item.type == BaseItemKind.EPISODE
175+
}
176+
177+
val playbackController = playbackControllerContainer.playbackController
178+
179+
return playbackController?.currentlyPlayingItem?.type == BaseItemKind.EPISODE
180+
}
181+
182+
private fun resetSession() {
183+
watchTime = 0L
184+
episodeCount = 0
185+
episodeInteractMs = 0L
186+
}
187+
188+
private fun calculateWatchTime() {
189+
val duration = playbackControllerContainer.playbackController!!.duration
190+
val durationWatchedUninterrupted = if (episodeWasInterrupted) duration - episodeInteractMs else duration
191+
watchTime += durationWatchedUninterrupted
192+
}
193+
}

app/src/main/java/org/jellyfin/androidtv/ui/ScreensaverViewModel.kt

-103
This file was deleted.

0 commit comments

Comments
 (0)