mirror of
https://github.com/jellyfin/jellyfin-roku.git
synced 2024-11-30 09:50:46 +00:00
1127 lines
38 KiB
Plaintext
1127 lines
38 KiB
Plaintext
import "pkg:/source/utils/misc.bs"
|
|
import "pkg:/source/api/baserequest.bs"
|
|
|
|
' Returns the Device Capabilities for Roku.
|
|
' Also prints out the device profile for debugging
|
|
function getDeviceCapabilities() as object
|
|
deviceProfile = {
|
|
"PlayableMediaTypes": [
|
|
"Audio",
|
|
"Video",
|
|
"Photo"
|
|
],
|
|
"SupportedCommands": [],
|
|
"SupportsPersistentIdentifier": true,
|
|
"SupportsMediaControl": false,
|
|
"SupportsContentUploading": false,
|
|
"SupportsSync": false,
|
|
"DeviceProfile": getDeviceProfile(),
|
|
"AppStoreUrl": "https://channelstore.roku.com/details/cc5e559d08d9ec87c5f30dcebdeebc12/jellyfin"
|
|
}
|
|
|
|
printDeviceProfile(deviceProfile)
|
|
|
|
return deviceProfile
|
|
end function
|
|
|
|
function getDeviceProfile() as object
|
|
globalDevice = m.global.device
|
|
return {
|
|
"Name": "Official Roku Client",
|
|
"Id": globalDevice.id,
|
|
"Identification": {
|
|
"FriendlyName": globalDevice.friendlyName,
|
|
"ModelNumber": globalDevice.model,
|
|
"SerialNumber": "string",
|
|
"ModelName": globalDevice.name,
|
|
"ModelDescription": "Type: " + globalDevice.modelType,
|
|
"Manufacturer": globalDevice.modelDetails.VendorName
|
|
},
|
|
"FriendlyName": globalDevice.friendlyName,
|
|
"Manufacturer": globalDevice.modelDetails.VendorName,
|
|
"ModelName": globalDevice.name,
|
|
"ModelDescription": "Type: " + globalDevice.modelType,
|
|
"ModelNumber": globalDevice.model,
|
|
"SerialNumber": globalDevice.serial,
|
|
"MaxStreamingBitrate": 120000000,
|
|
"MaxStaticBitrate": 100000000,
|
|
"MusicStreamingTranscodingBitrate": 192000,
|
|
"DirectPlayProfiles": GetDirectPlayProfiles(),
|
|
"TranscodingProfiles": getTranscodingProfiles(),
|
|
"ContainerProfiles": getContainerProfiles(),
|
|
"CodecProfiles": getCodecProfiles(),
|
|
"SubtitleProfiles": getSubtitleProfiles()
|
|
}
|
|
end function
|
|
|
|
function GetDirectPlayProfiles() as object
|
|
globalUserSettings = m.global.session.user.settings
|
|
directPlayProfiles = []
|
|
di = CreateObject("roDeviceInfo")
|
|
' all possible containers
|
|
supportedCodecs = {
|
|
mp4: {
|
|
audio: [],
|
|
video: []
|
|
},
|
|
hls: {
|
|
audio: [],
|
|
video: []
|
|
},
|
|
mkv: {
|
|
audio: [],
|
|
video: []
|
|
},
|
|
ism: {
|
|
audio: [],
|
|
video: []
|
|
},
|
|
dash: {
|
|
audio: [],
|
|
video: []
|
|
},
|
|
ts: {
|
|
audio: [],
|
|
video: []
|
|
}
|
|
}
|
|
' all possible codecs (besides those restricted by user settings)
|
|
videoCodecs = ["h264", "mpeg4 avc", "vp8", "vp9", "h263", "mpeg1"]
|
|
audioCodecs = ["mp3", "mp2", "pcm", "lpcm", "wav", "ac3", "ac4", "aiff", "wma", "flac", "alac", "aac", "opus", "dts", "wmapro", "vorbis", "eac3", "mpg123"]
|
|
|
|
' check if hevc is disabled
|
|
if globalUserSettings["playback.compatibility.disablehevc"] = false
|
|
videoCodecs.push("hevc")
|
|
end if
|
|
|
|
' check video codecs for each container
|
|
for each container in supportedCodecs
|
|
for each videoCodec in videoCodecs
|
|
if di.CanDecodeVideo({ Codec: videoCodec, Container: container }).Result
|
|
if videoCodec = "hevc"
|
|
supportedCodecs[container]["video"].push("hevc")
|
|
supportedCodecs[container]["video"].push("h265")
|
|
else
|
|
' device profile string matches codec string
|
|
supportedCodecs[container]["video"].push(videoCodec)
|
|
end if
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' user setting overrides
|
|
if globalUserSettings["playback.mpeg4"]
|
|
for each container in supportedCodecs
|
|
supportedCodecs[container]["video"].push("mpeg4")
|
|
end for
|
|
end if
|
|
if globalUserSettings["playback.mpeg2"]
|
|
for each container in supportedCodecs
|
|
supportedCodecs[container]["video"].push("mpeg2video")
|
|
end for
|
|
end if
|
|
|
|
' video codec overrides
|
|
' these codecs play fine but are not correctly detected using CanDecodeVideo()
|
|
if di.CanDecodeVideo({ Codec: "av1" }).Result
|
|
' codec must be checked by itself or the result will always be false
|
|
for each container in supportedCodecs
|
|
supportedCodecs[container]["video"].push("av1")
|
|
end for
|
|
end if
|
|
|
|
' check audio codecs for each container
|
|
for each container in supportedCodecs
|
|
for each audioCodec in audioCodecs
|
|
if di.CanDecodeAudio({ Codec: audioCodec, Container: container }).Result
|
|
supportedCodecs[container]["audio"].push(audioCodec)
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' remove audio codecs not supported as standalone audio files (opus)
|
|
' also add aac back to the list so it gets added to the direct play profile
|
|
audioCodecs = ["aac", "mp3", "mp2", "pcm", "lpcm", "wav", "ac3", "ac4", "aiff", "wma", "flac", "alac", "aac", "dts", "wmapro", "vorbis", "eac3", "mpg123"]
|
|
|
|
' check audio codecs with no container
|
|
supportedAudio = []
|
|
for each audioCodec in audioCodecs
|
|
if di.CanDecodeAudio({ Codec: audioCodec }).Result
|
|
supportedAudio.push(audioCodec)
|
|
end if
|
|
end for
|
|
|
|
' build return array
|
|
for each container in supportedCodecs
|
|
videoCodecString = supportedCodecs[container]["video"].Join(",")
|
|
if videoCodecString <> ""
|
|
containerString = container
|
|
|
|
if container = "mp4"
|
|
containerString = "mp4,mov,m4v"
|
|
else if container = "mkv"
|
|
containerString = "mkv,webm"
|
|
end if
|
|
|
|
directPlayProfiles.push({
|
|
"Container": containerString,
|
|
"Type": "Video",
|
|
"VideoCodec": videoCodecString,
|
|
"AudioCodec": supportedCodecs[container]["audio"].Join(",")
|
|
})
|
|
end if
|
|
end for
|
|
|
|
directPlayProfiles.push({
|
|
"Container": supportedAudio.Join(","),
|
|
"Type": "Audio"
|
|
})
|
|
return directPlayProfiles
|
|
end function
|
|
|
|
function getTranscodingProfiles() as object
|
|
globalUserSettings = m.global.session.user.settings
|
|
transcodingProfiles = []
|
|
|
|
di = CreateObject("roDeviceInfo")
|
|
|
|
transcodingContainers = ["mp4", "ts"]
|
|
' use strings to preserve order
|
|
mp4AudioCodecs = "aac"
|
|
mp4VideoCodecs = "h264"
|
|
tsAudioCodecs = "aac"
|
|
tsVideoCodecs = "h264"
|
|
|
|
' does the users setup support surround sound?
|
|
maxAudioChannels = "2" ' jellyfin expects this as a string
|
|
' in order of preference from left to right
|
|
audioCodecs = ["eac3", "ac3", "dts", "mp3", "vorbis", "opus", "flac", "alac", "ac4", "pcm", "wma", "wmapro"]
|
|
|
|
if di.GetAudioOutputChannel() = "5.1 surround"
|
|
maxAudioChannels = "6"
|
|
for each codec in audioCodecs
|
|
if di.CanDecodeAudio({ Codec: codec, ChCnt: 8 }).Result
|
|
maxAudioChannels = "8"
|
|
exit for
|
|
end if
|
|
end for
|
|
end if
|
|
|
|
' VIDEO CODECS
|
|
'
|
|
' AVC / h264 / MPEG4 AVC
|
|
for each container in transcodingContainers
|
|
if di.CanDecodeVideo({ Codec: "h264", Container: container }).Result
|
|
if container = "mp4"
|
|
' check for codec string before adding it
|
|
if mp4VideoCodecs.Instr(0, ",h264") = -1
|
|
mp4VideoCodecs = mp4VideoCodecs + ",h264"
|
|
end if
|
|
else if container = "ts"
|
|
' check for codec string before adding it
|
|
if tsVideoCodecs.Instr(0, ",h264") = -1
|
|
tsVideoCodecs = tsVideoCodecs + ",h264"
|
|
end if
|
|
end if
|
|
end if
|
|
if di.CanDecodeVideo({ Codec: "mpeg4 avc", Container: container }).Result
|
|
if container = "mp4"
|
|
' check for codec string before adding it
|
|
if mp4VideoCodecs.Instr(0, ",mpeg4 avc") = -1
|
|
mp4VideoCodecs = mp4VideoCodecs + ",mpeg4 avc"
|
|
end if
|
|
else if container = "ts"
|
|
' check for codec string before adding it
|
|
if tsVideoCodecs.Instr(0, ",mpeg4 avc") = -1
|
|
tsVideoCodecs = tsVideoCodecs + ",mpeg4 avc"
|
|
end if
|
|
end if
|
|
end if
|
|
end for
|
|
|
|
' HEVC / h265
|
|
if globalUserSettings["playback.compatibility.disablehevc"] = false
|
|
for each container in transcodingContainers
|
|
if di.CanDecodeVideo({ Codec: "hevc", Container: container }).Result
|
|
if container = "mp4"
|
|
' check for codec string before adding it
|
|
if mp4VideoCodecs.Instr(0, "h265,") = -1
|
|
mp4VideoCodecs = "h265," + mp4VideoCodecs
|
|
end if
|
|
if mp4VideoCodecs.Instr(0, "hevc,") = -1
|
|
mp4VideoCodecs = "hevc," + mp4VideoCodecs
|
|
end if
|
|
else if container = "ts"
|
|
' check for codec string before adding it
|
|
if tsVideoCodecs.Instr(0, "h265,") = -1
|
|
tsVideoCodecs = "h265," + tsVideoCodecs
|
|
end if
|
|
if tsVideoCodecs.Instr(0, "hevc,") = -1
|
|
tsVideoCodecs = "hevc," + tsVideoCodecs
|
|
end if
|
|
end if
|
|
end if
|
|
end for
|
|
end if
|
|
|
|
' VP9
|
|
for each container in transcodingContainers
|
|
if di.CanDecodeAudio({ Codec: "vp9", Container: container }).Result
|
|
if container = "mp4"
|
|
' check for codec string before adding it
|
|
if mp4VideoCodecs.Instr(0, ",vp9") = -1
|
|
mp4VideoCodecs = mp4VideoCodecs + ",vp9"
|
|
end if
|
|
else if container = "ts"
|
|
' check for codec string before adding it
|
|
if tsVideoCodecs.Instr(0, ",vp9") = -1
|
|
tsVideoCodecs = tsVideoCodecs + ",vp9"
|
|
end if
|
|
end if
|
|
end if
|
|
end for
|
|
|
|
' MPEG2
|
|
if globalUserSettings["playback.mpeg2"]
|
|
for each container in transcodingContainers
|
|
if di.CanDecodeVideo({ Codec: "mpeg2", Container: container }).Result
|
|
if container = "mp4"
|
|
' check for codec string before adding it
|
|
if mp4VideoCodecs.Instr(0, ",mpeg2video") = -1
|
|
mp4VideoCodecs = mp4VideoCodecs + ",mpeg2video"
|
|
end if
|
|
else if container = "ts"
|
|
' check for codec string before adding it
|
|
if tsVideoCodecs.Instr(0, ",mpeg2video") = -1
|
|
tsVideoCodecs = tsVideoCodecs + ",mpeg2video"
|
|
end if
|
|
end if
|
|
end if
|
|
end for
|
|
end if
|
|
|
|
' AV1
|
|
' CanDecodeVideo() returns false for AV1 when the container is provided
|
|
' Manually add AV1 to the mp4VideoCodecs list if support is detected
|
|
if di.CanDecodeVideo({ Codec: "av1" }).Result
|
|
mp4VideoCodecs = mp4VideoCodecs + ",av1"
|
|
end if
|
|
|
|
' AUDIO CODECS
|
|
for each container in transcodingContainers
|
|
for each codec in audioCodecs
|
|
if di.CanDecodeAudio({ Codec: codec, Container: container }).result
|
|
if container = "mp4"
|
|
mp4AudioCodecs = mp4AudioCodecs + "," + codec
|
|
else if container = "ts"
|
|
tsAudioCodecs = tsAudioCodecs + "," + codec
|
|
end if
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' add aac to TranscodingProfile for stereo audio
|
|
' NOTE: multichannel aac is not supported. only decode to stereo on some devices
|
|
transcodingProfiles.push({
|
|
"Container": "aac",
|
|
"Type": "Audio",
|
|
"AudioCodec": "aac",
|
|
"Context": "Streaming",
|
|
"Protocol": "http",
|
|
"MaxAudioChannels": "2"
|
|
})
|
|
transcodingProfiles.push({
|
|
"Container": "aac",
|
|
"Type": "Audio",
|
|
"AudioCodec": "aac",
|
|
"Context": "Static",
|
|
"Protocol": "http",
|
|
"MaxAudioChannels": "2"
|
|
})
|
|
' add mp3 to TranscodingProfile for multichannel music
|
|
transcodingProfiles.push({
|
|
"Container": "mp3",
|
|
"Type": "Audio",
|
|
"AudioCodec": "mp3",
|
|
"Context": "Streaming",
|
|
"Protocol": "http",
|
|
"MaxAudioChannels": maxAudioChannels
|
|
})
|
|
transcodingProfiles.push({
|
|
"Container": "mp3",
|
|
"Type": "Audio",
|
|
"AudioCodec": "mp3",
|
|
"Context": "Static",
|
|
"Protocol": "http",
|
|
"MaxAudioChannels": maxAudioChannels
|
|
})
|
|
|
|
tsArray = {
|
|
"Container": "ts",
|
|
"Context": "Streaming",
|
|
"Protocol": "hls",
|
|
"Type": "Video",
|
|
"AudioCodec": tsAudioCodecs,
|
|
"VideoCodec": tsVideoCodecs,
|
|
"MaxAudioChannels": maxAudioChannels,
|
|
"MinSegments": 1,
|
|
"BreakOnNonKeyFrames": false
|
|
}
|
|
mp4Array = {
|
|
"Container": "mp4",
|
|
"Context": "Streaming",
|
|
"Protocol": "hls",
|
|
"Type": "Video",
|
|
"AudioCodec": mp4AudioCodecs,
|
|
"VideoCodec": mp4VideoCodecs,
|
|
"MaxAudioChannels": maxAudioChannels,
|
|
"MinSegments": 1,
|
|
"BreakOnNonKeyFrames": false
|
|
}
|
|
|
|
' apply max res to transcoding profile
|
|
if globalUserSettings["playback.resolution.max"] <> "off"
|
|
tsArray.Conditions = [getMaxHeightArray(), getMaxWidthArray()]
|
|
mp4Array.Conditions = [getMaxHeightArray(), getMaxWidthArray()]
|
|
end if
|
|
|
|
' add user-selected preferred codec to the front of the list
|
|
if globalUserSettings["playback.preferredAudioCodec"] <> "auto"
|
|
tsArray.AudioCodec = globalUserSettings["playback.preferredAudioCodec"] + "," + tsArray.AudioCodec
|
|
mp4Array.AudioCodec = globalUserSettings["playback.preferredAudioCodec"] + "," + mp4Array.AudioCodec
|
|
end if
|
|
|
|
transcodingProfiles.push(tsArray)
|
|
transcodingProfiles.push(mp4Array)
|
|
|
|
return transcodingProfiles
|
|
end function
|
|
|
|
function getContainerProfiles() as object
|
|
containerProfiles = []
|
|
|
|
return containerProfiles
|
|
end function
|
|
|
|
function getCodecProfiles() as object
|
|
myGlobal = m.global
|
|
globalUserSettings = myGlobal.session.user.settings
|
|
|
|
codecProfiles = []
|
|
profileSupport = {
|
|
"h264": {},
|
|
"mpeg4 avc": {},
|
|
"h265": {},
|
|
"hevc": {},
|
|
"vp9": {},
|
|
"mpeg2": {},
|
|
"av1": {}
|
|
}
|
|
maxResSetting = globalUserSettings["playback.resolution.max"]
|
|
di = CreateObject("roDeviceInfo")
|
|
maxHeightArray = getMaxHeightArray()
|
|
maxWidthArray = getMaxWidthArray()
|
|
|
|
' AUDIO
|
|
' test each codec to see how many channels are supported
|
|
audioCodecs = ["aac", "mp3", "mp2", "opus", "pcm", "lpcm", "wav", "flac", "alac", "ac3", "ac4", "aiff", "dts", "wmapro", "vorbis", "eac3", "mpg123"]
|
|
audioChannels = [8, 6, 2] ' highest first
|
|
for each audioCodec in audioCodecs
|
|
for each audioChannel in audioChannels
|
|
channelSupportFound = false
|
|
if di.CanDecodeAudio({ Codec: audioCodec, ChCnt: audioChannel }).Result
|
|
channelSupportFound = true
|
|
for each codecType in ["VideoAudio", "Audio"]
|
|
if audioCodec = "aac"
|
|
codecProfiles.push({
|
|
"Type": codecType,
|
|
"Codec": audioCodec,
|
|
"Conditions": [
|
|
{
|
|
"Condition": "NotEquals",
|
|
"Property": "AudioProfile",
|
|
"Value": "Main",
|
|
"IsRequired": true
|
|
},
|
|
{
|
|
"Condition": "NotEquals",
|
|
"Property": "AudioProfile",
|
|
"Value": "HE-AAC",
|
|
"IsRequired": true
|
|
},
|
|
{
|
|
"Condition": "LessThanEqual",
|
|
"Property": "AudioChannels",
|
|
"Value": audioChannel,
|
|
"IsRequired": true
|
|
}
|
|
]
|
|
})
|
|
else if audioCodec = "opus" and codecType = "Audio"
|
|
' opus audio files not supported by roku
|
|
else
|
|
codecProfiles.push({
|
|
"Type": codecType,
|
|
"Codec": audioCodec,
|
|
"Conditions": [
|
|
{
|
|
"Condition": "LessThanEqual",
|
|
"Property": "AudioChannels",
|
|
"Value": audioChannel,
|
|
"IsRequired": true
|
|
}
|
|
]
|
|
})
|
|
end if
|
|
end for
|
|
end if
|
|
if channelSupportFound
|
|
' if 8 channels are supported we don't need to test for 6 or 2
|
|
' if 6 channels are supported we don't need to test 2
|
|
exit for
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' check device for codec profile and level support
|
|
' AVC / h264 / MPEG4 AVC
|
|
h264Profiles = ["main", "high"]
|
|
h264Levels = ["4.1", "4.2"]
|
|
for each profile in h264Profiles
|
|
for each level in h264Levels
|
|
if di.CanDecodeVideo({ Codec: "h264", Profile: profile, Level: level }).Result
|
|
profileSupport = updateProfileArray(profileSupport, "h264", profile, level)
|
|
end if
|
|
if di.CanDecodeVideo({ Codec: "mpeg4 avc", Profile: profile, Level: level }).Result
|
|
profileSupport = updateProfileArray(profileSupport, "mpeg4 avc", profile, level)
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' HEVC / h265
|
|
hevcProfiles = ["main", "main 10"]
|
|
hevcLevels = ["4.1", "5.0", "5.1"]
|
|
for each profile in hevcProfiles
|
|
for each level in hevcLevels
|
|
if di.CanDecodeVideo({ Codec: "hevc", Profile: profile, Level: level }).Result
|
|
profileSupport = updateProfileArray(profileSupport, "h265", profile, level)
|
|
profileSupport = updateProfileArray(profileSupport, "hevc", profile, level)
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' VP9
|
|
vp9Profiles = ["profile 0", "profile 2"]
|
|
vp9Levels = ["4.1", "5.0", "5.1"]
|
|
for each profile in vp9Profiles
|
|
for each level in vp9Levels
|
|
if di.CanDecodeVideo({ Codec: "vp9", Profile: profile, Level: level }).Result
|
|
profileSupport = updateProfileArray(profileSupport, "vp9", profile, level)
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' MPEG2
|
|
' mpeg2 uses levels with no profiles. see https://developer.roku.com/en-ca/docs/references/brightscript/interfaces/ifdeviceinfo.md#candecodevideovideo_format-as-object-as-object
|
|
' NOTE: the mpeg2 levels are being saved in the profileSupport array as if they were profiles
|
|
mpeg2Levels = ["main", "high"]
|
|
for each level in mpeg2Levels
|
|
if di.CanDecodeVideo({ Codec: "mpeg2", Level: level }).Result
|
|
profileSupport = updateProfileArray(profileSupport, "mpeg2", level)
|
|
end if
|
|
end for
|
|
|
|
' AV1
|
|
av1Profiles = ["main", "main 10"]
|
|
av1Levels = ["4.1", "5.0", "5.1"]
|
|
for each profile in av1Profiles
|
|
for each level in av1Levels
|
|
if di.CanDecodeVideo({ Codec: "av1", Profile: profile, Level: level }).Result
|
|
profileSupport = updateProfileArray(profileSupport, "av1", profile, level)
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' HDR SUPPORT
|
|
h264VideoRangeTypes = "SDR"
|
|
hevcVideoRangeTypes = "SDR"
|
|
vp9VideoRangeTypes = "SDR"
|
|
av1VideoRangeTypes = "SDR"
|
|
canPlayDovi = false
|
|
|
|
if canPlay4k()
|
|
dp = di.GetDisplayProperties()
|
|
|
|
if dp.DolbyVision
|
|
canPlayDovi = true
|
|
|
|
h264VideoRangeTypes = h264VideoRangeTypes + "|DOVI|DOVIWithSDR"
|
|
hevcVideoRangeTypes = hevcVideoRangeTypes + "|DOVI|DOVIWithSDR"
|
|
av1VideoRangeTypes = av1VideoRangeTypes + "|DOVI|DOVIWithSDR"
|
|
end if
|
|
|
|
if dp.Hdr10
|
|
hevcVideoRangeTypes = hevcVideoRangeTypes + "|HDR10"
|
|
vp9VideoRangeTypes = vp9VideoRangeTypes + "|HDR10"
|
|
av1VideoRangeTypes = av1VideoRangeTypes + "|HDR10"
|
|
|
|
if canPlayDovi
|
|
hevcVideoRangeTypes = hevcVideoRangeTypes + "|DOVIWithHDR10"
|
|
av1VideoRangeTypes = av1VideoRangeTypes + "|DOVIWithHDR10"
|
|
end if
|
|
end if
|
|
|
|
if dp.Hdr10Plus
|
|
av1VideoRangeTypes = av1VideoRangeTypes + "|HDR10+"
|
|
end if
|
|
|
|
if dp.HLG
|
|
hevcVideoRangeTypes = hevcVideoRangeTypes + "|HLG"
|
|
vp9VideoRangeTypes = vp9VideoRangeTypes + "|HLG"
|
|
av1VideoRangeTypes = av1VideoRangeTypes + "|HLG"
|
|
|
|
if canPlayDovi
|
|
hevcVideoRangeTypes = hevcVideoRangeTypes + "|DOVIWithHLG"
|
|
vp9VideoRangeTypes = vp9VideoRangeTypes + "|DOVIWithHLG"
|
|
av1VideoRangeTypes = av1VideoRangeTypes + "|DOVIWithHLG"
|
|
end if
|
|
end if
|
|
end if
|
|
|
|
' H264
|
|
h264LevelSupported = 0.0
|
|
h264AssProfiles = {}
|
|
for each profile in profileSupport["h264"]
|
|
h264AssProfiles.AddReplace(profile, true)
|
|
for each level in profileSupport["h264"][profile]
|
|
levelFloat = level.ToFloat()
|
|
if levelFloat > h264LevelSupported
|
|
h264LevelSupported = levelFloat
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
' convert to string
|
|
h264LevelString = h264LevelSupported.ToStr()
|
|
' remove decimals
|
|
h264LevelString = removeDecimals(h264LevelString)
|
|
|
|
h264ProfileArray = {
|
|
"Type": "Video",
|
|
"Codec": "h264",
|
|
"Conditions": [
|
|
{
|
|
"Condition": "NotEquals",
|
|
"Property": "IsAnamorphic",
|
|
"Value": "true",
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoProfile",
|
|
"Value": h264AssProfiles.Keys().join("|"),
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoRangeType",
|
|
"Value": h264VideoRangeTypes,
|
|
"IsRequired": false
|
|
}
|
|
|
|
]
|
|
}
|
|
|
|
' check user setting before adding video level restrictions
|
|
if not globalUserSettings["playback.tryDirect.h264ProfileLevel"]
|
|
h264ProfileArray.Conditions.push({
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoLevel",
|
|
"Value": h264LevelString,
|
|
"IsRequired": false
|
|
})
|
|
end if
|
|
|
|
' set max resolution
|
|
if globalUserSettings["playback.resolution.mode"] = "everything" and maxResSetting <> "off"
|
|
h264ProfileArray.Conditions.push(maxHeightArray)
|
|
h264ProfileArray.Conditions.push(maxWidthArray)
|
|
end if
|
|
|
|
' set bitrate restrictions based on user settings
|
|
bitRateArray = GetBitRateLimit("h264")
|
|
if bitRateArray.count() > 0
|
|
h264ProfileArray.Conditions.push(bitRateArray)
|
|
end if
|
|
|
|
codecProfiles.push(h264ProfileArray)
|
|
|
|
' MPEG2
|
|
' NOTE: the mpeg2 levels are being saved in the profileSupport array as if they were profiles
|
|
if globalUserSettings["playback.mpeg2"]
|
|
mpeg2Levels = []
|
|
for each level in profileSupport["mpeg2"]
|
|
if not arrayHasValue(mpeg2Levels, level)
|
|
mpeg2Levels.push(level)
|
|
end if
|
|
end for
|
|
|
|
mpeg2ProfileArray = {
|
|
"Type": "Video",
|
|
"Codec": "mpeg2",
|
|
"Conditions": [
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoLevel",
|
|
"Value": mpeg2Levels.join("|"),
|
|
"IsRequired": false
|
|
}
|
|
]
|
|
}
|
|
|
|
' set max resolution
|
|
if globalUserSettings["playback.resolution.mode"] = "everything" and maxResSetting <> "off"
|
|
mpeg2ProfileArray.Conditions.push(maxHeightArray)
|
|
mpeg2ProfileArray.Conditions.push(maxWidthArray)
|
|
end if
|
|
|
|
' set bitrate restrictions based on user settings
|
|
bitRateArray = GetBitRateLimit("mpeg2")
|
|
if bitRateArray.count() > 0
|
|
mpeg2ProfileArray.Conditions.push(bitRateArray)
|
|
end if
|
|
|
|
codecProfiles.push(mpeg2ProfileArray)
|
|
end if
|
|
|
|
if di.CanDecodeVideo({ Codec: "av1" }).Result
|
|
av1LevelSupported = 0.0
|
|
av1AssProfiles = {}
|
|
for each profile in profileSupport["av1"]
|
|
av1AssProfiles.AddReplace(profile, true)
|
|
for each level in profileSupport["av1"][profile]
|
|
levelFloat = level.ToFloat()
|
|
if levelFloat > av1LevelSupported
|
|
av1LevelSupported = levelFloat
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
av1ProfileArray = {
|
|
"Type": "Video",
|
|
"Codec": "av1",
|
|
"Conditions": [
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoProfile",
|
|
"Value": av1AssProfiles.Keys().join("|"),
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoRangeType",
|
|
"Value": av1VideoRangeTypes,
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoLevel",
|
|
"Value": (120 * av1LevelSupported).ToStr(),
|
|
"IsRequired": false
|
|
}
|
|
]
|
|
}
|
|
|
|
' set max resolution
|
|
if globalUserSettings["playback.resolution.mode"] = "everything" and maxResSetting <> "off"
|
|
av1ProfileArray.Conditions.push(maxHeightArray)
|
|
av1ProfileArray.Conditions.push(maxWidthArray)
|
|
end if
|
|
|
|
' set bitrate restrictions based on user settings
|
|
bitRateArray = GetBitRateLimit("av1")
|
|
if bitRateArray.count() > 0
|
|
av1ProfileArray.Conditions.push(bitRateArray)
|
|
end if
|
|
|
|
codecProfiles.push(av1ProfileArray)
|
|
end if
|
|
|
|
if not globalUserSettings["playback.compatibility.disablehevc"] and di.CanDecodeVideo({ Codec: "hevc" }).Result
|
|
hevcLevelSupported = 0.0
|
|
hevcAssProfiles = {}
|
|
|
|
for each profile in profileSupport["hevc"]
|
|
hevcAssProfiles.AddReplace(profile, true)
|
|
for each level in profileSupport["hevc"][profile]
|
|
levelFloat = level.ToFloat()
|
|
if levelFloat > hevcLevelSupported
|
|
hevcLevelSupported = levelFloat
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
hevcLevelString = "120"
|
|
if hevcLevelSupported = 5.1
|
|
hevcLevelString = "153"
|
|
end if
|
|
|
|
hevcProfileArray = {
|
|
"Type": "Video",
|
|
"Codec": "hevc",
|
|
"Conditions": [
|
|
{
|
|
"Condition": "NotEquals",
|
|
"Property": "IsAnamorphic",
|
|
"Value": "true",
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoProfile",
|
|
"Value": profileSupport["hevc"].Keys().join("|"),
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoRangeType",
|
|
"Value": hevcVideoRangeTypes,
|
|
"IsRequired": false
|
|
}
|
|
]
|
|
}
|
|
|
|
' check user setting before adding VideoLevel restrictions
|
|
if not globalUserSettings["playback.tryDirect.hevcProfileLevel"]
|
|
hevcProfileArray.Conditions.push({
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoLevel",
|
|
"Value": hevcLevelString,
|
|
"IsRequired": false
|
|
})
|
|
end if
|
|
|
|
' set max resolution
|
|
if globalUserSettings["playback.resolution.mode"] = "everything" and maxResSetting <> "off"
|
|
hevcProfileArray.Conditions.push(maxHeightArray)
|
|
hevcProfileArray.Conditions.push(maxWidthArray)
|
|
end if
|
|
|
|
' set bitrate restrictions based on user settings
|
|
bitRateArray = GetBitRateLimit("h265")
|
|
if bitRateArray.count() > 0
|
|
hevcProfileArray.Conditions.push(bitRateArray)
|
|
end if
|
|
|
|
codecProfiles.push(hevcProfileArray)
|
|
end if
|
|
|
|
if di.CanDecodeVideo({ Codec: "vp9" }).Result
|
|
vp9Profiles = []
|
|
vp9LevelSupported = 0.0
|
|
|
|
for each profile in profileSupport["vp9"]
|
|
vp9Profiles.push(profile)
|
|
for each level in profileSupport["vp9"][profile]
|
|
levelFloat = level.ToFloat()
|
|
if levelFloat > vp9LevelSupported
|
|
vp9LevelSupported = levelFloat
|
|
end if
|
|
end for
|
|
end for
|
|
|
|
vp9LevelString = "120"
|
|
if vp9LevelSupported = 5.1
|
|
vp9LevelString = "153"
|
|
end if
|
|
|
|
vp9ProfileArray = {
|
|
"Type": "Video",
|
|
"Codec": "vp9",
|
|
"Conditions": [
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoProfile",
|
|
"Value": vp9Profiles.join("|"),
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "EqualsAny",
|
|
"Property": "VideoRangeType",
|
|
"Value": vp9VideoRangeTypes,
|
|
"IsRequired": false
|
|
},
|
|
{
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoLevel",
|
|
"Value": vp9LevelString,
|
|
"IsRequired": false
|
|
}
|
|
]
|
|
}
|
|
|
|
' set max resolution
|
|
if globalUserSettings["playback.resolution.mode"] = "everything" and maxResSetting <> "off"
|
|
vp9ProfileArray.Conditions.push(maxHeightArray)
|
|
vp9ProfileArray.Conditions.push(maxWidthArray)
|
|
end if
|
|
|
|
' set bitrate restrictions based on user settings
|
|
bitRateArray = GetBitRateLimit("vp9")
|
|
if bitRateArray.count() > 0
|
|
vp9ProfileArray.Conditions.push(bitRateArray)
|
|
end if
|
|
|
|
codecProfiles.push(vp9ProfileArray)
|
|
end if
|
|
|
|
return codecProfiles
|
|
end function
|
|
|
|
function getSubtitleProfiles() as object
|
|
subtitleProfiles = []
|
|
|
|
subtitleProfiles.push({
|
|
"Format": "vtt",
|
|
"Method": "External"
|
|
})
|
|
subtitleProfiles.push({
|
|
"Format": "srt",
|
|
"Method": "External"
|
|
})
|
|
subtitleProfiles.push({
|
|
"Format": "ttml",
|
|
"Method": "External"
|
|
})
|
|
subtitleProfiles.push({
|
|
"Format": "sub",
|
|
"Method": "External"
|
|
})
|
|
|
|
return subtitleProfiles
|
|
end function
|
|
|
|
function GetBitRateLimit(codec as string) as object
|
|
globalUserSettings = m.global.session.user.settings
|
|
if globalUserSettings["playback.bitrate.maxlimited"]
|
|
userSetLimit = globalUserSettings["playback.bitrate.limit"].ToInt()
|
|
if isValid(userSetLimit) and type(userSetLimit) = "Integer" and userSetLimit > 0
|
|
userSetLimit *= 1000000
|
|
return {
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoBitrate",
|
|
"Value": userSetLimit.ToStr(),
|
|
"IsRequired": true
|
|
}
|
|
else
|
|
codec = Lcase(codec)
|
|
' Some repeated values (e.g. same "40mbps" for several codecs)
|
|
' but this makes it easy to update in the future if the bitrates start to deviate.
|
|
if codec = "h264"
|
|
' Roku only supports h264 up to 10Mpbs
|
|
return {
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoBitrate",
|
|
"Value": "10000000",
|
|
"IsRequired": true
|
|
}
|
|
else if codec = "av1"
|
|
' Roku only supports AV1 up to 40Mpbs
|
|
return {
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoBitrate",
|
|
"Value": "40000000",
|
|
"IsRequired": true
|
|
}
|
|
else if codec = "h265"
|
|
' Roku only supports h265 up to 40Mpbs
|
|
return {
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoBitrate",
|
|
"Value": "40000000",
|
|
"IsRequired": true
|
|
}
|
|
else if codec = "vp9"
|
|
' Roku only supports VP9 up to 40Mpbs
|
|
return {
|
|
"Condition": "LessThanEqual",
|
|
"Property": "VideoBitrate",
|
|
"Value": "40000000",
|
|
"IsRequired": true
|
|
}
|
|
end if
|
|
end if
|
|
end if
|
|
return {}
|
|
end function
|
|
|
|
function getMaxHeightArray() as object
|
|
myGlobal = m.global
|
|
|
|
maxResSetting = myGlobal.session.user.settings["playback.resolution.max"]
|
|
if maxResSetting = "off" then return {}
|
|
|
|
maxVideoHeight = maxResSetting
|
|
|
|
if maxResSetting = "auto"
|
|
maxVideoHeight = myGlobal.device.videoHeight
|
|
end if
|
|
|
|
return {
|
|
"Condition": "LessThanEqual",
|
|
"Property": "Height",
|
|
"Value": maxVideoHeight,
|
|
"IsRequired": true
|
|
}
|
|
end function
|
|
|
|
function getMaxWidthArray() as object
|
|
myGlobal = m.global
|
|
|
|
maxResSetting = myGlobal.session.user.settings["playback.resolution.max"]
|
|
if maxResSetting = "off" then return {}
|
|
|
|
maxVideoWidth = invalid
|
|
|
|
if maxResSetting = "auto"
|
|
maxVideoWidth = myGlobal.device.videoWidth
|
|
else if maxResSetting = "360"
|
|
maxVideoWidth = "480"
|
|
else if maxResSetting = "480"
|
|
maxVideoWidth = "640"
|
|
else if maxResSetting = "720"
|
|
maxVideoWidth = "1280"
|
|
else if maxResSetting = "1080"
|
|
maxVideoWidth = "1920"
|
|
else if maxResSetting = "2160"
|
|
maxVideoWidth = "3840"
|
|
else if maxResSetting = "4320"
|
|
maxVideoWidth = "7680"
|
|
end if
|
|
|
|
return {
|
|
"Condition": "LessThanEqual",
|
|
"Property": "Width",
|
|
"Value": maxVideoWidth,
|
|
"IsRequired": true
|
|
}
|
|
end function
|
|
|
|
' Recieves and returns an assArray of supported profiles and levels for each video codec
|
|
function updateProfileArray(profileArray as object, videoCodec as string, videoProfile as string, profileLevel = "" as string) as object
|
|
' validate params
|
|
if profileArray = invalid then return {}
|
|
if videoCodec = "" or videoProfile = "" then return profileArray
|
|
|
|
if profileArray[videoCodec] = invalid
|
|
profileArray[videoCodec] = {}
|
|
end if
|
|
|
|
if profileArray[videoCodec][videoProfile] = invalid
|
|
profileArray[videoCodec][videoProfile] = {}
|
|
end if
|
|
|
|
' add profileLevel if a value was provided
|
|
if profileLevel <> ""
|
|
if profileArray[videoCodec][videoProfile][profileLevel] = invalid
|
|
profileArray[videoCodec][videoProfile].AddReplace(profileLevel, true)
|
|
end if
|
|
end if
|
|
|
|
return profileArray
|
|
end function
|
|
|
|
' Remove all decimals from a string
|
|
function removeDecimals(value as string) as string
|
|
r = CreateObject("roRegex", "\.", "")
|
|
value = r.ReplaceAll(value, "")
|
|
return value
|
|
end function
|
|
|
|
' Print out the deviceProfile for debugging
|
|
sub printDeviceProfile(profile as object)
|
|
print "profile =", profile
|
|
print "profile.DeviceProfile =", profile.DeviceProfile
|
|
print "profile.DeviceProfile.CodecProfiles ="
|
|
for each prof in profile.DeviceProfile.CodecProfiles
|
|
print prof
|
|
for each cond in prof.Conditions
|
|
print cond
|
|
end for
|
|
end for
|
|
print "profile.DeviceProfile.ContainerProfiles =", profile.DeviceProfile.ContainerProfiles
|
|
print "profile.DeviceProfile.DirectPlayProfiles ="
|
|
for each prof in profile.DeviceProfile.DirectPlayProfiles
|
|
print prof
|
|
end for
|
|
print "profile.DeviceProfile.SubtitleProfiles ="
|
|
for each prof in profile.DeviceProfile.SubtitleProfiles
|
|
print prof
|
|
end for
|
|
print "profile.DeviceProfile.TranscodingProfiles ="
|
|
for each prof in profile.DeviceProfile.TranscodingProfiles
|
|
print prof
|
|
if isValid(prof.Conditions)
|
|
for each condition in prof.Conditions
|
|
print condition
|
|
end for
|
|
end if
|
|
end for
|
|
print "profile.PlayableMediaTypes =", profile.PlayableMediaTypes
|
|
print "profile.SupportedCommands =", profile.SupportedCommands
|
|
end sub
|
|
' Takes and returns a comma delimited string of codecs.
|
|
' Moves the preferred codec to the front of the string
|
|
function setPreferredCodec(codecString as string, preferredCodec as string) as string
|
|
if preferredCodec = "" then return ""
|
|
if codecString = "" then return preferredCodec
|
|
|
|
preferredCodecSize = Len(preferredCodec)
|
|
|
|
' is the codec already in front?
|
|
if Left(codecString, preferredCodecSize) = preferredCodec
|
|
return codecString
|
|
else
|
|
' convert string to array
|
|
codecArray = codecString.Split(",")
|
|
' remove preferred codec from array
|
|
newArray = []
|
|
for each codec in codecArray
|
|
if codec <> preferredCodec
|
|
newArray.push(codec)
|
|
end if
|
|
end for
|
|
' convert newArray to string
|
|
newCodecString = newArray.Join(",")
|
|
' add preferred codec to front of newCodecString
|
|
newCodecString = preferredCodec + "," + newCodecString
|
|
|
|
return newCodecString
|
|
end if
|
|
end function
|
|
|
|
' does the connected display support playing 4k video?
|
|
function canPlay4k() as boolean
|
|
deviceInfo = CreateObject("roDeviceInfo")
|
|
hdmiStatus = CreateObject("roHdmiStatus")
|
|
|
|
' Check if the output mode is 2160p or higher
|
|
maxVideoHeight = m.global.device.videoHeight
|
|
if maxVideoHeight = invalid then return false
|
|
if maxVideoHeight.ToInt() < 2160
|
|
return false
|
|
end if
|
|
|
|
' Check if HDCP 2.2 is enabled, skip check for TVs
|
|
if deviceInfo.GetModelType() = "STB" and hdmiStatus.IsHdcpActive("2.2") <> true
|
|
return false
|
|
end if
|
|
|
|
' Check if the Roku player can decode 4K 60fps HEVC streams
|
|
if deviceInfo.CanDecodeVideo({ Codec: "hevc", Profile: "main", Level: "5.1" }).result <> true
|
|
return false
|
|
end if
|
|
|
|
return true
|
|
end function
|