Skip to content

Commit c682a55

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

23 files changed

+712
-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,194 @@
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 timber.log.Timber
23+
import kotlin.time.Duration.Companion.milliseconds
24+
25+
class InteractionTrackerViewModel : ViewModel(), KoinComponent {
26+
// Screensaver vars
27+
private var timer: Job? = null
28+
private var locks = 0
29+
30+
// Still Watching vars
31+
private var isWatchingEpisodes = false
32+
private var episodeCount = 0
33+
private var watchTime = 0L
34+
private var episodeInteractMs = 0L
35+
private var episodeWasInterrupted: Boolean = false
36+
private var showStillWatching: Boolean = false
37+
38+
// Injects
39+
private val userPreferences by inject<UserPreferences>()
40+
private val playbackControllerContainer by inject<PlaybackControllerContainer>()
41+
42+
// Preferences
43+
44+
private val inAppEnabled get() = userPreferences[UserPreferences.screensaverInAppEnabled]
45+
private val timeout get() = userPreferences[UserPreferences.screensaverInAppTimeout].milliseconds
46+
47+
// State
48+
49+
private val _screensaverVisible = MutableStateFlow(false)
50+
val visible get() = _screensaverVisible.asStateFlow()
51+
52+
private val _keepScreenOn = MutableStateFlow(inAppEnabled)
53+
val keepScreenOn get() = _keepScreenOn.asStateFlow()
54+
55+
var activityPaused: Boolean = false
56+
set(value) {
57+
field = value
58+
notifyInteraction(canCancel = true, userInitiated = false)
59+
}
60+
61+
init {
62+
notifyInteraction(canCancel = true, userInitiated = false)
63+
}
64+
65+
fun getShowStillWatching(): Boolean {
66+
return showStillWatching
67+
}
68+
69+
fun notifyStartSession(item: BaseItemDto, items: List<BaseItemDto>) {
70+
// No need to track when only watching 1 episode
71+
if (itemIsEpisode(item) && items.size > 1) {
72+
Timber.i("Start tracker")
73+
resetSession()
74+
episodeWasInterrupted = false
75+
}
76+
}
77+
78+
fun onEpisodeWatched() {
79+
Timber.i("Watcher onEpisodeWatched")
80+
if (!episodeWasInterrupted) episodeCount++
81+
calculateWatchTime()
82+
episodeWasInterrupted = false
83+
checkStillWatchingStatus()
84+
}
85+
86+
fun notifyStart(item: BaseItemDto) {
87+
if (itemIsEpisode(item)) {
88+
isWatchingEpisodes = true
89+
}
90+
}
91+
92+
private fun checkStillWatchingStatus() {
93+
val stillWatchingSetting = StillWatchingStates.getSetting(userPreferences[UserPreferences.stillWatchingBehavior].toString())
94+
95+
val minMinutesInMs = stillWatchingSetting.minMinutes.toLong() * MILLIS_PER_MIN
96+
97+
val episodeRequirementMet = episodeCount == stillWatchingSetting.episodeCount
98+
val watchTimeRequirementMet = watchTime >= minMinutesInMs
99+
100+
if (episodeRequirementMet || watchTimeRequirementMet) {
101+
showStillWatching = true
102+
}
103+
}
104+
105+
fun notifyInteraction(canCancel: Boolean, userInitiated: Boolean) {
106+
// Cancel pending screensaver timer (if any)
107+
timer?.cancel()
108+
109+
// If watching episodes, reset episode count and watch time
110+
if (isWatchingEpisodes && userInitiated) {
111+
resetSession()
112+
113+
val playbackController: PlaybackController = playbackControllerContainer.playbackController!!
114+
115+
episodeInteractMs = playbackController.currentPosition
116+
episodeWasInterrupted = true
117+
}
118+
119+
// Hide screensaver when interacted with allowed cancellation or when disabled
120+
if (_screensaverVisible.value && (canCancel || !inAppEnabled || activityPaused)) {
121+
_screensaverVisible.value = false
122+
}
123+
124+
// Create new timer to show screensaver when enabled
125+
if (inAppEnabled && !activityPaused && locks == 0) {
126+
timer = viewModelScope.launch {
127+
delay(timeout)
128+
_screensaverVisible.value = true
129+
}
130+
}
131+
132+
// Update KEEP_SCREEN_ON flag value
133+
_keepScreenOn.value = inAppEnabled || locks > 0
134+
}
135+
136+
/**
137+
* Create a lock that prevents the screensaver for running until the returned function is called
138+
* or the lifecycle is destroyed.
139+
*
140+
* @return Function to cancel the lock
141+
*/
142+
fun addLifecycleLock(lifecycle: Lifecycle): () -> Unit {
143+
if (lifecycle.currentState == Lifecycle.State.DESTROYED) return {}
144+
145+
val lock = ScreensaverLock(lifecycle)
146+
lock.activate()
147+
return lock::cancel
148+
}
149+
150+
private inner class ScreensaverLock(private val lifecycle: Lifecycle) : LifecycleEventObserver {
151+
private var active: Boolean = false
152+
153+
override fun onStateChanged(source: LifecycleOwner, event: Lifecycle.Event) {
154+
if (event == Lifecycle.Event.ON_DESTROY) cancel()
155+
}
156+
157+
fun activate() {
158+
if (active) return
159+
lifecycle.addObserver(this)
160+
locks++
161+
notifyInteraction(canCancel = true, userInitiated = false)
162+
active = true
163+
}
164+
165+
fun cancel() {
166+
if (!active) return
167+
locks--
168+
lifecycle.removeObserver(this)
169+
notifyInteraction(canCancel = false, userInitiated = false)
170+
active = false
171+
}
172+
}
173+
174+
private fun itemIsEpisode(item: BaseItemDto? = null): Boolean {
175+
if (item != null) {
176+
return item.type == BaseItemKind.EPISODE
177+
}
178+
179+
val playbackController = playbackControllerContainer.playbackController
180+
181+
return playbackController?.currentlyPlayingItem?.type == BaseItemKind.EPISODE
182+
}
183+
184+
private fun resetSession() {
185+
watchTime = 0L
186+
episodeCount = 0
187+
}
188+
189+
private fun calculateWatchTime() {
190+
val duration = playbackControllerContainer.playbackController!!.duration
191+
val durationWatchedUninterrupted = if (episodeWasInterrupted) duration - episodeInteractMs else duration
192+
watchTime += durationWatchedUninterrupted
193+
}
194+
}

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

-103
This file was deleted.

0 commit comments

Comments
 (0)