2019-10-12 21:00:07 +00:00
|
|
|
sub init()
|
2021-09-05 02:43:00 +00:00
|
|
|
m.playbackTimer = m.top.findNode("playbackTimer")
|
|
|
|
m.bufferCheckTimer = m.top.findNode("bufferCheckTimer")
|
2020-05-24 09:33:07 +00:00
|
|
|
m.top.observeField("state", "onState")
|
2022-08-27 06:27:45 +00:00
|
|
|
m.top.observeField("position", "onPositionChanged")
|
|
|
|
m.top.trickPlayBar.observeField("visible", "onTrickPlayBarVisibilityChange")
|
2021-09-05 02:43:00 +00:00
|
|
|
m.playbackTimer.observeField("fire", "ReportPlayback")
|
2021-07-09 20:08:32 +00:00
|
|
|
m.bufferPercentage = 0 ' Track whether content is being loaded
|
2021-09-05 02:43:00 +00:00
|
|
|
m.playReported = false
|
2021-06-12 15:03:47 +00:00
|
|
|
m.top.transcodeReasons = []
|
2022-05-11 15:56:33 +00:00
|
|
|
m.bufferCheckTimer.duration = 10
|
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")
|
2022-08-12 06:48:14 +00:00
|
|
|
if clockNode[0] <> invalid then clockNode[0].parent.removeChild(clockNode[0].node)
|
2022-07-16 02:28:59 +00:00
|
|
|
end if
|
2022-08-27 06:27:45 +00:00
|
|
|
|
|
|
|
' Skip Intro Button
|
|
|
|
m.skipIntroButton = m.top.findNode("skipIntro")
|
|
|
|
m.skipIntroButton.text = tr("Skip Intro")
|
|
|
|
m.introCompleted = false
|
|
|
|
m.showskipIntroButtonAnimation = m.top.findNode("showskipIntroButton")
|
|
|
|
m.hideskipIntroButtonAnimation = m.top.findNode("hideskipIntroButton")
|
|
|
|
m.moveUpskipIntroButtonAnimation = m.top.findNode("moveUpskipIntroButton")
|
|
|
|
m.moveDownskipIntroButtonAnimation = m.top.findNode("moveDownskipIntroButton")
|
2022-09-07 04:06:10 +00:00
|
|
|
|
|
|
|
'Play Next Episode button
|
|
|
|
m.nextEpisodeButton = m.top.findNode("nextEpisode")
|
|
|
|
m.nextEpisodeButton.text = tr("Next Episode")
|
|
|
|
m.nextEpisodeButton.setFocus(false)
|
|
|
|
m.shownextEpisodeButtonAnimation = m.top.findNode("shownextEpisodeButton")
|
|
|
|
m.hidenextEpisodeButtonAnimation = m.top.findNode("hidenextEpisodeButton")
|
|
|
|
m.moveUpnextEpisodeButtonAnimation = m.top.findNode("moveUpnextEpisodeButton")
|
|
|
|
m.moveDownnextEpisodeButtonAnimation = m.top.findNode("moveDownnextEpisodeButton")
|
2022-08-27 06:27:45 +00:00
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' Checks if we have valid skip intro param data
|
|
|
|
function haveSkipIntroParams() as boolean
|
2022-09-07 04:06:10 +00:00
|
|
|
'check current position
|
|
|
|
if int(m.top.position) >= (m.top.runTime - 30)
|
|
|
|
shownextEpisode()
|
|
|
|
updateCount()
|
|
|
|
end if
|
2022-08-27 06:27:45 +00:00
|
|
|
' Intro data is invalid, skip
|
|
|
|
if not isValid(m.top.skipIntroParams?.Valid)
|
|
|
|
return false
|
|
|
|
end if
|
|
|
|
|
|
|
|
' Returned intro data is not valid, return
|
|
|
|
if not m.top.skipIntroParams.Valid
|
|
|
|
return false
|
|
|
|
end if
|
|
|
|
|
|
|
|
return true
|
|
|
|
end function
|
|
|
|
|
|
|
|
'
|
|
|
|
' Handles showing / hiding the skip intro button
|
|
|
|
sub handleSkipIntro()
|
|
|
|
' We've already shown the intro, return
|
|
|
|
if m.introCompleted then return
|
|
|
|
|
|
|
|
' We don't have valid data, return
|
|
|
|
if not haveSkipIntroParams() then return
|
|
|
|
|
|
|
|
' Check if it's time to hide the skip prompt
|
|
|
|
if m.top.position >= m.top.skipIntroParams.HideSkipPromptAt
|
|
|
|
if skipIntroButtonVisible()
|
|
|
|
hideSkipIntro()
|
|
|
|
end if
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
' Check if it's time to show the skip prompt
|
|
|
|
if m.top.position >= m.top.skipIntroParams.ShowSkipPromptAt
|
|
|
|
if not skipIntroButtonVisible()
|
|
|
|
showSkipIntro()
|
|
|
|
end if
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' When Trick Playbar Visibility changes
|
|
|
|
sub onTrickPlayBarVisibilityChange()
|
|
|
|
' Skip Intro button isn't visible, return
|
|
|
|
if not skipIntroButtonVisible() then return
|
|
|
|
|
|
|
|
' Trick Playbar is visible, move the skip intro button up and fade it out
|
|
|
|
if m.top.trickPlayBar.visible
|
|
|
|
m.moveUpskipIntroButtonAnimation.control = "start"
|
|
|
|
|
|
|
|
m.skipIntroButton.setFocus(false)
|
|
|
|
m.top.setFocus(true)
|
|
|
|
|
|
|
|
return
|
|
|
|
end if
|
|
|
|
|
|
|
|
' Trick Playbar is not visible, move the skip intro button down and fade it in
|
|
|
|
m.moveDownskipIntroButtonAnimation.control = "start"
|
|
|
|
m.skipIntroButton.setFocus(true)
|
|
|
|
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' When Video Player state changes
|
|
|
|
sub onPositionChanged()
|
|
|
|
' Check if content is episode
|
|
|
|
if m.top.content.contenttype = 4
|
|
|
|
handleSkipIntro()
|
|
|
|
end if
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' Returns if skip intro button is currently visible
|
|
|
|
function skipIntroButtonVisible() as boolean
|
|
|
|
return m.skipIntroButton.opacity > 0
|
|
|
|
end function
|
|
|
|
|
|
|
|
'
|
|
|
|
' Runs skip intro button animation and sets focus to button
|
|
|
|
sub showSkipIntro()
|
|
|
|
m.showskipIntroButtonAnimation.control = "start"
|
|
|
|
m.skipIntroButton.setFocus(true)
|
|
|
|
end sub
|
|
|
|
|
|
|
|
'
|
|
|
|
' Runs hide intro button animation and sets focus back to video
|
|
|
|
sub hideSkipIntro()
|
|
|
|
m.top.trickPlayBar.unobserveField("visible")
|
|
|
|
m.hideskipIntroButtonAnimation.control = "start"
|
|
|
|
m.introCompleted = true
|
|
|
|
m.skipIntroButton.setFocus(false)
|
|
|
|
m.top.setFocus(true)
|
2019-10-12 21:00:07 +00:00
|
|
|
end sub
|
2020-03-28 20:04:57 +00:00
|
|
|
|
2022-09-07 04:06:10 +00:00
|
|
|
'
|
|
|
|
' Runs Next Episode button animation and sets focus to button
|
|
|
|
sub shownextEpisode()
|
|
|
|
if m.nextEpisodeButton.hasFocus() = false
|
|
|
|
m.shownextEpisodeButtonAnimation.control = "start"
|
|
|
|
m.nextEpisodeButton.setFocus(true)
|
|
|
|
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
|
|
|
|
sub hidenextEpisode()
|
|
|
|
m.top.trickPlayBar.unobserveField("visible")
|
|
|
|
m.hidenextEpisodeButtonAnimation.control = "start"
|
|
|
|
m.nextEpisodeButton.setFocus(false)
|
|
|
|
m.top.setFocus(true)
|
|
|
|
end sub
|
|
|
|
|
|
|
|
|
2020-05-24 09:33:07 +00:00
|
|
|
'
|
|
|
|
' When Video Player state changes
|
2021-07-09 20:08:32 +00:00
|
|
|
sub onState(msg)
|
2020-05-24 09:33:07 +00:00
|
|
|
|
|
|
|
' When buffering, start timer to monitor buffering process
|
2021-06-26 13:52:16 +00:00
|
|
|
if m.top.state = "buffering" and m.bufferCheckTimer <> invalid
|
2020-05-30 08:44:22 +00:00
|
|
|
|
2020-05-24 09:33:07 +00:00
|
|
|
' start timer
|
|
|
|
m.bufferCheckTimer.control = "start"
|
|
|
|
m.bufferCheckTimer.ObserveField("fire", "bufferCheck")
|
2021-06-26 13:52:16 +00:00
|
|
|
else if m.top.state = "error"
|
2022-07-04 09:31:28 +00:00
|
|
|
if not m.playReported and m.top.transcodeAvailable
|
|
|
|
m.top.retryWithTranscoding = true ' If playback was not reported, retry with transcoding
|
|
|
|
else
|
2022-07-04 09:09:16 +00:00
|
|
|
' 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
|
2022-07-04 09:31:28 +00:00
|
|
|
|
2020-05-30 08:44:22 +00:00
|
|
|
' Stop playback and exit player
|
|
|
|
m.top.control = "stop"
|
|
|
|
m.top.backPressed = true
|
2021-09-05 02:43:00 +00:00
|
|
|
else if m.top.state = "playing"
|
|
|
|
if m.playReported = false
|
|
|
|
ReportPlayback("start")
|
|
|
|
m.playReported = true
|
|
|
|
else
|
|
|
|
m.playbackTimer.control = "start"
|
|
|
|
ReportPlayback()
|
|
|
|
end if
|
|
|
|
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,
|
2021-09-05 02:43:00 +00:00
|
|
|
"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")
|
2021-09-05 02:43:00 +00:00
|
|
|
}
|
|
|
|
if m.top.content.live
|
|
|
|
params.append({
|
|
|
|
"MediaSourceId": m.top.transcodeParams.MediaSourceId,
|
|
|
|
"LiveStreamId": m.top.transcodeParams.LiveStreamId
|
|
|
|
})
|
2022-05-11 15:56:33 +00:00
|
|
|
m.bufferCheckTimer.duration = 30
|
2020-05-24 09:33:07 +00:00
|
|
|
end if
|
|
|
|
|
2021-09-05 02:43:00 +00:00
|
|
|
' Report playstate via worker task
|
|
|
|
playstateTask = m.global.playstateTask
|
|
|
|
playstateTask.setFields({ status: state, params: params })
|
|
|
|
playstateTask.control = "RUN"
|
2020-05-24 09:33:07 +00:00
|
|
|
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
|
2021-06-26 13:52:16 +00:00
|
|
|
if m.top.bufferingStatus <> invalid
|
2020-05-24 09:33:07 +00:00
|
|
|
|
|
|
|
' Check that the buffering percentage is increasing
|
2021-06-26 13:52:16 +00:00
|
|
|
if m.top.bufferingStatus["percentage"] > m.bufferPercentage
|
2020-05-24 09:33:07 +00:00
|
|
|
m.bufferPercentage = m.top.bufferingStatus["percentage"]
|
2022-05-11 15:56:33 +00:00
|
|
|
else if m.top.content.live = true
|
|
|
|
m.top.callFunc("refresh")
|
2020-05-24 09:33:07 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
|
2020-03-28 20:04:57 +00:00
|
|
|
function onKeyEvent(key as string, press as boolean) as boolean
|
2022-08-27 06:27:45 +00:00
|
|
|
if key = "OK"
|
|
|
|
if not m.top.trickPlayBar.visible
|
|
|
|
if m.skipIntroButton.hasFocus()
|
|
|
|
m.top.seek = m.top.skipIntroParams.IntroEnd
|
|
|
|
hideSkipIntro()
|
|
|
|
return true
|
2022-09-07 04:06:10 +00:00
|
|
|
else if m.nextEpisodeButton.hasFocus()
|
|
|
|
m.top.state = "finished"
|
|
|
|
hidenextEpisode()
|
|
|
|
return true
|
2022-08-27 06:27:45 +00:00
|
|
|
end if
|
|
|
|
end if
|
|
|
|
end if
|
|
|
|
|
2021-07-09 20:08:32 +00:00
|
|
|
if not press then return false
|
2020-03-28 20:04:57 +00:00
|
|
|
|
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
|
|
|
|
end if
|
2020-03-28 20:04:57 +00:00
|
|
|
|
2021-07-09 20:08:32 +00:00
|
|
|
return false
|
2020-03-28 20:04:57 +00:00
|
|
|
end function
|