From 7ee20f6e448786e9da7763a8f55afa0715d3540b Mon Sep 17 00:00:00 2001 From: Henning Weiss Date: Sat, 14 Dec 2024 15:23:40 +0100 Subject: [PATCH 1/3] Tweaks to watch next row. - episodes now have meta data in line with other streaming apps - watch next is now correctly ordered - movies to resume will now be inserted correctly --- .../integration/LeanbackChannelWorker.kt | 93 ++++++++++++++----- 1 file changed, 69 insertions(+), 24 deletions(-) diff --git a/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt b/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt index 0c7c85bd46..779d8244d6 100644 --- a/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt +++ b/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt @@ -48,7 +48,7 @@ import org.koin.core.component.KoinComponent import org.koin.core.component.inject import timber.log.Timber import java.time.Instant -import java.time.ZoneOffset +import java.time.ZoneId import java.time.format.DateTimeFormatter import kotlin.time.Duration @@ -400,16 +400,56 @@ class LeanbackChannelWorker( * or other types of media. Uses the [nextUpItems] parameter to store items returned by a * NextUpQuery(). */ + @SuppressLint("RestrictedApi") private fun updateWatchNext(nextUpItems: List) { - // Delete current items - context.contentResolver.delete(WatchNextPrograms.CONTENT_URI, null, null) + deletePrograms(nextUpItems) - // Add new items + // Get current watch next state + val currentWatchNextPrograms = getCurrentWatchNext() + + // Create all programs in nextUpItems but not in watch next + val programsToAdd = nextUpItems + .filter { next -> currentWatchNextPrograms.none{ it.internalProviderId == next.id.toString() }} context.contentResolver.bulkInsert( WatchNextPrograms.CONTENT_URI, - nextUpItems.map { item -> getBaseItemAsWatchNextProgram(item).toContentValues() } - .toTypedArray() - ) + programsToAdd.map{item -> getBaseItemAsWatchNextProgram(item).toContentValues() } + .toTypedArray()) + } + + /** + * Delete stale programs from the watch next row. Items that don't need to be touched are + * kept as is, so they keep their ordering in the watch next row. + */ + @SuppressLint("RestrictedApi") + private fun deletePrograms(nextUpItems: List) { + // Retrieve current watch next row + val currentWatchNextPrograms = getCurrentWatchNext() + + // Find all stale programs to delete + val deletedByUser = currentWatchNextPrograms.filter { !it.isBrowsable } + val noLongerInWatchNext = currentWatchNextPrograms.filter { (nextUpItems).none { next -> it.internalProviderId == next.id.toString() } } + val continueWatching = currentWatchNextPrograms.filter { it.watchNextType == WatchNextPrograms.WATCH_NEXT_TYPE_CONTINUE} + + // Delete the programs + (deletedByUser + noLongerInWatchNext + continueWatching) + .forEach { context.contentResolver.delete(TvContractCompat.buildWatchNextProgramUri(it.id), null, null) } + } + + /** + * Retrieves the current watch next row state. + */ + @SuppressLint("RestrictedApi") + private fun getCurrentWatchNext(): MutableList { + val currentWatchNextPrograms: MutableList = mutableListOf() + context.contentResolver.query(WatchNextPrograms.CONTENT_URI, WatchNextProgram.PROJECTION, null, null, null) + .use { cursor -> + if (cursor != null && cursor.moveToFirst()) { + do { + currentWatchNextPrograms.add(WatchNextProgram.fromCursor(cursor)) + } while (cursor.moveToNext()) + } + } + return currentWatchNextPrograms } /** @@ -431,39 +471,44 @@ class LeanbackChannelWorker( setPosterArtAspectRatio(WatchNextPrograms.ASPECT_RATIO_MOVIE_POSTER) } - // Name - if (item.seriesName != null) setTitle("${item.seriesName} - ${item.name}") + // Name and episode details + if (item.seriesName != null) { + setTitle(item.seriesName) + setEpisodeTitle(item.name) + setEpisodeNumber(item.indexNumber ?: 0) + setSeasonNumber(item.parentIndexNumber ?: 0) + } else setTitle(item.name) + setDescription(item.overview?.stripHtml()) + // Poster setPosterArtUri(item.getPosterArtImageUrl(preferParentThumb)) - // Use date created or fallback to current time if unavailable - var engagement = item.dateCreated - when { // User has started playing the episode (item.userData?.playbackPositionTicks ?: 0) > 0 -> { setWatchNextType(WatchNextPrograms.WATCH_NEXT_TYPE_CONTINUE) setLastPlaybackPositionMillis(item.userData!!.playbackPositionTicks.ticks.inWholeMilliseconds.toInt()) // Use last played date to prioritize - engagement = item.userData?.lastPlayedDate + + setLastEngagementTimeUtcMillis(item.userData?.lastPlayedDate?.atZone(ZoneId.systemDefault())?.toInstant()?.toEpochMilli() + ?: Instant.now().toEpochMilli()) } // First episode of the season - item.indexNumber == 1 -> setWatchNextType(WatchNextPrograms.WATCH_NEXT_TYPE_NEW) + item.indexNumber == 1 -> { + setWatchNextType(WatchNextPrograms.WATCH_NEXT_TYPE_NEW) + setLastEngagementTimeUtcMillis(Instant.now().toEpochMilli()) + } // Default - else -> setWatchNextType(WatchNextPrograms.WATCH_NEXT_TYPE_NEXT) + else -> { + setWatchNextType(WatchNextPrograms.WATCH_NEXT_TYPE_NEXT) + setLastEngagementTimeUtcMillis(Instant.now().toEpochMilli()) + } } - setLastEngagementTimeUtcMillis( - engagement?.toInstant(ZoneOffset.UTC)?.toEpochMilli() - ?: Instant.now().toEpochMilli() - ) - - // Episode runtime has been determined - item.runTimeTicks?.let { runTimeTicks -> - setDurationMillis(runTimeTicks.ticks.inWholeMilliseconds.toInt()) - } + // Runtime has been determined + setDurationMillis(item.runTimeTicks?.ticks?.inWholeMilliseconds?.toInt() ?: 0) // Set intent to open the episode setIntent(Intent(context, StartupActivity::class.java).apply { From 73af1188fb2651b85126dd8c474c421cf525a2f0 Mon Sep 17 00:00:00 2001 From: Henning Weiss Date: Wed, 25 Dec 2024 12:23:25 +0100 Subject: [PATCH 2/3] WATCH_NEXT_TYPE_NEW now ordered by item creation date. --- .../jellyfin/androidtv/integration/LeanbackChannelWorker.kt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt b/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt index 779d8244d6..d623b65c8c 100644 --- a/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt +++ b/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt @@ -498,7 +498,8 @@ class LeanbackChannelWorker( // First episode of the season item.indexNumber == 1 -> { setWatchNextType(WatchNextPrograms.WATCH_NEXT_TYPE_NEW) - setLastEngagementTimeUtcMillis(Instant.now().toEpochMilli()) + setLastEngagementTimeUtcMillis(item.dateCreated?.atZone(ZoneId.systemDefault())?.toInstant()?.toEpochMilli() + ?: Instant.now().toEpochMilli()) } // Default else -> { From 8915b92a9d1cc06e3c8c1fa1aae0cbf850eff5f7 Mon Sep 17 00:00:00 2001 From: Henning Weiss Date: Sat, 28 Dec 2024 11:27:23 +0100 Subject: [PATCH 3/3] Overview field is retrieved from server. --- .../jellyfin/androidtv/integration/LeanbackChannelWorker.kt | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt b/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt index d623b65c8c..ce51c0a3e6 100644 --- a/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt +++ b/app/src/main/java/org/jellyfin/androidtv/integration/LeanbackChannelWorker.kt @@ -270,7 +270,7 @@ class LeanbackChannelWorker( withContext(Dispatchers.IO) { val resume = async { api.itemsApi.getResumeItems( - fields = listOf(ItemFields.DATE_CREATED), + fields = listOf(ItemFields.DATE_CREATED, ItemFields.OVERVIEW), imageTypeLimit = 1, limit = 10, mediaTypes = listOf(MediaType.VIDEO), @@ -284,7 +284,7 @@ class LeanbackChannelWorker( imageTypeLimit = 1, limit = 10, enableResumable = false, - fields = listOf(ItemFields.DATE_CREATED), + fields = listOf(ItemFields.DATE_CREATED, ItemFields.OVERVIEW), ).content.items }