jf-roku/components/JFVideo.brs

250 lines
7.7 KiB
Plaintext
Raw Normal View History

sub init()
m.playbackTimer = m.top.findNode("playbackTimer")
m.bufferCheckTimer = m.top.findNode("bufferCheckTimer")
m.top.observeField("state", "onState")
m.top.observeField("content", "onContentChange")
m.playbackTimer.observeField("fire", "ReportPlayback")
2021-07-09 20:08:32 +00:00
m.bufferPercentage = 0 ' Track whether content is being loaded
m.playReported = false
2021-06-12 15:03:47 +00:00
m.top.transcodeReasons = []
m.bufferCheckTimer.duration = 30
2021-07-09 20:08:32 +00:00
2022-07-16 02:28:59 +00:00
if get_user_setting("ui.design.hideclock") = "true"
clockNode = findNodeBySubtype(m.top, "clock")
if clockNode[0] <> invalid then clockNode[0].parent.removeChild(clockNode[0].node)
2022-07-16 02:28:59 +00:00
end if
2022-11-03 01:18:16 +00:00
'Play Next Episode button
2022-09-07 04:06:10 +00:00
m.nextEpisodeButton = m.top.findNode("nextEpisode")
m.nextEpisodeButton.text = tr("Next Episode")
m.nextEpisodeButton.setFocus(false)
2022-11-21 01:18:23 +00:00
m.showNextEpisodeButtonAnimation = m.top.findNode("showNextEpisodeButton")
m.hideNextEpisodeButtonAnimation = m.top.findNode("hideNextEpisodeButton")
2022-11-21 21:48:24 +00:00
m.checkedForNextEpisode = false
m.getNextEpisodeTask = createObject("roSGNode", "GetNextEpisodeTask")
m.getNextEpisodeTask.observeField("nextEpisodeData", "onNextEpisodeDataLoaded")
end sub
2022-11-18 02:23:09 +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")
2022-11-18 02:23:09 +00:00
' If video content type is not episode, remove position observer
if m.top.content.contenttype <> 4
m.top.unobserveField("position")
end if
end sub
sub onNextEpisodeDataLoaded()
2022-11-21 21:48:24 +00:00
m.checkedForNextEpisode = true
m.top.observeField("position", "onPositionChanged")
if m.getNextEpisodeTask.nextEpisodeData.Items.count() <> 2
m.top.unobserveField("position")
end if
end sub
2022-09-07 04:06:10 +00:00
'
' Runs Next Episode button animation and sets focus to button
2022-11-21 01:18:23 +00:00
sub showNextEpisodeButton()
2022-11-18 02:30:33 +00:00
if not m.nextEpisodeButton.visible
2022-11-21 01:18:23 +00:00
m.showNextEpisodeButtonAnimation.control = "start"
2022-09-07 04:06:10 +00:00
m.nextEpisodeButton.setFocus(true)
m.nextEpisodeButton.visible = true
2022-09-07 04:06:10 +00:00
end if
end sub
'
'Update count down text
sub updateCount()
m.nextEpisodeButton.text = tr("Next Episode") + " " + Int(m.top.runTime - m.top.position).toStr()
end sub
'
' Runs hide Next Episode button animation and sets focus back to video
2022-11-21 01:18:23 +00:00
sub hideNextEpisodeButton()
m.hideNextEpisodeButtonAnimation.control = "start"
2022-09-07 04:06:10 +00:00
m.nextEpisodeButton.setFocus(false)
m.top.setFocus(true)
end sub
2022-11-21 01:18:23 +00:00
' Checks if we need to display the Next Episode button
sub checkTimeToDisplayNextEpisode()
2022-11-07 04:35:02 +00:00
if int(m.top.position) >= (m.top.runTime - 30)
2022-11-21 01:18:23 +00:00
showNextEpisodeButton()
2022-11-07 04:35:02 +00:00
updateCount()
2022-11-21 01:18:23 +00:00
return
end if
if m.nextEpisodeButton.visible or m.nextEpisodeButton.hasFocus()
m.nextEpisodeButton.visible = false
m.nextEpisodeButton.setFocus(false)
2022-11-07 04:35:02 +00:00
end if
end sub
' When Video Player state changes
sub onPositionChanged()
' Check if dialog is open
2022-11-15 21:14:05 +00:00
m.dialog = m.top.getScene().findNode("dialogBackground")
if not isValid(m.dialog)
2022-11-21 01:18:23 +00:00
checkTimeToDisplayNextEpisode()
2022-11-07 04:35:02 +00:00
end if
end sub
2022-09-07 04:06:10 +00:00
'
' When Video Player state changes
2021-07-09 20:08:32 +00:00
sub onState(msg)
' 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
dialog = createObject("roSGNode", "Dialog")
dialog.title = tr("Error During Playback")
dialog.buttons = [tr("OK")]
dialog.message = tr("An error was encountered while playing this item.")
dialog.observeField("buttonSelected", "dialogClosed")
m.top.getScene().dialog = dialog
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)
2022-11-21 21:48:24 +00:00
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
2022-10-06 16:01:36 +00:00
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 = {
2022-05-30 12:59:24 +00:00
"ItemId": m.top.id,
"PlaySessionId": m.top.PlaySessionId,
"PositionTicks": int(m.top.position) * 10000000&, 'Ensure a LongInteger is used
2022-05-30 13:00:14 +00:00
"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
dialog = createObject("roSGNode", "Dialog")
dialog.title = tr("Error Retrieving Content")
dialog.buttons = [tr("OK")]
dialog.message = tr("There was an error retrieving the data for this item from the server.")
dialog.observeField("buttonSelected", "dialogClosed")
m.top.getScene().dialog = dialog
' Stop playback and exit player
m.top.control = "stop"
m.top.backPressed = true
end if
end if
end sub
'
' Clean up on Dialog Closed
sub dialogClosed(msg)
sourceNode = msg.getRoSGNode()
sourceNode.unobserveField("buttonSelected")
sourceNode.close = true
end sub
function onKeyEvent(key as string, press as boolean) as boolean
2022-11-15 21:14:05 +00:00
if key = "OK" and m.nextEpisodeButton.hasfocus() and m.top.trickPlayMode = "play"
2022-11-03 01:18:16 +00:00
m.top.state = "finished"
2022-11-21 01:18:23 +00:00
hideNextEpisodeButton()
2022-11-03 01:18:16 +00:00
return true
else
'Hide Next Episode Button
2022-11-21 01:18:23 +00:00
if m.nextEpisodeButton.visible or m.nextEpisodeButton.hasFocus()
m.nextEpisodeButton.visible = false
m.nextEpisodeButton.setFocus(false)
m.top.setFocus(true)
end if
end if
2021-07-09 20:08:32 +00:00
if not press then return false
2022-09-07 04:06:10 +00:00
if key = "down"
2021-07-09 20:08:32 +00:00
m.top.selectSubtitlePressed = true
return true
2022-09-06 04:38:37 +00:00
else if key = "up"
m.top.selectPlaybackInfoPressed = true
return true
2021-07-09 20:08:32 +00:00
end if
2021-07-09 20:08:32 +00:00
return false
end function