components_liveTv_ProgramDetails.bs
import "pkg:/source/utils/config.bs"
import "pkg:/source/utils/misc.bs"
sub init()
' Max "Overview" lines to show in Preview and Detail
m.maxPreviewLines = 5
m.maxDetailLines = 14
m.detailsView = m.top.findNode("detailsView")
m.noInfoView = m.top.findNode("noInformation")
m.programName = m.top.findNode("programName")
m.episodeTitle = m.top.findNode("episodeTitle")
m.episodeNumber = m.top.findNode("episodeNumber")
m.overview = m.top.findNode("overview")
m.episodeDetailsGroup = m.top.findNode("episodeDetailsGroup")
m.isLiveGroup = m.top.findNode("isLive")
m.isRepeatGroup = m.top.findNode("isRepeat")
m.broadcastDetails = m.top.findNode("broadcastDetails")
m.duration = m.top.findNode("duration")
m.channelName = m.top.findNode("channelName")
m.image = m.top.findNode("image")
m.favorite = m.top.findNode("favorite")
m.buttonsFocusAnimationOpacity = m.top.findNode("buttonsFocusAnimationOpacity")
m.focusAnimation = m.top.findNode("focusAnimation")
m.buttons = m.top.findNode("buttons")
m.viewChannelButton = m.top.findNode("viewChannelButton")
m.recordButton = m.top.findNode("recordButton")
m.recordSeriesButton = m.top.findNode("recordSeriesButton")
m.focusAnimation.observeField("state", "onAnimationComplete")
m.buttons.observeField("buttonSelected", "onButtonSelected")
setupLabels()
end sub
' Set up Live and Repeat label sizes
sub setupLabels()
boundingRect = m.top.findNode("isLiveText").boundingRect()
isLiveBackground = m.top.findNode("isLiveBackground")
isLiveBackground.width = boundingRect.width + 16
isLiveBackground.height = boundingRect.height + 8
m.episodeDetailsGroup.removeChildIndex(0)
boundingRect = m.top.findNode("isRepeatText").boundingRect()
isRepeatBackground = m.top.findNode("isRepeatBackground")
isRepeatBackground.width = boundingRect.width + 16
isRepeatBackground.height = boundingRect.height + 8
m.episodeDetailsGroup.removeChildIndex(0)
' Handle user recording permissions
m.userCanRecord = m.global.user.policy.enableLiveTvManagement
if m.userCanRecord = false
m.recordButton.enabled = false
m.recordSeriesButton.enabled = false
end if
end sub
sub updateLabels(recordText = tr("Record"), recordSeriesText = tr("Record Series"))
m.recordButton.text = recordText
m.recordSeriesButton.text = recordSeriesText
end sub
sub channelUpdated()
if not isValid(m.top.channel)
m.top.findNode("noInfoChannelName").text = ""
m.channelName.text = ""
else
m.top.findNode("noInfoChannelName").text = m.top.channel.Title
m.channelName.text = m.top.channel.Title
if not isValid(m.top.programDetails)
m.image.uri = m.top.channel.posterURL
end if
m.favorite.visible = m.top.channel.favorite
end if
end sub
sub programUpdated()
m.top.watchSelectedChannel = false
m.top.recordSelectedChannel = false
m.top.recordSeriesSelectedChannel = false
m.overview.maxLines = m.maxDetailLines
prog = m.top.programDetails
' If no program selected, hide details view
if not isValid(prog)
channelUpdated()
m.detailsView.visible = "false"
m.noInfoView.visible = "true"
return
end if
m.programName.text = prog.Title
m.overview.text = prog.description
m.episodeDetailsGroup.removeChildrenIndex(m.episodeDetailsGroup.getChildCount(), 0)
if prog.isLive
m.episodeDetailsGroup.appendChild(m.isLiveGroup)
else if prog.isRepeat
m.episodeDetailsGroup.appendChild(m.isRepeatGroup)
end if
' Episode Number
if prog.seasonNumber > 0 and prog.episodeNumber > 0
m.episodeNumber.text = "S" + StrI(prog.seasonNumber).trim() + ":E" + StrI(prog.episodeNumber).trim()
if prog.episodeTitle <> "" then m.episodeNumber.text = m.episodeNumber.text + " -" ' Add a Dash if showing Episode Number and Title
m.episodeDetailsGroup.appendChild(m.episodeNumber)
end if
if isValid(prog.episodeTitle) and prog.episodeTitle <> ""
m.episodeTitle.text = prog.episodeTitle
m.episodeTitle.visible = true
m.episodeDetailsGroup.appendChild(m.episodeTitle)
end if
m.duration.text = getDurationStringFromSeconds(prog.PlayDuration)
' Calculate Broadcast Details
now = createObject("roDateTime")
startDate = createObject("roDateTime")
endDate = createObject("roDateTime")
startDate.FromISO8601String(prog.StartDate)
endDate.FromISO8601String(prog.EndDate)
day = getRelativeDayName(startDate)
' Get Start Date in local timezone for display to user
localStartDate = createObject("roDateTime")
localStartDate.FromISO8601String(prog.StartDate)
localStartDate.ToLocalTime()
if startDate.AsSeconds() < now.AsSeconds() and endDate.AsSeconds() > now.AsSeconds()
if day = "today"
m.broadcastDetails.text = tr("Started at") + " " + formatTime(localStartDate)
else
m.broadcastDetails.text = tr("Started") + " " + tr(day) + ", " + formatTime(localStartDate)
end if
else if startDate.AsSeconds() > now.AsSeconds()
if day = "today"
m.broadcastDetails.text = tr("Starts at") + " " + formatTime(localStartDate)
else
m.broadcastDetails.text = tr("Starts") + " " + tr(day) + ", " + formatTime(localStartDate)
end if
else
if day = "today"
m.broadcastDetails.text = tr("Ended at") + " " + formatTime(localStartDate)
else
m.broadcastDetails.text = tr("Ended") + " " + tr(day) + ", " + formatTime(localStartDate)
end if
end if
m.image.uri = prog.PosterURL
' If currently being recorded, change button to "Stop Recording"
if isValid(prog.json.TimerId)
if isValid(prog.json.SeriesTimerId)
updateLabels(tr("Cancel Recording"), tr("Cancel Series Recording"))
else
updateLabels(tr("Cancel Recording"), tr("Record Series"))
end if
else
updateLabels()
end if
' If not a series, hide Record Series button (doesn't make sense for non-series content)
if prog.json.isSeries <> true ' could be invalid or false
m.recordSeriesButton.visible = false
m.recordSeriesButton.enabled = false
else
m.recordSeriesButton.visible = true
' Restore enabled state based on user permissions (set in setupLabels)
m.recordSeriesButton.enabled = m.userCanRecord
end if
m.detailsView.visible = "true"
m.noInfoView.visible = "false"
m.top.height = m.detailsView.boundingRect().height
m.overview.maxLines = m.maxPreviewLines
end sub
'
' Get relative date name for a date (yesterday, today, tomorrow, or otherwise weekday name )
function getRelativeDayName(date) as string
now = createObject("roDateTime")
' needed for update-translations script to prevent strings from being removed
' tr("today")
' tr("yesterday")
' tr("tomorrow")
' tr("Monday")
' tr("Tuesday")
' tr("Wednesday")
' tr("Thursday")
' tr("Friday")
' tr("Saturday")
' tr("Sunday")
' Check for Today
if now.AsDateString("short-date-dashes") = date.AsDateString("short-date-dashes")
return "today"
end if
' Check for Yesterday
todayMidnight = now.AsSeconds() - (now.AsSeconds() mod 86400)
dateMidnight = date.AsSeconds() - (date.AsSeconds() mod 86400)
if todayMidnight - dateMidnight = 86400
return "yesterday"
end if
if dateMidnight - todayMidnight = 86400
return "tomorrow"
end if
return date.GetWeekday()
end function
'
' Get program duration string (e.g. 1h 20m)
function getDurationStringFromSeconds(seconds) as string
hours = 0
minutes = seconds / 60.0
if minutes > 60
hours = (minutes - (minutes mod 60)) / 60
minutes = minutes mod 60
end if
if hours > 0
return "%1h %2m".Replace("%1", StrI(hours).trim()).Replace("%2", StrI(minutes).trim())
else
return "%1m".Replace("%1", StrI(minutes).trim())
end if
end function
'
' Show button group when item has Focus
sub focusChanged()
if m.top.hasFocus = true
m.overview.maxLines = m.maxDetailLines
m.buttonsFocusAnimationOpacity.keyValue = [0, 1]
m.buttons.callFunc("focus") ' JRButtonGroup handles button focus
else
m.top.watchSelectedChannel = false
m.top.recordSelectedChannel = false
m.top.recordSeriesSelectedChannel = false
m.buttonsFocusAnimationOpacity.keyValue = [1, 0]
end if
m.focusAnimation.control = "start"
end sub
sub onAnimationComplete()
if m.focusAnimation.state = "stopped" and m.top.hasFocus = false
m.overview.maxLines = m.maxPreviewLines
end if
end sub
' Handle button selection from JRButtonGroup
sub onButtonSelected()
selectedIndex = m.buttons.buttonSelected
' Button indices: 0 = viewChannel, 1 = record, 2 = recordSeries
if selectedIndex = 0
m.top.watchSelectedChannel = true
else if selectedIndex = 1
m.top.recordSelectedChannel = true
else if selectedIndex = 2
m.top.recordSeriesSelectedChannel = true
end if
end sub
function onKeyEvent(key as string, press as boolean) as boolean
if not press then return false
' Just prevent up/down from bubbling
if key = "up" or key = "down"
return true
end if
return false
end function