mirror of
https://github.com/jellyfin/jellyfin-androidtv.git
synced 2025-08-06 15:20:34 +08:00
Fix watch next row ordering, Change watch next row meta data (#4298)
* 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 * WATCH_NEXT_TYPE_NEW now ordered by item creation date. * Overview field is retrieved from server. * Improvements to handling null values in LeanbackChannelWorker. * Additional null check. * Style improvements.
This commit is contained in:
@ -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
|
||||
|
||||
@ -355,23 +355,11 @@ class LeanbackChannelWorker(
|
||||
)
|
||||
.setTitle(item.seriesName ?: item.name)
|
||||
.setEpisodeTitle(if (item.type == BaseItemKind.EPISODE) item.name else null)
|
||||
.setSeasonNumber(seasonString, item.parentIndexNumber ?: 0)
|
||||
.setEpisodeNumber(episodeString, item.indexNumber ?: 0)
|
||||
.setDescription(item.overview?.stripHtml())
|
||||
.setReleaseDate(
|
||||
if (item.premiereDate != null) DateTimeFormatter.ISO_DATE.format(item.premiereDate)
|
||||
else null
|
||||
)
|
||||
.setDurationMillis(
|
||||
if (item.runTimeTicks?.ticks != null) {
|
||||
// If we are resuming, we need to show remaining time, cause GoogleTV
|
||||
// ignores setLastPlaybackPositionMillis
|
||||
val duration = item.runTimeTicks?.ticks ?: Duration.ZERO
|
||||
val playbackPosition = item.userData?.playbackPositionTicks?.ticks
|
||||
?: Duration.ZERO
|
||||
(duration - playbackPosition).inWholeMilliseconds.toInt()
|
||||
} else 0
|
||||
)
|
||||
.setPosterArtUri(imageUri)
|
||||
.setPosterArtAspectRatio(
|
||||
when (item.type) {
|
||||
@ -385,8 +373,22 @@ class LeanbackChannelWorker(
|
||||
putExtra(StartupActivity.EXTRA_ITEM_ID, item.id.toString())
|
||||
putExtra(StartupActivity.EXTRA_ITEM_IS_USER_VIEW, item.type == BaseItemKind.COLLECTION_FOLDER)
|
||||
})
|
||||
.build()
|
||||
.toContentValues()
|
||||
.setDurationMillis(
|
||||
if (item.runTimeTicks?.ticks != null) {
|
||||
// If we are resuming, we need to show remaining time, cause GoogleTV
|
||||
// ignores setLastPlaybackPositionMillis
|
||||
val duration = item.runTimeTicks?.ticks ?: Duration.ZERO
|
||||
val playbackPosition = item.userData?.playbackPositionTicks?.ticks
|
||||
?: Duration.ZERO
|
||||
(duration - playbackPosition).inWholeMilliseconds.toInt()
|
||||
} else 0
|
||||
)
|
||||
.apply {
|
||||
if ((item.parentIndexNumber ?: 0) > 0)
|
||||
setSeasonNumber(seasonString, item.parentIndexNumber!!)
|
||||
if ((item.indexNumber ?: 0) > 0)
|
||||
setEpisodeNumber(episodeString, item.indexNumber!!)
|
||||
}.build().toContentValues()
|
||||
}
|
||||
|
||||
/**
|
||||
@ -394,16 +396,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<BaseItemDto>) {
|
||||
// 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<BaseItemDto>) {
|
||||
// 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<WatchNextProgram> {
|
||||
val currentWatchNextPrograms: MutableList<WatchNextProgram> = 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
|
||||
}
|
||||
|
||||
/**
|
||||
@ -425,39 +467,47 @@ class LeanbackChannelWorker(
|
||||
setPosterArtAspectRatio(WatchNextPrograms.ASPECT_RATIO_MOVIE_POSTER)
|
||||
}
|
||||
|
||||
// Name
|
||||
if (item.seriesName != null) setTitle("${item.seriesName} - ${item.name}")
|
||||
else setTitle(item.name)
|
||||
// Name and episode details
|
||||
if (item.seriesName != null) {
|
||||
setTitle(item.seriesName)
|
||||
setEpisodeTitle(item.name)
|
||||
|
||||
item.indexNumber?.takeIf { it > 0 }?.let { setEpisodeNumber(it) }
|
||||
item.parentIndexNumber?.takeIf { it > 0 }?.let { setSeasonNumber(it) }
|
||||
} 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(item.dateCreated?.atZone(ZoneId.systemDefault())?.toInstant()?.toEpochMilli()
|
||||
?: 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
|
||||
item.runTimeTicks?.ticks?.let { setDurationMillis(it.inWholeMilliseconds.toInt()) }
|
||||
|
||||
// Set intent to open the episode
|
||||
setIntent(Intent(context, StartupActivity::class.java).apply {
|
||||
|
Reference in New Issue
Block a user