In this lesson we will add a category picker that will trigger a network request to load a JSON file.
Most languages implement an event pattern for asynchronous communication between objects. In SceneGraph the technique is called Field Observers. For network requests, SceneGraph requires the use of a "task". All HTTP activity must be contained in a task, which spawns a new thread. Threading is a little odd in Roku SceneGraph, read more about it here: https://sdkdocs.roku.com/display/sdkdoc/SceneGraph+Threads
OK, let's go:
-
In the
components
directory, create a directory calledscreens
-
Inside
screens
, create 2 files:category_screen.xml
andcategory_screen.brs
. Opencategory_screen.xml
and add this:<?xml version="1.0" encoding="utf-8" ?> <component name="category_screen" extends="Group" > <script type="text/brightscript" uri="pkg:/components/screens/category_screen.brs" /> <children> <!-- color="0x000000" --> <Label translation="[75, 150]" id="env" font="font:LargeBoldSystemFont" text="PICK A CATEGORY:"/> <RadioButtonList id="category_list" translation="[100, 200]" itemSize="[400,48]" /> </children> </component>
In category_screen.brs add the following:
function init() m.category_list=m.top.findNode("category_list") m.category_list.setFocus(true) end function
-
Modify home_scene.brs to route to the new screen components by replacing what is there with the following:
function init() ? "[home_scene] init" m.center_square = m.top.findNode("category_screen") m.center_square.setFocus(true) end function
-
Save and run the app. It's pretty lame, nothing happens and the category list is empty. Go back to
category_screen.xml
and make this change:<RadioButtonList id="category_list" translation="[100, 200]" itemSize="[400,48]" > <ContentNode id = "categories" role = "content" > <ContentNode title = "DRAMA"/> <ContentNode title = "COMEDY"/> <ContentNode title = "HORROR"/> </ContentNode> </RadioButtonList>
Run it now as see the populated list:
All the UI components require a properly constructedContentNode
. Depending on the component, the general rule of thumb is to define a single parent node, then append child nodes to it, all of the same typeContentNode
.
The ContentNode is put to use throughout the Roku environment. Perhaps it's most obvious and important use is as the object that is expected by the Video node when playing content, but as demonstrated here it is also expected as the data container for a List.
Read more here: https://sdkdocs.roku.com/display/sdkdoc/ContentNode -
Next the app needs to handle the user interactions with the list. Go back to
category_screen.brs
and add the following code:function init() m.category_list=m.top.findNode("category_list") m.category_list.setFocus(true) m.category_list.observeField("itemSelected", "onCategorySelected") end function sub onCategorySelected(obj) ? "onCategorySelected field: ";obj.getField() ? "onCategorySelected data: ";obj.getData() end sub
You just wrote a field observer, the bedrock of SceneGraph object communication. Pay attention to the argument in the
onCategorySelected
handler. You can query the provided object for the name of the field (getField
) and the value of the field (getData
).
More info here: https://sdkdocs.roku.com/display/sdkdoc/Handling+Application+Events#HandlingApplicationEvents-HandlingNodeFieldValueChanges
https://sdkdocs.roku.com/display/sdkdoc/roSGNodeEvent -
Run the app, then select a category in the list and click OK. Check the debugger:
onCategorySelected field: itemSelected onCategorySelected data: 0
Notice how all you get is a number? You have to do some gymnastics to get the actual ContentNode value associated with that index. Make the following update to
onCategorySelected
:sub onCategorySelected(obj) ? "onCategorySelected field: ";obj.getField() ? "onCategorySelected data: ";obj.getData() ? "onCategorySelected checkedItem: ";m.category_list.checkedItem ? "onCategorySelected selected ContentNode: "; m.category_list.content.getChild(obj.getData()) end sub
Run it and check the debugger:
onCategorySelected field: itemSelected onCategorySelected data: 0 onCategorySelected checkedItem: 0 onCategorySelected selected ContentNode: <Component: roSGNode> = { change: <Component: roAssociativeArray> focusable: false focusedChild: <Component: roInvalid> id: "" feed_url: "http://10.0.0.42:8888/roku_lessons/drama.json" TITLE: "DRAMA" }
Notice how
obj.getData()
andm.category_list.checkedItem
provide the same value. More importantly, notice how using a lookup on thecontent
field of the list provides the actual selected node.One more thing. Open
components/home_scene.brs
and add this method:function onKeyEvent(key, press) as Boolean ? "[home_scene] onKeyEvent", key, press return false end function
This will log every remote control button press. Now that the app is handling user interaction, this code is handy for debugging inputs.
Read more here: https://sdkdocs.roku.com/pages/viewpage.action?pageId=1608547 -
Here's an opportunity to demonstrate how to extend a component. The list needs an update to store a custom value for our feed urls. To accomplish this, we must extend ContentNode to support a custom field.
In thecomponents
directory, add a directory calledmodels
. Inmodels
, add a file calledcategory_node.xml
and write the following:<?xml version="1.0" encoding="utf-8" ?> <component name="category_node" extends="ContentNode"> <interface> <field id="feed_url" type="string" /> </interface> </component>
This node now inherits all the fields from ContentNode, and supports a new field called
feed_url
. Now we can wire up the selected category to a category feed on the network. NOTICE - ContentNode transforms to category_node at this point. Editcategory_screen.xml
with the following code:<RadioButtonList id="category_list" translation="[100, 200]" itemSize="[400,48]" > <ContentNode id = "categories" role = "content" > <category_node title = "DRAMA" feed_url="http://10.0.0.42:8888/roku_lessons/drama.json"/> <category_node title = "COMEDY" feed_url="http://10.0.0.42:8888/roku_lessons/comedy.json"/> <category_node title = "HORROR" feed_url="http://10.0.0.42:8888/roku_lessons/horror.json"/> </ContentNode> </RadioButtonList>
Using
<category_node>
automatically finds and uses the extended node we wrote. Now update the code incategory_screen.brs
:sub onCategorySelected(obj) ? "onCategorySelected field: ";obj.getField() ? "onCategorySelected data: ";obj.getData() ? "onCategorySelected checkedItem: ";m.category_list.checkedItem ? "onCategorySelected selected ContentNode: ";m.category_list.content.getChild(obj.getData()) item = m.category_list.content.getChild(obj.getData()) loadFeed(item.feed_url) end sub sub loadFeed(url) ? "loadFeed! ";url end sub
Run the app to verify the correct value is provided to
loadFeed
. -
Now it's time to introduce a Task to load the data provided by the feed url. Create a new directory in
components
calledtasks
. Insidetasks
, create the fileload_feed_task.xml
with the following code:<?xml version="1.0" encoding="utf-8" ?> <component name = "load_feed_task" extends = "task" > <interface> <field id = "url" type = "string" /> <field id = "response" type = "string" /> </interface> <script type="text/brightscript" uri="pkg:/components/tasks/load_feed_task.brs"/> </component>
Create another file in
tasks
calledload_feed_task.brs
and add the following code:sub init() m.top.functionname = "request" m.top.response = "" end sub function request() url = m.top.url ? "Feed Task URL: ";url m.top.response ="test." end function
Notice the relationship between the fields defined in the XML
<interface>
node, and the values manipulated in the BrightScript code. The task code is providing a value forresponse
, but the code also expects a value to be set forurl
. Field definitions do not enforce access, like getters and setters, they are read/write from either within or without the task code. It is up to the developer to determine the field's use.
Within the task code, there is a special field calledfunctionname
, this is used to define the function the task will run when triggered elsewhere in your application.
Tasks are a critical part of SceneGraph, read more here: https://sdkdocs.roku.com/display/sdkdoc/TaskNow add some code to exercise the new task. Open
screens/category_screen.brs
and add this code:sub loadFeed(url) m.feed_task = createObject("roSGNode", "load_feed_task") m.feed_task.observeField("response", "onFeedResponse") m.feed_task.url = url m.feed_task.control = "RUN" end sub sub onFeedResponse(obj) ? "onFeedResponse: "; obj.getData() end sub
Run the app and select a category, you should see this in the debugger:
onCategorySelected field: itemSelected onCategorySelected data: 0 onCategorySelected checkedItem: 0 onCategorySelected selected ContentNode: <Component: roSGNode> = { change: <Component: roAssociativeArray> focusable: false focusedChild: <Component: roInvalid> id: "" feed_url: "http://10.0.0.42:8888/roku_lessons/drama.json" TITLE: "DRAMA" } Feed Task URL: http://10.0.0.42:8888/roku_lessons/drama.json onFeedResponse: test.
-
Time to make a real HTTP request from the task. At this point, you must have the
server/roku_lessons
directory hosted on a web server somewhere. In these examples I have it running locally at 10.0.0.42:8888.
Updateload_feed_task.brs
with the following code:sub init() m.top.functionname = "request" m.top.response = "" end sub function request() url = m.top.url http = createObject("roUrlTransfer") http.RetainBodyOnError(true) port = createObject("roMessagePort") http.setPort(port) http.setCertificatesFile("common:/certs/ca-bundle.crt") http.InitClientCertificates() http.enablehostverification(false) http.enablepeerverification(false) http.setUrl(url) if http.AsyncGetToString() Then msg = wait(10000, port) if (type(msg) = "roUrlEvent") if (msg.getresponsecode() > 0 and msg.getresponsecode() < 400) m.top.response = msg.getstring() else ? "feed load failed: "; msg.getfailurereason();" "; msg.getresponsecode();" "; m.top.url m.top.response = "" end if http.asynccancel() else if (msg = invalid) ? "feed load failed." m.top.response ="" http.asynccancel() end if end if end function
Say hello again to the weird
wait
loop! This code is how you load something from the internet on Roku. Get familiar with it.
If you dig in the docs, you will see anotherroUrlTransfer
method calledgetToString()
. This also preforms a network request, but it does not capture the diagnostic information available fromasyncGetToString()
. Which method you choose depends on your needs.
Read more here: https://sdkdocs.roku.com/display/sdkdoc/roUrlTransfer
https://sdkdocs.roku.com/display/sdkdoc/ifUrlTransferTake note of this snippet
if (msg.getresponsecode() > 0
. The request may return negative number response codes that are not part of the HTTP standard, read more here:
https://sdkdocs.roku.com/display/sdkdoc/roUrlEventRun this code and you will see the JSON file printed in the debugger.
That's it for Lesson 3.