2023-11-11 13:41:20 +00:00
|
|
|
<!DOCTYPE html><html lang="en" style="font-size:16px"><head><meta charset="utf-8"><meta name="viewport" content="width=device-width,initial-scale=1"><title>Source: components/video/VideoPlayerView.bs</title><!--[if lt IE 9]>
|
|
|
|
<script src="//html5shiv.googlecode.com/svn/trunk/html5.js"></script>
|
2024-02-09 14:58:42 +00:00
|
|
|
<![endif]--><script src="scripts/third-party/hljs.js" defer="defer"></script><script src="scripts/third-party/hljs-line-num.js" defer="defer"></script><script src="scripts/third-party/popper.js" defer="defer"></script><script src="scripts/third-party/tippy.js" defer="defer"></script><script src="scripts/third-party/tocbot.min.js"></script><script>var baseURL="/",locationPathname="";baseURL=(baseURL=(baseURL="https://jellyfin.github.io/jellyfin-roku/").replace(/https?:\/\//i,"")).substr(baseURL.indexOf("/"))</script><link rel="stylesheet" href="styles/clean-jsdoc-theme.min.css"><svg aria-hidden="true" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" style="display:none"><defs><symbol id="copy-icon" viewbox="0 0 488.3 488.3"><g><path d="M314.25,85.4h-227c-21.3,0-38.6,17.3-38.6,38.6v325.7c0,21.3,17.3,38.6,38.6,38.6h227c21.3,0,38.6-17.3,38.6-38.6V124 C352.75,102.7,335.45,85.4,314.25,85.4z M325.75,449.6c0,6.4-5.2,11.6-11.6,11.6h-227c-6.4,0-11.6-5.2-11.6-11.6V124 c0-6.4,5.2-11.6,11.6-11.6h227c6.4,0,11.6,5.2,11.6,11.6V449.6z"/><path d="M401.05,0h-227c-21.3,0-38.6,17.3-38.6,38.6c0,7.5,6,13.5,13.5,13.5s13.5-6,13.5-13.5c0-6.4,5.2-11.6,11.6-11.6h227 c6.4,0,11.6,5.2,11.6,11.6v325.7c0,6.4-5.2,11.6-11.6,11.6c-7.5,0-13.5,6-13.5,13.5s6,13.5,13.5,13.5c21.3,0,38.6-17.3,38.6-38.6 V38.6C439.65,17.3,422.35,0,401.05,0z"/></g></symbol><symbol id="search-icon" viewBox="0 0 512 512"><g><g><path d="M225.474,0C101.151,0,0,101.151,0,225.474c0,124.33,101.151,225.474,225.474,225.474 c124.33,0,225.474-101.144,225.474-225.474C450.948,101.151,349.804,0,225.474,0z M225.474,409.323 c-101.373,0-183.848-82.475-183.848-183.848S124.101,41.626,225.474,41.626s183.848,82.475,183.848,183.848 S326.847,409.323,225.474,409.323z"/></g></g><g><g><path d="M505.902,476.472L386.574,357.144c-8.131-8.131-21.299-8.131-29.43,0c-8.131,8.124-8.131,21.306,0,29.43l119.328,119.328 c4.065,4.065,9.387,6.098,14.715,6.098c5.321,0,10.649-2.033,14.715-6.098C514.033,497.778,514.033,484.596,505.902,476.472z"/></g></g></symbol><symbol id="font-size-icon" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M11.246 15H4.754l-2 5H.6L7 4h2l6.4 16h-2.154l-2-5zm-.8-2L8 6.885 5.554 13h4.892zM21 12.535V12h2v8h-2v-.535a4 4 0 1 1 0-6.93zM19 18a2 2 0 1 0 0-4 2 2 0 0 0 0 4z"/></symbol><symbol id="add-icon" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M11 11V5h2v6h6v2h-6v6h-2v-6H5v-2z"/></symbol><symbol id="minus-icon" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M5 11h14v2H5z"/></symbol><symbol id="dark-theme-icon" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M10 7a7 7 0 0 0 12 4.9v.1c0 5.523-4.477 10-10 10S2 17.523 2 12 6.477 2 12 2h.1A6.979 6.979 0 0 0 10 7zm-6 5a8 8 0 0 0 15.062 3.762A9 9 0 0 1 8.238 4.938 7.999 7.999 0 0 0 4 12z"/></symbol><symbol id="light-theme-icon" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M12 18a6 6 0 1 1 0-12 6 6 0 0 1 0 12zm0-2a4 4 0 1 0 0-8 4 4 0 0 0 0 8zM11 1h2v3h-2V1zm0 19h2v3h-2v-3zM3.515 4.929l1.414-1.414L7.05 5.636 5.636 7.05 3.515 4.93zM16.95 18.364l1.414-1.414 2.121 2.121-1.414 1.414-2.121-2.121zm2.121-14.85l1.414 1.415-2.121 2.121-1.414-1.414 2.121-2.121zM5.636 16.95l1.414 1.414-2.121 2.121-1.414-1.414 2.121-2.121zM23 11v2h-3v-2h3zM4 11v2H1v-2h3z"/></symbol><symbol id="reset-icon" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M18.537 19.567A9.961 9.961 0 0 1 12 22C6.477 22 2 17.523 2 12S6.477 2 12 2s10 4.477 10 10c0 2.136-.67 4.116-1.81 5.74L17 12h3a8 8 0 1 0-2.46 5.772l.997 1.795z"/></symbol><symbol id="down-icon" viewBox="0 0 16 16"><path fill-rule="evenodd" clip-rule="evenodd" d="M12.7803 6.21967C13.0732 6.51256 13.0732 6.98744 12.7803 7.28033L8.53033 11.5303C8.23744 11.8232 7.76256 11.8232 7.46967 11.5303L3.21967 7.28033C2.92678 6.98744 2.92678 6.51256 3.21967 6.21967C3.51256 5.92678 3.98744 5.92678 4.28033 6.21967L8 9.93934L11.7197 6.21967C12.0126 5.92678 12.4874 5.92678 12.7803 6.21967Z"></path></symbol><symbol id="codepen-icon" viewBox="0 0 24 24"><path fill="none" d=
|
2023-11-11 02:08:52 +00:00
|
|
|
import "pkg:/source/utils/config.bs"
|
2023-10-06 03:18:36 +00:00
|
|
|
|
|
|
|
sub init()
|
|
|
|
' Hide the overhang on init to prevent showing 2 clocks
|
|
|
|
m.top.getScene().findNode("overhang").visible = false
|
|
|
|
m.currentItem = m.global.queueManager.callFunc("getCurrentItem")
|
|
|
|
|
|
|
|
m.top.id = m.currentItem.id
|
2023-11-10 23:51:00 +00:00
|
|
|
m.top.seekMode = "accurate"
|
|
|
|
|
|
|
|
m.playbackEnum = {
|
|
|
|
null: -10
|
|
|
|
}
|
2023-10-06 03:18:36 +00:00
|
|
|
|
|
|
|
' Load meta data
|
|
|
|
m.LoadMetaDataTask = CreateObject("roSGNode", "LoadVideoContentTask")
|
|
|
|
m.LoadMetaDataTask.itemId = m.currentItem.id
|
|
|
|
m.LoadMetaDataTask.itemType = m.currentItem.type
|
|
|
|
m.LoadMetaDataTask.selectedAudioStreamIndex = m.currentItem.selectedAudioStreamIndex
|
|
|
|
m.LoadMetaDataTask.observeField("content", "onVideoContentLoaded")
|
|
|
|
m.LoadMetaDataTask.control = "RUN"
|
|
|
|
|
2023-11-10 23:51:00 +00:00
|
|
|
m.chapterList = m.top.findNode("chapterList")
|
|
|
|
m.chapterMenu = m.top.findNode("chapterMenu")
|
|
|
|
m.chapterContent = m.top.findNode("chapterContent")
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd = m.top.findNode("osd")
|
|
|
|
m.osd.observeField("action", "onOSDAction")
|
2023-11-10 23:51:00 +00:00
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
m.playbackTimer = m.top.findNode("playbackTimer")
|
|
|
|
m.bufferCheckTimer = m.top.findNode("bufferCheckTimer")
|
|
|
|
m.top.observeField("state", "onState")
|
|
|
|
m.top.observeField("content", "onContentChange")
|
|
|
|
m.top.observeField("selectedSubtitle", "onSubtitleChange")
|
2024-01-27 17:10:18 +00:00
|
|
|
m.top.observeField("audioIndex", "onAudioIndexChange")
|
2023-10-06 03:18:36 +00:00
|
|
|
|
|
|
|
' Custom Caption Function
|
|
|
|
m.top.observeField("allowCaptions", "onAllowCaptionsChange")
|
|
|
|
|
|
|
|
m.playbackTimer.observeField("fire", "ReportPlayback")
|
|
|
|
m.bufferPercentage = 0 ' Track whether content is being loaded
|
|
|
|
m.playReported = false
|
|
|
|
m.top.transcodeReasons = []
|
|
|
|
m.bufferCheckTimer.duration = 30
|
|
|
|
|
|
|
|
if m.global.session.user.settings["ui.design.hideclock"] = true
|
|
|
|
clockNode = findNodeBySubtype(m.top, "clock")
|
|
|
|
if clockNode[0] <> invalid then clockNode[0].parent.removeChild(clockNode[0].node)
|
|
|
|
end if
|
|
|
|
|
|
|
|
'Play Next Episode button
|
|
|
|
m.nextEpisodeButton = m.top.findNode("nextEpisode")
|
|
|
|
m.nextEpisodeButton.text = tr("Next Episode")
|
|
|
|
m.nextEpisodeButton.setFocus(false)
|
2023-11-01 00:01:18 +00:00
|
|
|
m.nextupbuttonseconds = m.global.session.user.settings["playback.nextupbuttonseconds"].ToInt()
|
2023-10-06 03:18:36 +00:00
|
|
|
|
|
|
|
m.showNextEpisodeButtonAnimation = m.top.findNode("showNextEpisodeButton")
|
|
|
|
m.hideNextEpisodeButtonAnimation = m.top.findNode("hideNextEpisodeButton")
|
|
|
|
|
|
|
|
m.checkedForNextEpisode = false
|
|
|
|
m.getNextEpisodeTask = createObject("roSGNode", "GetNextEpisodeTask")
|
|
|
|
m.getNextEpisodeTask.observeField("nextEpisodeData", "onNextEpisodeDataLoaded")
|
|
|
|
|
|
|
|
m.top.retrievingBar.filledBarBlendColor = m.global.constants.colors.blue
|
|
|
|
m.top.bufferingBar.filledBarBlendColor = m.global.constants.colors.blue
|
|
|
|
m.top.trickPlayBar.filledBarBlendColor = m.global.constants.colors.blue
|
|
|
|
end sub
|
|
|
|
|
2023-11-10 23:51:00 +00:00
|
|
|
' handleChapterSkipAction: Handles user command to skip chapters in playing video
|
|
|
|
'
|
|
|
|
sub handleChapterSkipAction(action as string)
|
|
|
|
if not isValidAndNotEmpty(m.chapters) then return
|
|
|
|
|
|
|
|
currentChapter = getCurrentChapterIndex()
|
|
|
|
|
|
|
|
if action = "chapternext"
|
|
|
|
gotoChapter = currentChapter + 1
|
|
|
|
' If there is no next chapter, exit
|
|
|
|
if gotoChapter > m.chapters.count() - 1 then return
|
|
|
|
|
|
|
|
m.top.seek = m.chapters[gotoChapter].StartPositionTicks / 10000000#
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
if action = "chapterback"
|
|
|
|
gotoChapter = currentChapter - 1
|
|
|
|
' If there is no previous chapter, restart current chapter
|
|
|
|
if gotoChapter < 0 then gotoChapter = 0
|
|
|
|
|
|
|
|
m.top.seek = m.chapters[gotoChapter].StartPositionTicks / 10000000#
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' handleHideAction: Handles action to hide OSD menu
|
2023-11-10 23:51:00 +00:00
|
|
|
'
|
|
|
|
' @param {boolean} resume - controls whether or not to resume video playback when sub is called
|
|
|
|
'
|
|
|
|
sub handleHideAction(resume as boolean)
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.visible = false
|
2023-11-10 23:51:00 +00:00
|
|
|
m.chapterList.visible = false
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.showChapterList = false
|
2023-11-10 23:51:00 +00:00
|
|
|
m.chapterList.setFocus(false)
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.hasFocus = false
|
|
|
|
m.osd.setFocus(false)
|
2023-11-10 23:51:00 +00:00
|
|
|
m.top.setFocus(true)
|
|
|
|
if resume
|
|
|
|
m.top.control = "resume"
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' handleChapterListAction: Handles action to show chapter list
|
|
|
|
'
|
|
|
|
sub handleChapterListAction()
|
2023-11-16 17:46:25 +00:00
|
|
|
m.chapterList.visible = m.osd.showChapterList
|
2023-11-10 23:51:00 +00:00
|
|
|
|
|
|
|
if not m.chapterList.visible then return
|
|
|
|
|
|
|
|
m.chapterMenu.jumpToItem = getCurrentChapterIndex()
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.hasFocus = false
|
|
|
|
m.osd.setFocus(false)
|
2023-11-10 23:51:00 +00:00
|
|
|
m.chapterMenu.setFocus(true)
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' getCurrentChapterIndex: Finds current chapter index
|
|
|
|
'
|
|
|
|
' @return {integer} indicating index of current chapter within chapter data or 0 if chapter lookup fails
|
|
|
|
'
|
|
|
|
function getCurrentChapterIndex() as integer
|
|
|
|
if not isValidAndNotEmpty(m.chapters) then return 0
|
|
|
|
|
|
|
|
' Give a 15 second buffer to compensate for user expectation and roku video position inaccuracy
|
|
|
|
' Web client uses 10 seconds, but this wasn't enough for Roku in testing
|
|
|
|
currentPosition = m.top.position + 15
|
|
|
|
currentChapter = 0
|
|
|
|
|
|
|
|
for i = m.chapters.count() - 1 to 0 step -1
|
|
|
|
if currentPosition >= (m.chapters[i].StartPositionTicks / 10000000#)
|
|
|
|
currentChapter = i
|
|
|
|
exit for
|
|
|
|
end if
|
|
|
|
end for
|
|
|
|
|
|
|
|
return currentChapter
|
|
|
|
end function
|
|
|
|
|
|
|
|
' handleVideoPlayPauseAction: Handles action to either play or pause the video content
|
|
|
|
'
|
|
|
|
sub handleVideoPlayPauseAction()
|
|
|
|
' If video is paused, resume it
|
|
|
|
if m.top.state = "paused"
|
|
|
|
handleHideAction(true)
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
' Pause video
|
|
|
|
m.top.control = "pause"
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' handleShowSubtitleMenuAction: Handles action to show subtitle selection menu
|
|
|
|
'
|
|
|
|
sub handleShowSubtitleMenuAction()
|
|
|
|
m.top.selectSubtitlePressed = true
|
|
|
|
end sub
|
|
|
|
|
2024-01-27 17:10:18 +00:00
|
|
|
' handleShowAudioMenuAction: Handles action to show audio selection menu
|
|
|
|
'
|
|
|
|
sub handleShowAudioMenuAction()
|
|
|
|
m.top.selectAudioPressed = true
|
|
|
|
end sub
|
|
|
|
|
2023-11-10 23:51:00 +00:00
|
|
|
' handleShowVideoInfoPopupAction: Handles action to show video info popup
|
|
|
|
'
|
|
|
|
sub handleShowVideoInfoPopupAction()
|
|
|
|
m.top.selectPlaybackInfoPressed = true
|
|
|
|
end sub
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' onOSDAction: Process action events from OSD to their respective handlers
|
2023-11-10 23:51:00 +00:00
|
|
|
'
|
2023-11-16 17:46:25 +00:00
|
|
|
sub onOSDAction()
|
|
|
|
action = LCase(m.osd.action)
|
2023-11-10 23:51:00 +00:00
|
|
|
|
|
|
|
if action = "hide"
|
|
|
|
handleHideAction(false)
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
if action = "play"
|
|
|
|
handleHideAction(true)
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
if action = "chapterback" or action = "chapternext"
|
|
|
|
handleChapterSkipAction(action)
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
if action = "chapterlist"
|
|
|
|
handleChapterListAction()
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
if action = "videoplaypause"
|
|
|
|
handleVideoPlayPauseAction()
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
if action = "showsubtitlemenu"
|
|
|
|
handleShowSubtitleMenuAction()
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
2024-01-27 17:10:18 +00:00
|
|
|
if action = "showaudiomenu"
|
|
|
|
handleShowAudioMenuAction()
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
2023-11-10 23:51:00 +00:00
|
|
|
if action = "showvideoinfopopup"
|
|
|
|
handleShowVideoInfoPopupAction()
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' Only setup caption items if captions are allowed
|
2023-10-06 03:18:36 +00:00
|
|
|
sub onAllowCaptionsChange()
|
|
|
|
if not m.top.allowCaptions then return
|
|
|
|
|
|
|
|
m.captionGroup = m.top.findNode("captionGroup")
|
|
|
|
m.captionGroup.createchildren(9, "LayoutGroup")
|
|
|
|
m.captionTask = createObject("roSGNode", "captionTask")
|
|
|
|
m.captionTask.observeField("currentCaption", "updateCaption")
|
|
|
|
m.captionTask.observeField("useThis", "checkCaptionMode")
|
|
|
|
m.top.observeField("subtitleTrack", "loadCaption")
|
|
|
|
m.top.observeField("globalCaptionMode", "toggleCaption")
|
|
|
|
|
|
|
|
if m.global.session.user.settings["playback.subs.custom"]
|
|
|
|
m.top.suppressCaptions = true
|
|
|
|
toggleCaption()
|
|
|
|
else
|
|
|
|
m.top.suppressCaptions = false
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' Set caption url to server subtitle track
|
|
|
|
sub loadCaption()
|
|
|
|
if m.top.suppressCaptions
|
|
|
|
m.captionTask.url = m.top.subtitleTrack
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' Toggles visibility of custom subtitles and sets captionTask's player state
|
|
|
|
sub toggleCaption()
|
|
|
|
m.captionTask.playerState = m.top.state + m.top.globalCaptionMode
|
|
|
|
if LCase(m.top.globalCaptionMode) = "on"
|
|
|
|
m.captionTask.playerState = m.top.state + m.top.globalCaptionMode + "w"
|
|
|
|
m.captionGroup.visible = true
|
|
|
|
else
|
|
|
|
m.captionGroup.visible = false
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' Removes old subtitle lines and adds new subtitle lines
|
|
|
|
sub updateCaption()
|
|
|
|
m.captionGroup.removeChildrenIndex(m.captionGroup.getChildCount(), 0)
|
|
|
|
m.captionGroup.appendChildren(m.captionTask.currentCaption)
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' Event handler for when selectedSubtitle changes
|
|
|
|
sub onSubtitleChange()
|
2024-01-14 02:05:18 +00:00
|
|
|
switchWithoutRefresh = true
|
|
|
|
|
|
|
|
if m.top.SelectedSubtitle <> -1
|
|
|
|
' If the global caption mode is off, then Roku can't display the subtitles natively and needs a video stop/start
|
|
|
|
if LCase(m.top.globalCaptionMode) <> "on" then switchWithoutRefresh = false
|
|
|
|
end if
|
|
|
|
|
|
|
|
' If previous sustitle was encoded, then we need to a video stop/start to change subtitle content
|
|
|
|
if m.top.previousSubtitleWasEncoded then switchWithoutRefresh = false
|
|
|
|
|
|
|
|
if switchWithoutRefresh then return
|
2024-01-06 01:31:17 +00:00
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
' Save the current video position
|
|
|
|
m.global.queueManager.callFunc("setTopStartingPoint", int(m.top.position) * 10000000&)
|
|
|
|
|
|
|
|
m.top.control = "stop"
|
|
|
|
|
|
|
|
m.LoadMetaDataTask.selectedSubtitleIndex = m.top.SelectedSubtitle
|
2024-01-27 17:10:18 +00:00
|
|
|
m.LoadMetaDataTask.selectedAudioStreamIndex = m.top.audioIndex
|
|
|
|
m.LoadMetaDataTask.itemId = m.currentItem.id
|
|
|
|
m.LoadMetaDataTask.observeField("content", "onVideoContentLoaded")
|
|
|
|
m.LoadMetaDataTask.control = "RUN"
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' Event handler for when audioIndex changes
|
|
|
|
sub onAudioIndexChange()
|
|
|
|
' Skip initial audio index setting
|
|
|
|
if m.top.position = 0 then return
|
|
|
|
|
|
|
|
' Save the current video position
|
|
|
|
m.global.queueManager.callFunc("setTopStartingPoint", int(m.top.position) * 10000000&)
|
|
|
|
|
|
|
|
m.top.control = "stop"
|
|
|
|
|
|
|
|
m.LoadMetaDataTask.selectedSubtitleIndex = m.top.SelectedSubtitle
|
|
|
|
m.LoadMetaDataTask.selectedAudioStreamIndex = m.top.audioIndex
|
2023-10-06 03:18:36 +00:00
|
|
|
m.LoadMetaDataTask.itemId = m.currentItem.id
|
|
|
|
m.LoadMetaDataTask.observeField("content", "onVideoContentLoaded")
|
|
|
|
m.LoadMetaDataTask.control = "RUN"
|
|
|
|
end sub
|
|
|
|
|
|
|
|
sub onPlaybackErrorDialogClosed(msg)
|
|
|
|
sourceNode = msg.getRoSGNode()
|
|
|
|
sourceNode.unobserveField("buttonSelected")
|
|
|
|
sourceNode.unobserveField("wasClosed")
|
|
|
|
|
|
|
|
m.global.sceneManager.callFunc("popScene")
|
|
|
|
end sub
|
|
|
|
|
|
|
|
sub onPlaybackErrorButtonSelected(msg)
|
|
|
|
sourceNode = msg.getRoSGNode()
|
|
|
|
sourceNode.close = true
|
|
|
|
end sub
|
|
|
|
|
|
|
|
sub showPlaybackErrorDialog(errorMessage as string)
|
|
|
|
dialog = createObject("roSGNode", "Dialog")
|
|
|
|
dialog.title = tr("Error During Playback")
|
|
|
|
dialog.buttons = [tr("OK")]
|
|
|
|
dialog.message = errorMessage
|
|
|
|
dialog.observeField("buttonSelected", "onPlaybackErrorButtonSelected")
|
|
|
|
dialog.observeField("wasClosed", "onPlaybackErrorDialogClosed")
|
|
|
|
m.top.getScene().dialog = dialog
|
|
|
|
end sub
|
|
|
|
|
|
|
|
sub onVideoContentLoaded()
|
|
|
|
m.LoadMetaDataTask.unobserveField("content")
|
|
|
|
m.LoadMetaDataTask.control = "STOP"
|
|
|
|
|
|
|
|
videoContent = m.LoadMetaDataTask.content
|
|
|
|
m.LoadMetaDataTask.content = []
|
|
|
|
|
2023-11-30 01:22:26 +00:00
|
|
|
stopLoadingSpinner()
|
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
' If we have nothing to play, return to previous screen
|
|
|
|
if not isValid(videoContent)
|
|
|
|
showPlaybackErrorDialog(tr("There was an error retrieving the data for this item from the server."))
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
if not isValid(videoContent[0])
|
|
|
|
showPlaybackErrorDialog(tr("There was an error retrieving the data for this item from the server."))
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
m.top.content = videoContent[0].content
|
|
|
|
m.top.PlaySessionId = videoContent[0].PlaySessionId
|
|
|
|
m.top.videoId = videoContent[0].id
|
|
|
|
m.top.container = videoContent[0].container
|
|
|
|
m.top.mediaSourceId = videoContent[0].mediaSourceId
|
|
|
|
m.top.fullSubtitleData = videoContent[0].fullSubtitleData
|
2024-01-27 17:10:18 +00:00
|
|
|
m.top.fullAudioData = videoContent[0].fullAudioData
|
2023-10-06 03:18:36 +00:00
|
|
|
m.top.audioIndex = videoContent[0].audioIndex
|
|
|
|
m.top.transcodeParams = videoContent[0].transcodeparams
|
2023-11-10 23:51:00 +00:00
|
|
|
m.chapters = videoContent[0].chapters
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.itemTitleText = m.top.content.title
|
2023-11-10 23:51:00 +00:00
|
|
|
|
|
|
|
populateChapterMenu()
|
2023-10-06 03:18:36 +00:00
|
|
|
|
|
|
|
if m.LoadMetaDataTask.isIntro
|
|
|
|
' Disable trackplay bar for intro videos
|
|
|
|
m.top.enableTrickPlay = false
|
|
|
|
else
|
|
|
|
' Allow custom captions for non intro videos
|
|
|
|
m.top.allowCaptions = true
|
|
|
|
end if
|
|
|
|
|
2024-01-06 01:31:17 +00:00
|
|
|
' Allow default subtitles
|
|
|
|
m.top.unobserveField("selectedSubtitle")
|
|
|
|
|
2024-01-14 02:05:18 +00:00
|
|
|
' Set subtitleTrack property if subs are natively supported by Roku
|
2024-01-06 01:31:17 +00:00
|
|
|
selectedSubtitle = invalid
|
|
|
|
for each subtitle in m.top.fullSubtitleData
|
|
|
|
if subtitle.Index = videoContent[0].selectedSubtitle
|
|
|
|
selectedSubtitle = subtitle
|
|
|
|
exit for
|
|
|
|
end if
|
|
|
|
end for
|
|
|
|
|
|
|
|
if isValid(selectedSubtitle)
|
|
|
|
availableSubtitleTrackIndex = availSubtitleTrackIdx(selectedSubtitle.Track.TrackName)
|
|
|
|
if availableSubtitleTrackIndex <> -1
|
2024-01-14 02:05:18 +00:00
|
|
|
if not selectedSubtitle.IsEncoded
|
|
|
|
m.top.globalCaptionMode = "On"
|
|
|
|
m.top.subtitleTrack = m.top.availableSubtitleTracks[availableSubtitleTrackIndex].TrackName
|
|
|
|
end if
|
2024-01-06 01:31:17 +00:00
|
|
|
end if
|
|
|
|
end if
|
|
|
|
|
|
|
|
m.top.selectedSubtitle = videoContent[0].selectedSubtitle
|
|
|
|
|
|
|
|
m.top.observeField("selectedSubtitle", "onSubtitleChange")
|
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
if isValid(m.top.audioIndex)
|
|
|
|
m.top.audioTrack = (m.top.audioIndex + 1).toStr()
|
|
|
|
else
|
|
|
|
m.top.audioTrack = "2"
|
|
|
|
end if
|
|
|
|
|
|
|
|
m.top.setFocus(true)
|
|
|
|
m.top.control = "play"
|
|
|
|
end sub
|
|
|
|
|
2023-11-10 23:51:00 +00:00
|
|
|
' populateChapterMenu: ' Parse chapter data from API and appeand to chapter list menu
|
|
|
|
'
|
|
|
|
sub populateChapterMenu()
|
|
|
|
' Clear any existing chapter list data
|
|
|
|
m.chapterContent.clear()
|
|
|
|
|
|
|
|
if not isValidAndNotEmpty(m.chapters)
|
|
|
|
chapterItem = CreateObject("roSGNode", "ContentNode")
|
|
|
|
chapterItem.title = tr("No Chapter Data Found")
|
|
|
|
chapterItem.playstart = m.playbackEnum.null
|
|
|
|
m.chapterContent.appendChild(chapterItem)
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
for each chapter in m.chapters
|
|
|
|
chapterItem = CreateObject("roSGNode", "ContentNode")
|
|
|
|
chapterItem.title = chapter.Name
|
|
|
|
chapterItem.playstart = chapter.StartPositionTicks / 10000000#
|
|
|
|
m.chapterContent.appendChild(chapterItem)
|
|
|
|
end for
|
|
|
|
end sub
|
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
' Event handler for when video content field changes
|
|
|
|
sub onContentChange()
|
|
|
|
if not isValid(m.top.content) then return
|
|
|
|
|
|
|
|
m.top.observeField("position", "onPositionChanged")
|
|
|
|
end sub
|
|
|
|
|
|
|
|
sub onNextEpisodeDataLoaded()
|
|
|
|
m.checkedForNextEpisode = true
|
|
|
|
|
|
|
|
m.top.observeField("position", "onPositionChanged")
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' Runs Next Episode button animation and sets focus to button
|
|
|
|
sub showNextEpisodeButton()
|
2023-11-16 17:46:25 +00:00
|
|
|
if m.osd.visible then return
|
2023-11-01 00:01:18 +00:00
|
|
|
if m.top.content.contenttype <> 4 then return ' only display when content is type "Episode"
|
|
|
|
if m.nextupbuttonseconds = 0 then return ' is the button disabled?
|
|
|
|
|
|
|
|
if m.nextEpisodeButton.opacity = 0 and m.global.session.user.configuration.EnableNextEpisodeAutoPlay
|
|
|
|
m.nextEpisodeButton.visible = true
|
2023-10-06 03:18:36 +00:00
|
|
|
m.showNextEpisodeButtonAnimation.control = "start"
|
|
|
|
m.nextEpisodeButton.setFocus(true)
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
'Update count down text
|
|
|
|
sub updateCount()
|
2023-11-01 00:01:18 +00:00
|
|
|
nextEpisodeCountdown = Int(m.top.duration - m.top.position)
|
|
|
|
if nextEpisodeCountdown < 0
|
|
|
|
nextEpisodeCountdown = 0
|
|
|
|
end if
|
|
|
|
m.nextEpisodeButton.text = tr("Next Episode") + " " + nextEpisodeCountdown.toStr()
|
2023-10-06 03:18:36 +00:00
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' Runs hide Next Episode button animation and sets focus back to video
|
|
|
|
sub hideNextEpisodeButton()
|
|
|
|
m.hideNextEpisodeButtonAnimation.control = "start"
|
|
|
|
m.nextEpisodeButton.setFocus(false)
|
|
|
|
m.top.setFocus(true)
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' Checks if we need to display the Next Episode button
|
|
|
|
sub checkTimeToDisplayNextEpisode()
|
2023-11-01 00:01:18 +00:00
|
|
|
if m.top.content.contenttype <> 4 then return ' only display when content is type "Episode"
|
|
|
|
if m.nextupbuttonseconds = 0 then return ' is the button disabled?
|
|
|
|
|
2023-11-20 03:09:14 +00:00
|
|
|
' Don't show Next Episode button if trickPlayBar is visible
|
|
|
|
if m.top.trickPlayBar.visible then return
|
|
|
|
|
2023-11-01 00:01:18 +00:00
|
|
|
if isValid(m.top.duration) and isValid(m.top.position)
|
|
|
|
nextEpisodeCountdown = Int(m.top.duration - m.top.position)
|
|
|
|
|
|
|
|
if nextEpisodeCountdown < 0 and m.nextEpisodeButton.opacity = 0.9
|
|
|
|
hideNextEpisodeButton()
|
|
|
|
return
|
|
|
|
else if nextEpisodeCountdown > 1 and int(m.top.position) >= (m.top.duration - m.nextupbuttonseconds - 1)
|
|
|
|
updateCount()
|
|
|
|
if m.nextEpisodeButton.opacity = 0
|
|
|
|
showNextEpisodeButton()
|
|
|
|
end if
|
|
|
|
return
|
|
|
|
end if
|
2023-10-06 03:18:36 +00:00
|
|
|
end if
|
|
|
|
|
|
|
|
if m.nextEpisodeButton.visible or m.nextEpisodeButton.hasFocus()
|
|
|
|
m.nextEpisodeButton.visible = false
|
|
|
|
m.nextEpisodeButton.setFocus(false)
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
' When Video Player state changes
|
|
|
|
sub onPositionChanged()
|
2023-11-12 18:13:07 +00:00
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' Pass video position data into OSD
|
|
|
|
m.osd.progressPercentage = m.top.position / m.top.duration
|
|
|
|
m.osd.positionTime = m.top.position
|
|
|
|
m.osd.remainingPositionTime = m.top.duration - m.top.position
|
2023-11-12 18:13:07 +00:00
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
if isValid(m.captionTask)
|
|
|
|
m.captionTask.currentPos = Int(m.top.position * 1000)
|
|
|
|
end if
|
|
|
|
|
|
|
|
' Check if dialog is open
|
|
|
|
m.dialog = m.top.getScene().findNode("dialogBackground")
|
|
|
|
if not isValid(m.dialog)
|
|
|
|
' Do not show Next Episode button for intro videos
|
|
|
|
if not m.LoadMetaDataTask.isIntro
|
|
|
|
checkTimeToDisplayNextEpisode()
|
|
|
|
end if
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' When Video Player state changes
|
|
|
|
sub onState(msg)
|
|
|
|
if isValid(m.captionTask)
|
|
|
|
m.captionTask.playerState = m.top.state + m.top.globalCaptionMode
|
|
|
|
end if
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' Pass video state into OSD
|
|
|
|
m.osd.playbackState = m.top.state
|
2023-11-10 23:51:00 +00:00
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
' When buffering, start timer to monitor buffering process
|
|
|
|
if m.top.state = "buffering" and m.bufferCheckTimer <> invalid
|
|
|
|
|
|
|
|
' start timer
|
|
|
|
m.bufferCheckTimer.control = "start"
|
|
|
|
m.bufferCheckTimer.ObserveField("fire", "bufferCheck")
|
|
|
|
else if m.top.state = "error"
|
|
|
|
if not m.playReported and m.top.transcodeAvailable
|
|
|
|
m.top.retryWithTranscoding = true ' If playback was not reported, retry with transcoding
|
|
|
|
else
|
|
|
|
' If an error was encountered, Display dialog
|
|
|
|
showPlaybackErrorDialog(tr("Error During Playback"))
|
|
|
|
end if
|
|
|
|
|
|
|
|
' Stop playback and exit player
|
|
|
|
m.top.control = "stop"
|
|
|
|
m.top.backPressed = true
|
|
|
|
else if m.top.state = "playing"
|
|
|
|
|
|
|
|
' Check if next episde is available
|
|
|
|
if isValid(m.top.showID)
|
|
|
|
if m.top.showID <> "" and not m.checkedForNextEpisode and m.top.content.contenttype = 4
|
|
|
|
m.getNextEpisodeTask.showID = m.top.showID
|
|
|
|
m.getNextEpisodeTask.videoID = m.top.id
|
|
|
|
m.getNextEpisodeTask.control = "RUN"
|
|
|
|
end if
|
|
|
|
end if
|
|
|
|
|
|
|
|
if m.playReported = false
|
|
|
|
ReportPlayback("start")
|
|
|
|
m.playReported = true
|
|
|
|
else
|
|
|
|
ReportPlayback()
|
|
|
|
end if
|
|
|
|
m.playbackTimer.control = "start"
|
|
|
|
else if m.top.state = "paused"
|
|
|
|
m.playbackTimer.control = "stop"
|
|
|
|
ReportPlayback()
|
|
|
|
else if m.top.state = "stopped"
|
|
|
|
m.playbackTimer.control = "stop"
|
|
|
|
ReportPlayback("stop")
|
|
|
|
m.playReported = false
|
|
|
|
end if
|
|
|
|
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' Report playback to server
|
|
|
|
sub ReportPlayback(state = "update" as string)
|
|
|
|
|
|
|
|
if m.top.position = invalid then return
|
|
|
|
|
|
|
|
params = {
|
|
|
|
"ItemId": m.top.id,
|
|
|
|
"PlaySessionId": m.top.PlaySessionId,
|
|
|
|
"PositionTicks": int(m.top.position) * 10000000&, 'Ensure a LongInteger is used
|
|
|
|
"IsPaused": (m.top.state = "paused")
|
|
|
|
}
|
|
|
|
if m.top.content.live
|
|
|
|
params.append({
|
|
|
|
"MediaSourceId": m.top.transcodeParams.MediaSourceId,
|
|
|
|
"LiveStreamId": m.top.transcodeParams.LiveStreamId
|
|
|
|
})
|
|
|
|
m.bufferCheckTimer.duration = 30
|
|
|
|
end if
|
|
|
|
|
|
|
|
' Report playstate via worker task
|
|
|
|
playstateTask = m.global.playstateTask
|
|
|
|
playstateTask.setFields({ status: state, params: params })
|
|
|
|
playstateTask.control = "RUN"
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' Check the the buffering has not hung
|
|
|
|
sub bufferCheck(msg)
|
|
|
|
|
|
|
|
if m.top.state <> "buffering"
|
|
|
|
' If video is not buffering, stop timer
|
|
|
|
m.bufferCheckTimer.control = "stop"
|
|
|
|
m.bufferCheckTimer.unobserveField("fire")
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
if m.top.bufferingStatus <> invalid
|
|
|
|
|
|
|
|
' Check that the buffering percentage is increasing
|
|
|
|
if m.top.bufferingStatus["percentage"] > m.bufferPercentage
|
|
|
|
m.bufferPercentage = m.top.bufferingStatus["percentage"]
|
|
|
|
else if m.top.content.live = true
|
|
|
|
m.top.callFunc("refresh")
|
|
|
|
else
|
|
|
|
' If buffering has stopped Display dialog
|
|
|
|
showPlaybackErrorDialog(tr("There was an error retrieving the data for this item from the server."))
|
|
|
|
|
|
|
|
' Stop playback and exit player
|
|
|
|
m.top.control = "stop"
|
|
|
|
m.top.backPressed = true
|
|
|
|
end if
|
|
|
|
end if
|
|
|
|
|
|
|
|
end sub
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' stateAllowsOSD: Check if current video state allows showing the OSD
|
2023-11-16 02:55:40 +00:00
|
|
|
'
|
2023-11-16 17:46:25 +00:00
|
|
|
' @return {boolean} indicating if video state allows the OSD to show
|
|
|
|
function stateAllowsOSD() as boolean
|
2023-11-16 02:55:40 +00:00
|
|
|
validStates = ["playing", "paused", "stopped"]
|
|
|
|
return inArray(validStates, m.top.state)
|
|
|
|
end function
|
2024-01-06 01:31:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
' availSubtitleTrackIdx: Returns Roku's index for requested subtitle track
|
|
|
|
'
|
|
|
|
' @param {string} tracknameToFind - TrackName for subtitle we're looking to match
|
|
|
|
' @return {integer} indicating Roku's index for requested subtitle track. Returns -1 if not found
|
|
|
|
function availSubtitleTrackIdx(tracknameToFind as string) as integer
|
|
|
|
idx = 0
|
|
|
|
for each availTrack in m.top.availableSubtitleTracks
|
|
|
|
' The TrackName must contain the URL we supplied originally, though
|
|
|
|
' Roku mangles the name a bit, so we check if the URL is a substring, rather
|
|
|
|
' than strict equality
|
|
|
|
if Instr(1, availTrack.TrackName, tracknameToFind)
|
|
|
|
return idx
|
|
|
|
end if
|
|
|
|
idx = idx + 1
|
|
|
|
end for
|
|
|
|
return -1
|
|
|
|
end function
|
2023-11-16 02:55:40 +00:00
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
function onKeyEvent(key as string, press as boolean) as boolean
|
|
|
|
|
2023-11-10 23:51:00 +00:00
|
|
|
' Keypress handler while user is inside the chapter menu
|
|
|
|
if m.chapterMenu.hasFocus()
|
|
|
|
if not press then return false
|
|
|
|
|
|
|
|
if key = "OK"
|
|
|
|
focusedChapter = m.chapterMenu.itemFocused
|
|
|
|
selectedChapter = m.chapterMenu.content.getChild(focusedChapter)
|
|
|
|
seekTime = selectedChapter.playstart
|
|
|
|
|
|
|
|
' Don't seek if user clicked on No Chapter Data
|
|
|
|
if seekTime = m.playbackEnum.null then return true
|
|
|
|
|
|
|
|
m.top.seek = seekTime
|
|
|
|
return true
|
|
|
|
end if
|
|
|
|
|
|
|
|
if key = "back" or key = "replay"
|
|
|
|
m.chapterList.visible = false
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.showChapterList = false
|
2023-11-10 23:51:00 +00:00
|
|
|
m.chapterMenu.setFocus(false)
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.hasFocus = true
|
|
|
|
m.osd.setFocus(true)
|
2023-11-10 23:51:00 +00:00
|
|
|
return true
|
|
|
|
end if
|
|
|
|
|
|
|
|
if key = "play"
|
|
|
|
handleVideoPlayPauseAction()
|
|
|
|
end if
|
|
|
|
|
|
|
|
return true
|
|
|
|
end if
|
|
|
|
|
2023-10-06 03:18:36 +00:00
|
|
|
if key = "OK" and m.nextEpisodeButton.hasfocus() and not m.top.trickPlayBar.visible
|
|
|
|
m.top.control = "stop"
|
|
|
|
m.top.state = "finished"
|
|
|
|
hideNextEpisodeButton()
|
|
|
|
return true
|
|
|
|
else
|
|
|
|
'Hide Next Episode Button
|
2023-11-01 00:01:18 +00:00
|
|
|
if m.nextEpisodeButton.opacity > 0 or m.nextEpisodeButton.hasFocus()
|
|
|
|
m.nextEpisodeButton.opacity = 0
|
2023-10-06 03:18:36 +00:00
|
|
|
m.nextEpisodeButton.setFocus(false)
|
|
|
|
m.top.setFocus(true)
|
|
|
|
end if
|
|
|
|
end if
|
|
|
|
|
|
|
|
if not press then return false
|
|
|
|
|
2023-11-12 18:13:07 +00:00
|
|
|
if key = "down" and not m.top.trickPlayBar.visible
|
2023-10-06 03:18:36 +00:00
|
|
|
if not m.LoadMetaDataTask.isIntro
|
2023-11-16 02:55:40 +00:00
|
|
|
' Don't allow user to open menu prior to video loading
|
2023-11-16 17:46:25 +00:00
|
|
|
if not stateAllowsOSD() then return true
|
2023-11-16 02:55:40 +00:00
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.visible = true
|
|
|
|
m.osd.hasFocus = true
|
|
|
|
m.osd.setFocus(true)
|
2023-10-06 03:18:36 +00:00
|
|
|
return true
|
|
|
|
end if
|
2023-11-10 23:51:00 +00:00
|
|
|
|
2023-11-12 18:13:07 +00:00
|
|
|
else if key = "up" and not m.top.trickPlayBar.visible
|
2023-10-06 03:18:36 +00:00
|
|
|
if not m.LoadMetaDataTask.isIntro
|
2023-11-16 02:55:40 +00:00
|
|
|
' Don't allow user to open menu prior to video loading
|
2023-11-16 17:46:25 +00:00
|
|
|
if not stateAllowsOSD() then return true
|
2023-11-16 02:55:40 +00:00
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.visible = true
|
|
|
|
m.osd.hasFocus = true
|
|
|
|
m.osd.setFocus(true)
|
2023-11-10 23:51:00 +00:00
|
|
|
return true
|
|
|
|
end if
|
|
|
|
|
|
|
|
else if key = "OK" and not m.top.trickPlayBar.visible
|
|
|
|
if not m.LoadMetaDataTask.isIntro
|
2023-11-16 02:55:40 +00:00
|
|
|
' Don't allow user to open menu prior to video loading
|
2023-11-16 17:46:25 +00:00
|
|
|
if not stateAllowsOSD() then return true
|
2023-11-16 02:55:40 +00:00
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' Show OSD, but don't pause video
|
|
|
|
m.osd.visible = true
|
|
|
|
m.osd.hasFocus = true
|
|
|
|
m.osd.setFocus(true)
|
2023-10-06 03:18:36 +00:00
|
|
|
return true
|
|
|
|
end if
|
2023-11-10 23:51:00 +00:00
|
|
|
|
|
|
|
return false
|
|
|
|
end if
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' Disable OSD for intro videos
|
2023-11-10 23:51:00 +00:00
|
|
|
if not m.LoadMetaDataTask.isIntro
|
|
|
|
if key = "play" and not m.top.trickPlayBar.visible
|
2023-11-16 02:55:40 +00:00
|
|
|
|
|
|
|
' Don't allow user to open menu prior to video loading
|
2023-11-16 17:46:25 +00:00
|
|
|
if not stateAllowsOSD() then return true
|
2023-11-16 02:55:40 +00:00
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' If video is paused, resume it and don't show OSD
|
2023-11-10 23:51:00 +00:00
|
|
|
if m.top.state = "paused"
|
|
|
|
m.top.control = "resume"
|
|
|
|
return true
|
|
|
|
end if
|
|
|
|
|
2023-11-16 17:46:25 +00:00
|
|
|
' Pause video and show OSD
|
2023-10-06 03:18:36 +00:00
|
|
|
m.top.control = "pause"
|
2023-11-16 17:46:25 +00:00
|
|
|
m.osd.visible = true
|
|
|
|
m.osd.hasFocus = true
|
|
|
|
m.osd.setFocus(true)
|
2023-11-10 23:51:00 +00:00
|
|
|
return true
|
2023-10-06 03:18:36 +00:00
|
|
|
end if
|
|
|
|
end if
|
|
|
|
|
|
|
|
if key = "back"
|
|
|
|
m.top.control = "stop"
|
|
|
|
end if
|
|
|
|
|
|
|
|
return false
|
|
|
|
end function
|
2024-02-09 14:58:42 +00:00
|
|
|
</code></pre></article></section><footer class="footer" id="PeOAagUepe"><div class="wrapper"><span class="jsdoc-message">Automatically generated using <a href="https://github.com/jsdoc/jsdoc" target="_blank">JSDoc</a> and the <a href="https://github.com/ankitskvmdam/clean-jsdoc-theme" target="_blank">clean-jsdoc-theme</a>.</span></div></footer></div></div></div><div class="search-container" id="PkfLWpAbet" style="display:none"><div class="wrapper" id="iCxFxjkHbP"><button class="icon-button search-close-button" id="VjLlGakifb" aria-label="close search"><svg><use xlink:href="#close-icon"></use></svg></button><div class="search-box-c"><svg><use xlink:href="#search-icon"></use></svg> <input type="text" id="vpcKVYIppa" class="search-input" placeholder="Search..." autofocus></div><div class="search-result-c" id="fWwVHRuDuN"><span class="search-result-c-text">Type anything to view search result</span></div></div></div><div class="mobile-menu-icon-container"><button class="icon-button" id="mobile-menu" data-isopen="false" aria-label="menu"><svg><use xlink:href="#menu-icon"></use></svg></button></div><div id="mobile-sidebar" class="mobile-sidebar-container"><div class="mobile-sidebar-wrapper"><a href="/" class="sidebar-title sidebar-title-anchor">jellyfin-roku Code Documentation</a><div class="mobile-nav-links"><div class="external-link navbar-item"><a id="jellyfin-link-mobile" href="https://jellyfin.org/" target="_blank">Jellyfin</a></div><div class="external-link navbar-item"><a id="github-link-mobile" href="https://github.com/jellyfin/jellyfin-roku" target="_blank">GitHub</a></div><div class="external-link navbar-item"><a id="forum-link-mobile" href="https://forum.jellyfin.org/f-roku-development" target="_blank">Forum</a></div><div class="external-link navbar-item"><a id="matrix-link-mobile" href="https://matrix.to/#/#jellyfin-dev-roku:matrix.org" target="_blank">Matrix</a></div></div><div class="mobile-sidebar-items-c"><div class="sidebar-section-title with-arrow" data-isopen="false" id="sidebar-modules"><div>Modules</div><svg><use xlink:href="#down-icon"></use></svg></div><div class="sidebar-section-children-container"><div class="sidebar-section-children"><a href="module-AlbumData.html">AlbumData</a></div><div class="sidebar-section-children"><a href="module-AlbumGrid.html">AlbumGrid</a></div><div class="sidebar-section-children"><a href="module-AlbumTrackList.html">AlbumTrackList</a></div><div class="sidebar-section-children"><a href="module-AlbumView.html">AlbumView</a></div><div class="sidebar-section-children"><a href="module-Alpha.html">Alpha</a></div><div class="sidebar-section-children"><a href="module-ArtistView.html">ArtistView</a></div><div class="sidebar-section-children"><a href="module-AudioPlayer.html">AudioPlayer</a></div><div class="sidebar-section-children"><a href="module-AudioPlayerView.html">AudioPlayerView</a></div><div class="sidebar-section-children"><a href="module-AudioTrackListItem.html">AudioTrackListItem</a></div><div class="sidebar-section-children"><a href="module-ButtonGroupHoriz.html">ButtonGroupHoriz</a></div><div class="sidebar-section-children"><a href="module-ButtonGroupVert.html">ButtonGroupVert</a></div><div class="sidebar-section-children"><a href="module-ChannelData.html">ChannelData</a></div><div class="sidebar-section-children"><a href="module-Clock.html">Clock</a></div><div class="sidebar-section-children"><a href="module-CollectionData.html">CollectionData</a></div><div class="sidebar-section-children"><a href="module-ConfigData.html">ConfigData</a></div><div class="sidebar-section-children"><a href="module-ConfigItem.html">ConfigItem</a></div><div class="sidebar-section-children"><a href="module-ConfigList.html">ConfigList</a></div><div class="sidebar-section-children"><a href="module-ExtrasItem.html">ExtrasItem</a></div><div class="sidebar-section-children"><a href="module-ExtrasRowList.html">ExtrasRowList</a></div><div class="sidebar-section-children"><a href="module-FavoriteItemsTask.html">FavoriteItemsTask</a></div><div class="sidebar-section-children"><a href="module-Folder
|