Merge pull request #881 from 1hitsong/musicGenreView
Update music library view & add genre view
This commit is contained in:
commit
e045d378de
|
@ -187,12 +187,20 @@ sub loadItems()
|
||||||
tmp = CreateObject("roSGNode", "MusicArtistData")
|
tmp = CreateObject("roSGNode", "MusicArtistData")
|
||||||
else if item.Type = "Audio"
|
else if item.Type = "Audio"
|
||||||
tmp = CreateObject("roSGNode", "MusicSongData")
|
tmp = CreateObject("roSGNode", "MusicSongData")
|
||||||
|
else if item.Type = "MusicGenre"
|
||||||
|
tmp = CreateObject("roSGNode", "FolderData")
|
||||||
|
tmp.title = item.name
|
||||||
|
tmp.parentFolder = m.top.itemId
|
||||||
|
tmp.json = item
|
||||||
|
tmp.type = "Folder"
|
||||||
|
tmp.posterUrl = api_API().items.getimageurl(item.id, "primary", 0, { "maxHeight": 280, "maxWidth": 280, "quality": "90" })
|
||||||
|
|
||||||
else
|
else
|
||||||
print "[LoadItems] Unknown Type: " item.Type
|
print "[LoadItems] Unknown Type: " item.Type
|
||||||
end if
|
end if
|
||||||
|
|
||||||
if tmp <> invalid
|
if tmp <> invalid
|
||||||
if item.Type <> "Genre"
|
if item.Type <> "Genre" and item.Type <> "MusicGenre"
|
||||||
tmp.parentFolder = m.top.itemId
|
tmp.parentFolder = m.top.itemId
|
||||||
tmp.json = item
|
tmp.json = item
|
||||||
if item.UserData <> invalid and item.UserData.isFavorite <> invalid
|
if item.UserData <> invalid and item.UserData.isFavorite <> invalid
|
||||||
|
|
48
components/ItemGrid/MusicArtistGridItem.brs
Normal file
48
components/ItemGrid/MusicArtistGridItem.brs
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
sub init()
|
||||||
|
m.itemPoster = m.top.findNode("itemPoster")
|
||||||
|
m.posterText = m.top.findNode("posterText")
|
||||||
|
m.posterText.font.size = 30
|
||||||
|
m.backdrop = m.top.findNode("backdrop")
|
||||||
|
|
||||||
|
m.itemPoster.observeField("loadStatus", "onPosterLoadStatusChanged")
|
||||||
|
|
||||||
|
'Parent is MarkupGrid and it's parent is the ItemGrid
|
||||||
|
m.topParent = m.top.GetParent().GetParent()
|
||||||
|
|
||||||
|
'Get the imageDisplayMode for these grid items
|
||||||
|
if m.topParent.imageDisplayMode <> invalid
|
||||||
|
m.itemPoster.loadDisplayMode = m.topParent.imageDisplayMode
|
||||||
|
end if
|
||||||
|
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub itemContentChanged()
|
||||||
|
m.backdrop.blendColor = "#101010"
|
||||||
|
|
||||||
|
itemData = m.top.itemContent
|
||||||
|
|
||||||
|
if not isValid(itemData) then return
|
||||||
|
|
||||||
|
if LCase(itemData.type) = "musicalbum"
|
||||||
|
m.backdrop.uri = "pkg:/images/icons/album.png"
|
||||||
|
else if LCase(itemData.type) = "musicartist"
|
||||||
|
m.backdrop.uri = "pkg:/images/missingArtist.png"
|
||||||
|
else if LCase(itemData.json.type) = "musicgenre"
|
||||||
|
m.backdrop.uri = "pkg:/images/icons/musicFolder.png"
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.itemPoster.uri = itemData.PosterUrl
|
||||||
|
m.posterText.text = itemData.title
|
||||||
|
|
||||||
|
'If Poster not loaded, ensure "blue box" is shown until loaded
|
||||||
|
if m.itemPoster.loadStatus <> "ready"
|
||||||
|
m.backdrop.visible = true
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'Hide backdrop and text when poster loaded
|
||||||
|
sub onPosterLoadStatusChanged()
|
||||||
|
if m.itemPoster.loadStatus = "ready"
|
||||||
|
m.backdrop.visible = false
|
||||||
|
end if
|
||||||
|
end sub
|
17
components/ItemGrid/MusicArtistGridItem.xml
Normal file
17
components/ItemGrid/MusicArtistGridItem.xml
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8" ?>
|
||||||
|
<component name="MusicArtistGridItem" extends="Group">
|
||||||
|
<children>
|
||||||
|
<Poster id="backdrop" translation="[0,15]" width="280" height="280" loadDisplayMode="scaleToZoom" uri="pkg:/images/white.9.png" />
|
||||||
|
<Poster id="itemPoster" translation="[0,15]" width="280" height="280" loadDisplayMode="scaleToZoom" />
|
||||||
|
<Rectangle id="postTextBackground" height="50" width="270" color="0x000000DD" translation = "[5, 240]">
|
||||||
|
<ScrollingLabel id="posterText" color="#FFFFFF" maxWidth="270" height="50" horizAlign="center" vertAlign="center" />
|
||||||
|
</Rectangle>
|
||||||
|
</children>
|
||||||
|
<interface>
|
||||||
|
<field id="itemContent" type="node" onChange="itemContentChanged" />
|
||||||
|
<field id="itemHasFocus" type="boolean" onChange="focusChanged" />
|
||||||
|
</interface>
|
||||||
|
<script type="text/brightscript" uri="MusicArtistGridItem.brs" />
|
||||||
|
<script type="text/brightscript" uri="pkg:/source/utils/misc.brs" />
|
||||||
|
<script type="text/brightscript" uri="pkg:/source/utils/config.brs" />
|
||||||
|
</component>
|
705
components/ItemGrid/MusicLibraryView.brs
Normal file
705
components/ItemGrid/MusicLibraryView.brs
Normal file
|
@ -0,0 +1,705 @@
|
||||||
|
sub setupNodes()
|
||||||
|
m.options = m.top.findNode("options")
|
||||||
|
m.itemGrid = m.top.findNode("itemGrid")
|
||||||
|
m.voiceBox = m.top.findNode("voiceBox")
|
||||||
|
m.backdrop = m.top.findNode("backdrop")
|
||||||
|
m.newBackdrop = m.top.findNode("backdropTransition")
|
||||||
|
m.emptyText = m.top.findNode("emptyText")
|
||||||
|
m.selectedArtistName = m.top.findNode("selectedArtistName")
|
||||||
|
m.artistLogo = m.top.findNode("artistLogo")
|
||||||
|
m.swapAnimation = m.top.findNode("backroundSwapAnimation")
|
||||||
|
m.spinner = m.top.findNode("spinner")
|
||||||
|
m.Alpha = m.top.findNode("AlphaMenu")
|
||||||
|
m.AlphaSelected = m.top.findNode("AlphaSelected")
|
||||||
|
m.micButton = m.top.findNode("micButton")
|
||||||
|
m.micButtonText = m.top.findNode("micButtonText")
|
||||||
|
m.overhang = m.top.getScene().findNode("overhang")
|
||||||
|
m.genreList = m.top.findNode("genrelist")
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub init()
|
||||||
|
setupNodes()
|
||||||
|
|
||||||
|
m.overhang.isVisible = false
|
||||||
|
|
||||||
|
m.showItemCount = get_user_setting("itemgrid.showItemCount") = "true"
|
||||||
|
|
||||||
|
m.swapAnimation.observeField("state", "swapDone")
|
||||||
|
|
||||||
|
m.loadedRows = 0
|
||||||
|
m.loadedItems = 0
|
||||||
|
|
||||||
|
m.data = CreateObject("roSGNode", "ContentNode")
|
||||||
|
|
||||||
|
m.itemGrid.content = m.data
|
||||||
|
|
||||||
|
m.genreData = CreateObject("roSGNode", "ContentNode")
|
||||||
|
m.genreList.observeField("itemSelected", "onGenreItemSelected")
|
||||||
|
m.genreList.observeField("itemFocused", "onGenreItemFocused")
|
||||||
|
m.genreList.content = m.genreData
|
||||||
|
|
||||||
|
m.itemGrid.observeField("itemFocused", "onItemFocused")
|
||||||
|
m.itemGrid.observeField("itemSelected", "onItemSelected")
|
||||||
|
m.itemGrid.observeField("alphaSelected", "onItemalphaSelected")
|
||||||
|
|
||||||
|
'Voice filter setup
|
||||||
|
m.voiceBox.voiceEnabled = true
|
||||||
|
m.voiceBox.active = true
|
||||||
|
m.voiceBox.observeField("text", "onvoiceFilter")
|
||||||
|
'set voice help text
|
||||||
|
m.voiceBox.hintText = tr("Use voice remote to search")
|
||||||
|
|
||||||
|
'backdrop
|
||||||
|
m.newBackdrop.observeField("loadStatus", "newBGLoaded")
|
||||||
|
|
||||||
|
'Background Image Queued for loading
|
||||||
|
m.queuedBGUri = ""
|
||||||
|
|
||||||
|
'Item sort - maybe load defaults from user prefs?
|
||||||
|
m.sortField = "SortName"
|
||||||
|
m.sortAscending = true
|
||||||
|
|
||||||
|
m.filter = "All"
|
||||||
|
m.favorite = "Favorite"
|
||||||
|
|
||||||
|
m.loadItemsTask = createObject("roSGNode", "LoadItemsTask2")
|
||||||
|
m.loadLogoTask = createObject("roSGNode", "LoadItemsTask2")
|
||||||
|
|
||||||
|
'set inital counts for overhang before content is loaded.
|
||||||
|
m.loadItemsTask.totalRecordCount = 0
|
||||||
|
|
||||||
|
m.spinner.visible = true
|
||||||
|
|
||||||
|
'Get reset folder setting
|
||||||
|
m.resetGrid = get_user_setting("itemgrid.reset") = "true"
|
||||||
|
|
||||||
|
'Check if device has voice remote
|
||||||
|
devinfo = CreateObject("roDeviceInfo")
|
||||||
|
|
||||||
|
'Hide voice search if device does not have voice remote
|
||||||
|
if devinfo.HasFeature("voice_remote") = false
|
||||||
|
m.micButton.visible = false
|
||||||
|
m.micButtonText.visible = false
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub OnScreenHidden()
|
||||||
|
if not m.overhang.isVisible
|
||||||
|
m.overhang.disableMoveAnimation = true
|
||||||
|
m.overhang.isVisible = true
|
||||||
|
m.overhang.disableMoveAnimation = false
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub OnScreenShown()
|
||||||
|
m.overhang.isVisible = false
|
||||||
|
|
||||||
|
if m.top.lastFocus <> invalid
|
||||||
|
m.top.lastFocus.setFocus(true)
|
||||||
|
else
|
||||||
|
m.top.setFocus(true)
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Load initial set of Data
|
||||||
|
sub loadInitialItems()
|
||||||
|
m.loadItemsTask.control = "stop"
|
||||||
|
m.spinner.visible = true
|
||||||
|
|
||||||
|
if LCase(m.top.parentItem.json.Type) = "collectionfolder"
|
||||||
|
m.top.HomeLibraryItem = m.top.parentItem.Id
|
||||||
|
end if
|
||||||
|
|
||||||
|
if m.top.parentItem.backdropUrl <> invalid
|
||||||
|
SetBackground(m.top.parentItem.backdropUrl)
|
||||||
|
else
|
||||||
|
SetBackground("")
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.sortField = get_user_setting("display." + m.top.parentItem.Id + ".sortField")
|
||||||
|
sortAscendingStr = get_user_setting("display." + m.top.parentItem.Id + ".sortAscending")
|
||||||
|
m.filter = get_user_setting("display." + m.top.parentItem.Id + ".filter")
|
||||||
|
m.view = get_user_setting("display." + m.top.parentItem.Id + ".landing")
|
||||||
|
|
||||||
|
if not isValid(m.sortField) then m.sortField = "SortName"
|
||||||
|
if not isValid(m.filter) then m.filter = "All"
|
||||||
|
if not isValid(m.view) then m.view = "Artists"
|
||||||
|
|
||||||
|
if sortAscendingStr = invalid or LCase(sortAscendingStr) = "true"
|
||||||
|
m.sortAscending = true
|
||||||
|
else
|
||||||
|
m.sortAscending = false
|
||||||
|
end if
|
||||||
|
|
||||||
|
if LCase(m.top.parentItem.json.type) = "musicgenre"
|
||||||
|
m.itemGrid.translation = "[96, 60]"
|
||||||
|
m.loadItemsTask.itemType = "MusicAlbum"
|
||||||
|
m.loadItemsTask.recursive = true
|
||||||
|
m.loadItemsTask.genreIds = m.top.parentItem.id
|
||||||
|
m.loadItemsTask.itemId = m.top.parentItem.parentFolder
|
||||||
|
else if LCase(m.view) = "artists" or LCase(m.options.view) = "artists"
|
||||||
|
m.loadItemsTask.genreIds = ""
|
||||||
|
else
|
||||||
|
m.loadItemsTask.itemId = m.top.parentItem.Id
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.loadItemsTask.nameStartsWith = m.top.alphaSelected
|
||||||
|
m.loadItemsTask.searchTerm = m.voiceBox.text
|
||||||
|
m.emptyText.visible = false
|
||||||
|
m.loadItemsTask.sortField = m.sortField
|
||||||
|
m.loadItemsTask.sortAscending = m.sortAscending
|
||||||
|
m.loadItemsTask.filter = m.filter
|
||||||
|
m.loadItemsTask.startIndex = 0
|
||||||
|
|
||||||
|
' Load Item Types
|
||||||
|
if getCollectionType() = "music"
|
||||||
|
m.loadItemsTask.itemType = "MusicArtist"
|
||||||
|
m.loadItemsTask.itemId = m.top.parentItem.Id
|
||||||
|
end if
|
||||||
|
|
||||||
|
' By default we load Artists
|
||||||
|
m.loadItemsTask.view = "Artists"
|
||||||
|
m.itemGrid.translation = "[96, 420]"
|
||||||
|
m.itemGrid.numRows = "3"
|
||||||
|
|
||||||
|
if LCase(m.options.view) = "albums" or LCase(m.view) = "albums"
|
||||||
|
m.itemGrid.translation = "[96, 60]"
|
||||||
|
m.itemGrid.numRows = "4"
|
||||||
|
m.loadItemsTask.itemType = "MusicAlbum"
|
||||||
|
m.top.imageDisplayMode = "scaleToFit"
|
||||||
|
else if LCase(m.options.view) = "genres" or LCase(m.view) = "genres"
|
||||||
|
m.loadItemsTask.itemType = ""
|
||||||
|
m.loadItemsTask.recursive = true
|
||||||
|
m.loadItemsTask.view = "Genres"
|
||||||
|
m.artistLogo.visible = false
|
||||||
|
m.selectedArtistName.visible = false
|
||||||
|
end if
|
||||||
|
|
||||||
|
if LCase(m.top.parentItem.json.type) = "musicgenre"
|
||||||
|
m.itemGrid.translation = "[96, 60]"
|
||||||
|
m.itemGrid.numRows = "4"
|
||||||
|
m.artistLogo.visible = false
|
||||||
|
m.selectedArtistName.visible = false
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.loadItemsTask.observeField("content", "ItemDataLoaded")
|
||||||
|
m.spinner.visible = true
|
||||||
|
m.loadItemsTask.control = "RUN"
|
||||||
|
SetUpOptions()
|
||||||
|
end sub
|
||||||
|
|
||||||
|
' Set Music view, sort, and filter options
|
||||||
|
sub setMusicOptions(options)
|
||||||
|
|
||||||
|
options.views = [
|
||||||
|
{ "Title": tr("Artists"), "Name": "Artists" },
|
||||||
|
{ "Title": tr("Albums"), "Name": "Albums" },
|
||||||
|
{ "Title": tr("Genres"), "Name": "Genres" }
|
||||||
|
]
|
||||||
|
|
||||||
|
if LCase(m.top.parentItem.json.type) = "musicgenre"
|
||||||
|
options.views = [
|
||||||
|
{ "Title": tr("Albums"), "Name": "Albums" }
|
||||||
|
]
|
||||||
|
end if
|
||||||
|
|
||||||
|
options.sort = [
|
||||||
|
{ "Title": tr("TITLE"), "Name": "SortName" },
|
||||||
|
{ "Title": tr("DATE_ADDED"), "Name": "DateCreated" },
|
||||||
|
{ "Title": tr("DATE_PLAYED"), "Name": "DatePlayed" },
|
||||||
|
{ "Title": tr("RELEASE_DATE"), "Name": "PremiereDate" },
|
||||||
|
]
|
||||||
|
|
||||||
|
options.filter = [
|
||||||
|
{ "Title": tr("All"), "Name": "All" },
|
||||||
|
{ "Title": tr("Favorites"), "Name": "Favorites" }
|
||||||
|
]
|
||||||
|
|
||||||
|
if LCase(m.options.view) = "genres" or LCase(m.view) = "genres"
|
||||||
|
options.sort = [
|
||||||
|
{ "Title": tr("TITLE"), "Name": "SortName" },
|
||||||
|
]
|
||||||
|
options.filter = []
|
||||||
|
end if
|
||||||
|
|
||||||
|
if LCase(m.options.view) = "albums" or LCase(m.view) = "albums"
|
||||||
|
options.sort = [
|
||||||
|
{ "Title": tr("TITLE"), "Name": "SortName" },
|
||||||
|
{ "Title": tr("DATE_ADDED"), "Name": "DateCreated" },
|
||||||
|
]
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
' Return parent collection type
|
||||||
|
function getCollectionType() as string
|
||||||
|
if m.top.parentItem.collectionType = invalid
|
||||||
|
return LCase(m.top.parentItem.Type)
|
||||||
|
else
|
||||||
|
return LCase(m.top.parentItem.CollectionType)
|
||||||
|
end if
|
||||||
|
end function
|
||||||
|
|
||||||
|
' Search string array for search value. Return if it's found
|
||||||
|
function inStringArray(array, searchValue) as boolean
|
||||||
|
for each item in array
|
||||||
|
if lcase(item) = lcase(searchValue) then return true
|
||||||
|
end for
|
||||||
|
return false
|
||||||
|
end function
|
||||||
|
|
||||||
|
' Data to display when options button selected
|
||||||
|
sub SetUpOptions()
|
||||||
|
options = {}
|
||||||
|
options.filter = []
|
||||||
|
options.favorite = []
|
||||||
|
|
||||||
|
setMusicOptions(options)
|
||||||
|
|
||||||
|
' Set selected view option
|
||||||
|
for each o in options.views
|
||||||
|
if LCase(o.Name) = LCase(m.view)
|
||||||
|
o.Selected = true
|
||||||
|
o.Ascending = m.sortAscending
|
||||||
|
m.options.view = o.Name
|
||||||
|
end if
|
||||||
|
end for
|
||||||
|
|
||||||
|
' Set selected sort option
|
||||||
|
for each o in options.sort
|
||||||
|
if LCase(o.Name) = LCase(m.sortField)
|
||||||
|
o.Selected = true
|
||||||
|
o.Ascending = m.sortAscending
|
||||||
|
m.options.sortField = o.Name
|
||||||
|
end if
|
||||||
|
end for
|
||||||
|
|
||||||
|
' Set selected filter option
|
||||||
|
for each o in options.filter
|
||||||
|
if LCase(o.Name) = LCase(m.filter)
|
||||||
|
o.Selected = true
|
||||||
|
m.options.filter = o.Name
|
||||||
|
end if
|
||||||
|
end for
|
||||||
|
|
||||||
|
m.options.options = options
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
' Logo Image Loaded Event Handler
|
||||||
|
sub LogoImageLoaded(msg)
|
||||||
|
data = msg.GetData()
|
||||||
|
m.loadLogoTask.unobserveField("content")
|
||||||
|
m.loadLogoTask.content = []
|
||||||
|
|
||||||
|
if data.Count() > 0
|
||||||
|
m.artistLogo.uri = data[0]
|
||||||
|
m.artistLogo.visible = true
|
||||||
|
else
|
||||||
|
m.selectedArtistName.visible = true
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Handle loaded data, and add to Grid
|
||||||
|
sub ItemDataLoaded(msg)
|
||||||
|
m.top.alphaActive = false
|
||||||
|
itemData = msg.GetData()
|
||||||
|
m.loadItemsTask.unobserveField("content")
|
||||||
|
m.loadItemsTask.content = []
|
||||||
|
|
||||||
|
if itemData = invalid
|
||||||
|
m.Loading = false
|
||||||
|
return
|
||||||
|
end if
|
||||||
|
|
||||||
|
if LCase(m.loadItemsTask.view) = "genres"
|
||||||
|
for each item in itemData
|
||||||
|
m.genreData.appendChild(item)
|
||||||
|
end for
|
||||||
|
|
||||||
|
m.itemGrid.opacity = "0"
|
||||||
|
m.genreList.opacity = "1"
|
||||||
|
|
||||||
|
m.itemGrid.setFocus(false)
|
||||||
|
m.genreList.setFocus(true)
|
||||||
|
|
||||||
|
m.loadedItems = m.genreList.content.getChildCount()
|
||||||
|
m.loadedRows = m.loadedItems / m.genreList.numColumns
|
||||||
|
|
||||||
|
m.loading = false
|
||||||
|
m.spinner.visible = false
|
||||||
|
return
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.itemGrid.opacity = "1"
|
||||||
|
m.genreList.opacity = "0"
|
||||||
|
|
||||||
|
m.itemGrid.setFocus(true)
|
||||||
|
m.genreList.setFocus(false)
|
||||||
|
|
||||||
|
for each item in itemData
|
||||||
|
m.data.appendChild(item)
|
||||||
|
end for
|
||||||
|
|
||||||
|
'Update the stored counts
|
||||||
|
m.loadedItems = m.itemGrid.content.getChildCount()
|
||||||
|
m.loadedRows = m.loadedItems / m.itemGrid.numColumns
|
||||||
|
m.Loading = false
|
||||||
|
'If there are no items to display, show message
|
||||||
|
if m.loadedItems = 0
|
||||||
|
m.emptyText.text = tr("NO_ITEMS").Replace("%1", m.top.parentItem.Type)
|
||||||
|
m.emptyText.visible = true
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.spinner.visible = false
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Set Selected Artist Name
|
||||||
|
sub SetName(artistName as string)
|
||||||
|
m.selectedArtistName.text = artistName
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Set Background Image
|
||||||
|
sub SetBackground(backgroundUri as string)
|
||||||
|
if backgroundUri = ""
|
||||||
|
m.backdrop.opacity = 0
|
||||||
|
end if
|
||||||
|
|
||||||
|
'If a new image is being loaded, or transitioned to, store URL to load next
|
||||||
|
if LCase(m.swapAnimation.state) <> "stopped" or LCase(m.newBackdrop.loadStatus) = "loading"
|
||||||
|
m.queuedBGUri = backgroundUri
|
||||||
|
return
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.newBackdrop.uri = backgroundUri
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Handle new item being focused
|
||||||
|
sub onItemFocused()
|
||||||
|
focusedRow = m.itemGrid.currFocusRow
|
||||||
|
|
||||||
|
itemInt = m.itemGrid.itemFocused
|
||||||
|
|
||||||
|
' If no selected item, set background to parent backdrop
|
||||||
|
if itemInt = -1
|
||||||
|
return
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.artistLogo.visible = false
|
||||||
|
m.selectedArtistName.visible = false
|
||||||
|
|
||||||
|
' Load more data if focus is within last 5 rows, and there are more items to load
|
||||||
|
if focusedRow >= m.loadedRows - 5 and m.loadeditems < m.loadItemsTask.totalRecordCount
|
||||||
|
loadMoreData()
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.selectedFavoriteItem = getItemFocused()
|
||||||
|
|
||||||
|
if LCase(m.options.view) = "albums" or LCase(m.view) = "albums" or LCase(m.top.parentItem.json.type) = "musicgenre"
|
||||||
|
return
|
||||||
|
end if
|
||||||
|
|
||||||
|
itemData = m.selectedFavoriteItem.json
|
||||||
|
|
||||||
|
if isValid(itemData.Name)
|
||||||
|
SetName(itemData.Name)
|
||||||
|
else
|
||||||
|
SetName("")
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.loadLogoTask.itemId = itemData.id
|
||||||
|
m.loadLogoTask.itemType = "LogoImage"
|
||||||
|
m.loadLogoTask.observeField("content", "LogoImageLoaded")
|
||||||
|
m.loadLogoTask.control = "RUN"
|
||||||
|
|
||||||
|
' Set Background to item backdrop
|
||||||
|
SetBackground(m.selectedFavoriteItem.backdropUrl)
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub setFieldText(field, value)
|
||||||
|
node = m.top.findNode(field)
|
||||||
|
if node = invalid or value = invalid then return
|
||||||
|
|
||||||
|
' Handle non strings... Which _shouldn't_ happen, but hey
|
||||||
|
if type(value) = "roInt" or type(value) = "Integer"
|
||||||
|
value = str(value)
|
||||||
|
else if type(value) = "roFloat" or type(value) = "Float"
|
||||||
|
value = str(value)
|
||||||
|
else if type(value) <> "roString" and type(value) <> "String"
|
||||||
|
value = ""
|
||||||
|
end if
|
||||||
|
|
||||||
|
node.text = value
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'When Image Loading Status changes
|
||||||
|
sub newBGLoaded()
|
||||||
|
'If image load was sucessful, start the fade swap
|
||||||
|
if LCase(m.newBackdrop.loadStatus) = "ready"
|
||||||
|
m.swapAnimation.control = "start"
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Swap Complete
|
||||||
|
sub swapDone()
|
||||||
|
if LCase(m.swapAnimation.state) = "stopped"
|
||||||
|
'Set main BG node image and hide transitioning node
|
||||||
|
m.backdrop.uri = m.newBackdrop.uri
|
||||||
|
m.backdrop.opacity = 1
|
||||||
|
m.newBackdrop.opacity = 0
|
||||||
|
|
||||||
|
'If there is another one to load
|
||||||
|
if m.newBackdrop.uri <> m.queuedBGUri and m.queuedBGUri <> ""
|
||||||
|
SetBackground(m.queuedBGUri)
|
||||||
|
m.queuedBGUri = ""
|
||||||
|
end if
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Load next set of items
|
||||||
|
sub loadMoreData()
|
||||||
|
m.spinner.visible = true
|
||||||
|
if m.Loading = true then return
|
||||||
|
m.Loading = true
|
||||||
|
m.loadItemsTask.startIndex = m.loadedItems
|
||||||
|
m.loadItemsTask.observeField("content", "ItemDataLoaded")
|
||||||
|
m.loadItemsTask.control = "RUN"
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Item Selected
|
||||||
|
sub onItemSelected()
|
||||||
|
m.top.selectedItem = m.itemGrid.content.getChild(m.itemGrid.itemSelected)
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Returns Focused Item
|
||||||
|
function getItemFocused()
|
||||||
|
return m.itemGrid.content.getChild(m.itemGrid.itemFocused)
|
||||||
|
end function
|
||||||
|
|
||||||
|
'
|
||||||
|
'Genre Item Selected
|
||||||
|
sub onGenreItemSelected()
|
||||||
|
m.top.selectedItem = m.genreList.content.getChild(m.genreList.itemSelected)
|
||||||
|
end sub
|
||||||
|
|
||||||
|
'
|
||||||
|
'Genre Item Focused
|
||||||
|
sub onGenreItemFocused()
|
||||||
|
focusedRow = m.genreList.currFocusRow
|
||||||
|
|
||||||
|
' Load more data if focus is within last 5 rows, and there are more items to load
|
||||||
|
if focusedRow >= m.loadedRows - 5 and m.loadeditems < m.loadItemsTask.totalRecordCount
|
||||||
|
loadMoreData()
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub onItemalphaSelected()
|
||||||
|
if m.top.alphaSelected <> ""
|
||||||
|
m.loadedRows = 0
|
||||||
|
m.loadedItems = 0
|
||||||
|
|
||||||
|
m.data = CreateObject("roSGNode", "ContentNode")
|
||||||
|
m.itemGrid.content = m.data
|
||||||
|
|
||||||
|
m.genreData = CreateObject("roSGNode", "ContentNode")
|
||||||
|
m.genreList.content = m.genreData
|
||||||
|
|
||||||
|
m.loadItemsTask.searchTerm = ""
|
||||||
|
m.VoiceBox.text = ""
|
||||||
|
m.loadItemsTask.nameStartsWith = m.alpha.itemAlphaSelected
|
||||||
|
m.spinner.visible = true
|
||||||
|
loadInitialItems()
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub onvoiceFilter()
|
||||||
|
if m.VoiceBox.text <> ""
|
||||||
|
m.loadedRows = 0
|
||||||
|
m.loadedItems = 0
|
||||||
|
m.data = CreateObject("roSGNode", "ContentNode")
|
||||||
|
m.itemGrid.content = m.data
|
||||||
|
m.top.alphaSelected = ""
|
||||||
|
m.loadItemsTask.NameStartsWith = " "
|
||||||
|
m.loadItemsTask.searchTerm = m.voiceBox.text
|
||||||
|
m.loadItemsTask.recursive = true
|
||||||
|
m.spinner.visible = true
|
||||||
|
loadInitialItems()
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
|
||||||
|
'
|
||||||
|
'Check if options updated and any reloading required
|
||||||
|
sub optionsClosed()
|
||||||
|
reload = false
|
||||||
|
|
||||||
|
if m.options.sortField <> m.sortField or m.options.sortAscending <> m.sortAscending
|
||||||
|
m.sortField = m.options.sortField
|
||||||
|
m.sortAscending = m.options.sortAscending
|
||||||
|
reload = true
|
||||||
|
|
||||||
|
sortAscendingStr = "true"
|
||||||
|
|
||||||
|
'Store sort settings
|
||||||
|
if not m.sortAscending
|
||||||
|
sortAscendingStr = "false"
|
||||||
|
end if
|
||||||
|
|
||||||
|
set_user_setting("display." + m.top.parentItem.Id + ".sortField", m.sortField)
|
||||||
|
set_user_setting("display." + m.top.parentItem.Id + ".sortAscending", sortAscendingStr)
|
||||||
|
end if
|
||||||
|
|
||||||
|
if m.options.filter <> m.filter
|
||||||
|
m.filter = m.options.filter
|
||||||
|
reload = true
|
||||||
|
set_user_setting("display." + m.top.parentItem.Id + ".filter", m.options.filter)
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.view = get_user_setting("display." + m.top.parentItem.Id + ".landing")
|
||||||
|
|
||||||
|
if m.options.view <> m.view
|
||||||
|
m.view = m.options.view
|
||||||
|
m.top.view = m.view
|
||||||
|
set_user_setting("display." + m.top.parentItem.Id + ".landing", m.view)
|
||||||
|
|
||||||
|
' Reset any filtering or search terms
|
||||||
|
m.top.alphaSelected = ""
|
||||||
|
m.loadItemsTask.NameStartsWith = " "
|
||||||
|
m.loadItemsTask.searchTerm = ""
|
||||||
|
m.filter = "All"
|
||||||
|
m.sortField = "SortName"
|
||||||
|
m.sortAscending = true
|
||||||
|
|
||||||
|
' Reset view to defaults
|
||||||
|
set_user_setting("display." + m.top.parentItem.Id + ".sortField", m.sortField)
|
||||||
|
set_user_setting("display." + m.top.parentItem.Id + ".sortAscending", "true")
|
||||||
|
set_user_setting("display." + m.top.parentItem.Id + ".filter", m.filter)
|
||||||
|
|
||||||
|
reload = true
|
||||||
|
end if
|
||||||
|
|
||||||
|
if reload
|
||||||
|
m.loadedRows = 0
|
||||||
|
m.loadedItems = 0
|
||||||
|
m.data = CreateObject("roSGNode", "ContentNode")
|
||||||
|
m.genreData = CreateObject("roSGNode", "ContentNode")
|
||||||
|
m.itemGrid.content = m.data
|
||||||
|
m.genreList.content = m.genreData
|
||||||
|
loadInitialItems()
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.itemGrid.setFocus(m.itemGrid.opacity = 1)
|
||||||
|
m.genreList.setFocus(m.genreList.opacity = 1)
|
||||||
|
end sub
|
||||||
|
|
||||||
|
sub onChannelSelected(msg)
|
||||||
|
node = msg.getRoSGNode()
|
||||||
|
m.top.lastFocus = lastFocusedChild(node)
|
||||||
|
if node.watchChannel <> invalid
|
||||||
|
' Clone the node when it's reused/update in the TimeGrid it doesn't automatically start playing
|
||||||
|
m.top.selectedItem = node.watchChannel.clone(false)
|
||||||
|
end if
|
||||||
|
end sub
|
||||||
|
|
||||||
|
function onKeyEvent(key as string, press as boolean) as boolean
|
||||||
|
if not press then return false
|
||||||
|
|
||||||
|
if key = "left" and m.voiceBox.isinFocusChain()
|
||||||
|
m.itemGrid.setFocus(m.itemGrid.opacity = 1)
|
||||||
|
m.genreList.setFocus(m.genreList.opacity = 1)
|
||||||
|
m.voiceBox.setFocus(false)
|
||||||
|
end if
|
||||||
|
|
||||||
|
if key = "options"
|
||||||
|
if m.options.visible = true
|
||||||
|
m.options.visible = false
|
||||||
|
m.top.removeChild(m.options)
|
||||||
|
optionsClosed()
|
||||||
|
else
|
||||||
|
|
||||||
|
itemSelected = m.selectedFavoriteItem
|
||||||
|
if itemSelected <> invalid
|
||||||
|
m.options.selectedFavoriteItem = itemSelected
|
||||||
|
end if
|
||||||
|
|
||||||
|
m.options.visible = true
|
||||||
|
m.top.appendChild(m.options)
|
||||||
|
m.options.setFocus(true)
|
||||||
|
end if
|
||||||
|
return true
|
||||||
|
else if key = "back"
|
||||||
|
if m.options.visible = true
|
||||||
|
m.options.visible = false
|
||||||
|
optionsClosed()
|
||||||
|
return true
|
||||||
|
else
|
||||||
|
m.global.sceneManager.callfunc("popScene")
|
||||||
|
m.loadItemsTask.control = "stop"
|
||||||
|
return true
|
||||||
|
end if
|
||||||
|
else if key = "left"
|
||||||
|
if m.itemGrid.isinFocusChain()
|
||||||
|
m.top.alphaActive = true
|
||||||
|
m.itemGrid.setFocus(false)
|
||||||
|
alpha = m.alpha.getChild(0).findNode("Alphamenu")
|
||||||
|
alpha.setFocus(true)
|
||||||
|
return true
|
||||||
|
else if m.genreList.isinFocusChain()
|
||||||
|
m.top.alphaActive = true
|
||||||
|
m.genreList.setFocus(false)
|
||||||
|
alpha = m.alpha.getChild(0).findNode("Alphamenu")
|
||||||
|
alpha.setFocus(true)
|
||||||
|
return true
|
||||||
|
end if
|
||||||
|
|
||||||
|
else if key = "right" and m.Alpha.isinFocusChain()
|
||||||
|
m.top.alphaActive = false
|
||||||
|
m.Alpha.setFocus(false)
|
||||||
|
m.Alpha.visible = true
|
||||||
|
|
||||||
|
m.itemGrid.setFocus(m.itemGrid.opacity = 1)
|
||||||
|
m.genreList.setFocus(m.genreList.opacity = 1)
|
||||||
|
|
||||||
|
return true
|
||||||
|
|
||||||
|
else if key = "replay" and m.itemGrid.isinFocusChain()
|
||||||
|
if m.resetGrid = true
|
||||||
|
m.itemGrid.animateToItem = 0
|
||||||
|
else
|
||||||
|
m.itemGrid.jumpToItem = 0
|
||||||
|
end if
|
||||||
|
|
||||||
|
else if key = "replay" and m.genreList.isinFocusChain()
|
||||||
|
if m.resetGrid = true
|
||||||
|
m.genreList.animateToItem = 0
|
||||||
|
else
|
||||||
|
m.genreList.jumpToItem = 0
|
||||||
|
end if
|
||||||
|
return true
|
||||||
|
end if
|
||||||
|
|
||||||
|
if key = "replay"
|
||||||
|
m.spinner.visible = true
|
||||||
|
m.loadItemsTask.searchTerm = ""
|
||||||
|
m.loadItemsTask.nameStartsWith = ""
|
||||||
|
m.voiceBox.text = ""
|
||||||
|
m.top.alphaSelected = ""
|
||||||
|
m.loadItemsTask.filter = "All"
|
||||||
|
m.filter = "All"
|
||||||
|
m.data = CreateObject("roSGNode", "ContentNode")
|
||||||
|
m.itemGrid.content = m.data
|
||||||
|
loadInitialItems()
|
||||||
|
return true
|
||||||
|
end if
|
||||||
|
|
||||||
|
return false
|
||||||
|
end function
|
48
components/ItemGrid/MusicLibraryView.xml
Normal file
48
components/ItemGrid/MusicLibraryView.xml
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8" ?>
|
||||||
|
<component name="MusicLibraryView" extends="JFScreen">
|
||||||
|
<children>
|
||||||
|
<Rectangle id="screenSaverBackground" width="1920" height="1080" color="#000000" />
|
||||||
|
|
||||||
|
<VoiceTextEditBox id="VoiceBox" visible="true" width = "40" translation = "[52, 120]" />
|
||||||
|
<Rectangle id="VoiceBoxCover" height="240" width="100" color="0x000000ff" translation = "[25, 75]" />
|
||||||
|
|
||||||
|
<maskGroup translation="[820, 0]" id="backgroundMask" maskUri="pkg:/images/backgroundmask.png" maskSize="[1220,445]">
|
||||||
|
<poster id="backdrop" loadDisplayMode="scaleToFill" width="1100" height="450" opacity="1" />
|
||||||
|
<poster id="backdropTransition" loadDisplayMode="scaleToFill" width="1100" height="450" opacity="1" />
|
||||||
|
</maskGroup>
|
||||||
|
|
||||||
|
<Label id="selectedArtistName" visible="false" translation="[120, 40]" wrap="true" font="font:LargeBoldSystemFont" width="850" height="196" horizAlign="left" vertAlign="center" />
|
||||||
|
<Poster id="artistLogo" visible="false" translation="[120, 40]" loadDisplayMode="scaleToFit" width="384" height="196" />
|
||||||
|
|
||||||
|
<MarkupGrid id="itemGrid" itemComponentName="MusicArtistGridItem" numColumns="6" numRows="2" vertFocusAnimationStyle="fixed" itemSize="[280, 280]" itemSpacing="[20, 20]" />
|
||||||
|
<MarkupGrid id="genrelist" itemComponentName="MusicArtistGridItem" numColumns="6" numRows="4" vertFocusAnimationStyle="fixed" translation="[96, 60]" itemSize="[280, 280]" itemSpacing="[20, 20]" opacity="0" />
|
||||||
|
|
||||||
|
<Label id="micButtonText" font="font:SmallSystemFont" visible="false" />
|
||||||
|
<Button id = "micButton" maxWidth = "20" translation = "[20, 120]" iconUri = "pkg:/images/icons/mic_icon.png"/>
|
||||||
|
<Label translation="[0,540]" id="emptyText" font="font:LargeSystemFont" width="1910" horizAlign="center" vertAlign="center" height="64" visible="false" />
|
||||||
|
<ItemGridOptions id="options" visible="false" />
|
||||||
|
<Spinner id="spinner" translation="[900, 450]" />
|
||||||
|
<Animation id="backroundSwapAnimation" duration="1" repeat="false" easeFunction="linear">
|
||||||
|
<FloatFieldInterpolator id = "fadeinLoading" key="[0.0, 1.0]" keyValue="[ 0.00, 1.00 ]" fieldToInterp="backdropTransition.opacity" />
|
||||||
|
<FloatFieldInterpolator id = "fadeoutLoaded" key="[0.0, 1.0]" keyValue="[ 1.00, 0.00 ]" fieldToInterp="backdrop.opacity" />
|
||||||
|
</Animation>
|
||||||
|
<Alpha id="AlphaMenu" />
|
||||||
|
</children>
|
||||||
|
<interface>
|
||||||
|
<field id="HomeLibraryItem" type="string"/>
|
||||||
|
<field id="View" type="string"/>
|
||||||
|
<field id="parentItem" type="node" onChange="loadInitialItems" />
|
||||||
|
<field id="selectedItem" type="node" alwaysNotify="true" />
|
||||||
|
<field id="quickPlayNode" type="node" alwaysNotify="true" />
|
||||||
|
<field id="imageDisplayMode" type="string" value="scaleToZoom" />
|
||||||
|
<field id="AlphaSelected" type="string" alias="AlphaMenu.itemAlphaSelected" alwaysNotify="true" onChange="onItemAlphaSelected" />
|
||||||
|
<field id="alphaActive" type="boolean" value="false" />
|
||||||
|
<field id="jumpToItem" type="integer" value="" />
|
||||||
|
</interface>
|
||||||
|
<script type="text/brightscript" uri="pkg:/source/utils/misc.brs" />
|
||||||
|
<script type="text/brightscript" uri="pkg:/source/utils/config.brs" />
|
||||||
|
<script type="text/brightscript" uri="pkg:/source/api/baserequest.brs" />
|
||||||
|
<script type="text/brightscript" uri="pkg:/source/api/Image.brs" />
|
||||||
|
<script type="text/brightscript" uri="pkg:/source/utils/deviceCapabilities.brs" />
|
||||||
|
<script type="text/brightscript" uri="MusicLibraryView.brs" />
|
||||||
|
</component>
|
BIN
images/icons/musicFolder.png
Normal file
BIN
images/icons/musicFolder.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.1 KiB |
|
@ -106,12 +106,16 @@ sub Main (args as dynamic) as void
|
||||||
if selectedItem.type = "CollectionFolder"
|
if selectedItem.type = "CollectionFolder"
|
||||||
if selectedItem.collectionType = "movies"
|
if selectedItem.collectionType = "movies"
|
||||||
group = CreateMovieLibraryView(selectedItem)
|
group = CreateMovieLibraryView(selectedItem)
|
||||||
|
else if selectedItem.collectionType = "music"
|
||||||
|
group = CreateMusicLibraryView(selectedItem)
|
||||||
else
|
else
|
||||||
group = CreateItemGrid(selectedItem)
|
group = CreateItemGrid(selectedItem)
|
||||||
end if
|
end if
|
||||||
sceneManager.callFunc("pushScene", group)
|
sceneManager.callFunc("pushScene", group)
|
||||||
else if selectedItem.type = "Folder" and selectedItem.json.type = "Genre"
|
else if selectedItem.type = "Folder" and selectedItem.json.type = "Genre"
|
||||||
group = CreateMovieLibraryView(selectedItem)
|
group = CreateMovieLibraryView(selectedItem)
|
||||||
|
else if selectedItem.type = "Folder" and selectedItem.json.type = "MusicGenre"
|
||||||
|
group = CreateMusicLibraryView(selectedItem)
|
||||||
sceneManager.callFunc("pushScene", group)
|
sceneManager.callFunc("pushScene", group)
|
||||||
else if selectedItem.type = "UserView" or selectedItem.type = "Folder" or selectedItem.type = "Channel" or selectedItem.type = "Boxset"
|
else if selectedItem.type = "UserView" or selectedItem.type = "Folder" or selectedItem.type = "Channel" or selectedItem.type = "Boxset"
|
||||||
group = CreateItemGrid(selectedItem)
|
group = CreateItemGrid(selectedItem)
|
||||||
|
|
|
@ -489,6 +489,14 @@ function CreateMovieLibraryView(libraryItem)
|
||||||
return group
|
return group
|
||||||
end function
|
end function
|
||||||
|
|
||||||
|
function CreateMusicLibraryView(libraryItem)
|
||||||
|
group = CreateObject("roSGNode", "MusicLibraryView")
|
||||||
|
group.parentItem = libraryItem
|
||||||
|
group.optionsAvailable = true
|
||||||
|
group.observeField("selectedItem", m.port)
|
||||||
|
return group
|
||||||
|
end function
|
||||||
|
|
||||||
function CreateSearchPage()
|
function CreateSearchPage()
|
||||||
' Search + Results Page
|
' Search + Results Page
|
||||||
group = CreateObject("roSGNode", "searchResults")
|
group = CreateObject("roSGNode", "searchResults")
|
||||||
|
|
Loading…
Reference in New Issue
Block a user