From e8ed687dc7acf3550fdcb3b72112ee3c75f570f7 Mon Sep 17 00:00:00 2001 From: jarodaerts Date: Sat, 14 Oct 2023 22:21:08 -0400 Subject: [PATCH 1/6] Add support for TV show season level extra features --- components/tvshows/TVEpisodes.brs | 32 +++++++++++++++++++++++++--- components/tvshows/TVEpisodes.xml | 3 +++ settings/settings.json | 4 +--- source/ShowScenes.brs | 6 ++++++ source/api/Items.brs | 35 +++++++++++++++++++++++++++++++ 5 files changed, 74 insertions(+), 6 deletions(-) diff --git a/components/tvshows/TVEpisodes.brs b/components/tvshows/TVEpisodes.brs index 8872ffc37..69167a734 100644 --- a/components/tvshows/TVEpisodes.brs +++ b/components/tvshows/TVEpisodes.brs @@ -9,6 +9,7 @@ sub init() m.rows = m.top.findNode("picker") m.poster = m.top.findNode("seasonPoster") m.Shuffle = m.top.findNode("Shuffle") + m.Extras = m.top.findNode("Extras") m.tvEpisodeRow = m.top.findNode("tvEpisodeRow") m.unplayedCount = m.top.findNode("unplayedCount") @@ -21,6 +22,12 @@ sub setSeasonLoading() m.top.overhangTitle = tr("Loading...") end sub +sub setExtraButtonVisibility() + if isValid(m.top.extrasObjects) and (not m.top.extrasObjects.items.IsEmpty()) + m.Extras.visible = true + end if +end sub + sub updateSeason() if m.global.session.user.settings["ui.tvshows.disableUnwatchedEpisodeCount"] = false if isValid(m.top.seasonData) and isValid(m.top.seasonData.UserData) and isValid(m.top.seasonData.UserData.UnplayedItemCount) @@ -45,13 +52,23 @@ function onKeyEvent(key as string, press as boolean) as boolean return true end if - if key = "right" and (m.Shuffle.hasFocus()) + if key = "right" and (m.Shuffle.hasFocus() or m.Extras.hasFocus()) m.tvEpisodeRow.setFocus(true) return true end if - if key = "OK" or key = "play" + showExtras = isValid(m.top.extrasObjects) and (not m.top.extrasObjects.IsEmpty()) + if showExtras and key = "up" and (m.Extras.hasFocus()) + m.Shuffle.setFocus(true) + return true + end if + if showExtras and key = "down" and (m.Shuffle.hasFocus()) + m.Extras.setFocus(true) + return true + end if + + if key = "OK" or key = "play" if m.Shuffle.hasFocus() episodeList = m.rows.getChild(0).objects.items @@ -66,6 +83,16 @@ function onKeyEvent(key as string, press as boolean) as boolean m.global.queueManager.callFunc("playQueue") return true end if + + if showExtras and m.Extras.hasFocus() + if m.Extras.text.trim() = "Extras" + m.Extras.text = "Episodes" + m.top.objects = m.top.extrasObjects + else + m.Extras.text = "Extras" + m.top.objects = m.top.episodeObjects + end if + end if end if focusedChild = m.top.focusedChild.focusedChild @@ -81,7 +108,6 @@ function onKeyEvent(key as string, press as boolean) as boolean m.top.lastFocus = focusedChild itemToPlay = focusedChild.content.getChild(focusedChild.rowItemFocused[0]).getChild(0) if isValid(itemToPlay) and isValid(itemToPlay.id) and itemToPlay.id <> "" - itemToPlay.type = "Episode" m.top.quickPlayNode = itemToPlay end if handled = true diff --git a/components/tvshows/TVEpisodes.xml b/components/tvshows/TVEpisodes.xml index 004c07fb9..8492abc67 100644 --- a/components/tvshows/TVEpisodes.xml +++ b/components/tvshows/TVEpisodes.xml @@ -7,6 +7,7 @@ + @@ -14,5 +15,7 @@ + + \ No newline at end of file diff --git a/settings/settings.json b/settings/settings.json index 422174741..6e5a2792b 100644 --- a/settings/settings.json +++ b/settings/settings.json @@ -10,7 +10,6 @@ "type": "bool", "default": "false" } - ] }, { @@ -126,7 +125,6 @@ ] } ] - }, { "title": "Video Codec Support", @@ -362,4 +360,4 @@ } ] } -] +] \ No newline at end of file diff --git a/source/ShowScenes.brs b/source/ShowScenes.brs index 48a1853db..a44a8427c 100644 --- a/source/ShowScenes.brs +++ b/source/ShowScenes.brs @@ -743,6 +743,9 @@ function CreateSeasonDetailsGroup(series as object, season as object) as dynamic m.global.sceneManager.callFunc("pushScene", group) group.seasonData = seasonMetaData.json group.objects = TVEpisodes(series.id, season.id) + group.episodeObjects = group.objects + group.extrasObjects = TVSeasonExtras(season.id) + ' watch for button presses group.observeField("episodeSelected", m.port) group.observeField("quickPlayNode", m.port) @@ -770,6 +773,9 @@ function CreateSeasonDetailsGroupByID(seriesID as string, seasonID as string) as m.global.sceneManager.callFunc("pushScene", group) group.seasonData = seasonMetaData.json group.objects = TVEpisodes(seriesID, seasonID) + group.episodeObjects = group.objects + group.extrasObjects = TVSeasonExtras(seasonID) + ' watch for button presses group.observeField("episodeSelected", m.port) group.observeField("quickPlayNode", m.port) diff --git a/source/api/Items.brs b/source/api/Items.brs index a3d376471..33a384018 100644 --- a/source/api/Items.brs +++ b/source/api/Items.brs @@ -445,6 +445,41 @@ function TVEpisodes(show_id as string, season_id as string) as dynamic return data end function +function TVSeasonExtras(season_id as string) as dynamic + url = Substitute("/Users/{0}/Items/{1}/SpecialFeatures", m.global.session.user.id, season_id) + resp = APIRequest(url) + + ' validate data + data = getJson(resp) + if data = invalid then return invalid + + results = [] + for each item in data + imgParams = { "maxWidth": 400, "maxheight": 250 } + tmp = CreateObject("roSGNode", "TVEpisodeData") + tmp.image = PosterImage(item.id, imgParams) + if tmp.image <> invalid + tmp.image.posterDisplayMode = "scaleToZoom" + end if + tmp.json = item + + ' Force item type to Video so episode auto queue is not attempted + tmp.type = "Video" + tmpMetaData = ItemMetaData(item.id) + + ' Validate meta data + if tmpMetaData <> invalid and tmpMetaData.overview <> invalid + tmp.overview = tmpMetaData.overview + end if + results.push(tmp) + end for + + ' Build that data format that the TVEpisodeRow expects + extrasData = { Items: [] } + extrasData.Items = results + return extrasData +end function + function TVEpisodeShuffleList(show_id as string) url = Substitute("Shows/{0}/Episodes", show_id) resp = APIRequest(url, { From 4fed592e37d72e9a861c389b288bbbab97dddc72 Mon Sep 17 00:00:00 2001 From: jarodaerts Date: Mon, 16 Oct 2023 20:33:06 -0500 Subject: [PATCH 2/6] Address PR feedback Switch to using sdk for calls to server, improve structure/style of code, and fix formating --- components/tvshows/TVEpisodes.brs | 32 +++++++++++++----------- components/tvshows/TVEpisodes.xml | 4 +-- locale/en_US/translations.ts | 8 ++++++ source/api/Items.brs | 41 +++++++++++++++---------------- 4 files changed, 47 insertions(+), 38 deletions(-) diff --git a/components/tvshows/TVEpisodes.brs b/components/tvshows/TVEpisodes.brs index 69167a734..24959c754 100644 --- a/components/tvshows/TVEpisodes.brs +++ b/components/tvshows/TVEpisodes.brs @@ -2,14 +2,15 @@ import "pkg:/source/api/Image.brs" import "pkg:/source/api/baserequest.brs" import "pkg:/source/utils/config.brs" import "pkg:/source/utils/misc.brs" +import "pkg:/source/api/sdk.bs" sub init() m.top.optionsAvailable = false m.rows = m.top.findNode("picker") m.poster = m.top.findNode("seasonPoster") - m.Shuffle = m.top.findNode("Shuffle") - m.Extras = m.top.findNode("Extras") + m.shuffle = m.top.findNode("shuffle") + m.extras = m.top.findNode("extras") m.tvEpisodeRow = m.top.findNode("tvEpisodeRow") m.unplayedCount = m.top.findNode("unplayedCount") @@ -22,8 +23,9 @@ sub setSeasonLoading() m.top.overhangTitle = tr("Loading...") end sub +' Updates the visibility of the Extras button based on if this season has any extra features sub setExtraButtonVisibility() - if isValid(m.top.extrasObjects) and (not m.top.extrasObjects.items.IsEmpty()) + if isValid(m.top.extrasObjects) and isValidAndNotEmpty(m.top.extrasObjects.items) m.Extras.visible = true end if end sub @@ -40,36 +42,36 @@ sub updateSeason() imgParams = { "maxHeight": 450, "maxWidth": 300 } m.poster.uri = ImageURL(m.top.seasonData.Id, "Primary", imgParams) - m.Shuffle.visible = true + m.shuffle.visible = true m.top.overhangTitle = m.top.seasonData.SeriesName + " - " + m.top.seasonData.name end sub +' Handle navigation input from the remote and act on it function onKeyEvent(key as string, press as boolean) as boolean handled = false if key = "left" and m.tvEpisodeRow.hasFocus() - m.Shuffle.setFocus(true) + m.shuffle.setFocus(true) return true end if - if key = "right" and (m.Shuffle.hasFocus() or m.Extras.hasFocus()) + if key = "right" and (m.shuffle.hasFocus() or m.Extras.hasFocus()) m.tvEpisodeRow.setFocus(true) return true end if - showExtras = isValid(m.top.extrasObjects) and (not m.top.extrasObjects.IsEmpty()) - if showExtras and key = "up" and (m.Extras.hasFocus()) - m.Shuffle.setFocus(true) + if m.extras.visible and key = "up" and (m.Extras.hasFocus()) + m.shuffle.setFocus(true) return true end if - if showExtras and key = "down" and (m.Shuffle.hasFocus()) + if m.extras.visible and key = "down" and (m.shuffle.hasFocus()) m.Extras.setFocus(true) return true end if if key = "OK" or key = "play" - if m.Shuffle.hasFocus() + if m.shuffle.hasFocus() episodeList = m.rows.getChild(0).objects.items for i = 0 to episodeList.count() - 1 @@ -84,12 +86,12 @@ function onKeyEvent(key as string, press as boolean) as boolean return true end if - if showExtras and m.Extras.hasFocus() - if m.Extras.text.trim() = "Extras" - m.Extras.text = "Episodes" + if m.extras.visible and m.extras.hasFocus() + if LCase(m.extras.text.trim()) = LCase(tr("Extras")) + m.extras.text = tr("Episodes") m.top.objects = m.top.extrasObjects else - m.Extras.text = "Extras" + m.extras.text = tr("Extras") m.top.objects = m.top.episodeObjects end if end if diff --git a/components/tvshows/TVEpisodes.xml b/components/tvshows/TVEpisodes.xml index 8492abc67..c87d404d9 100644 --- a/components/tvshows/TVEpisodes.xml +++ b/components/tvshows/TVEpisodes.xml @@ -6,8 +6,8 @@