Home STEM HQ TI-Nspire Authoring TI-Nspire Scripting HQ Scripting Tutorial - Lesson 43


Scripting Tutorial - Lesson 43: Lua Scripting and the TI Innovator™ Hub:

3. Exploring the OneShotTimer: Learning to Wait!

Download supporting file for this tutorial

Texas Instruments TI-Nspire Scripting Support Page


Scripting for the Innovator Hub: Lesson 3: Exploring the OneShotTimer: Learning to Wait!

Top of Page

One of the many useful TI-BASIC commands that you discover very quickly is the simple wait. We take it for granted without often appreciating its value - enter a command, then wait(2) and there is a satisfying 2 second pause before the next command is carried out.

Lua has no such in-built command, although one is readily created in a variety of ways. However, as I quickly discovered, a key difference between TI-BASIC programming and Lua scripting is the element of multi-tasking. This is not supported on the handheld, and hence in TI-BASIC, which means that the simple wait command can work just as you would expect it to: command is issued, wait for a given time, and then proceed to the next.

In Lua however, capable of multi-tasking, if you try this process you quickly discover that the first command is issued, then your wait command is issued - but while this is being carried out, the script happily continues to fulfil subsequent commands. These things can happen simultaneously.

Now overall, of course, such multi-tasking is a great advantage. Suppose you are using a TI-BASIC program to collect some data from a sensor attached to your Innovator Hub. Obviously, it would be nice to see that data as it is being collected - perhaps even see the graph unfold?

Unfortunately, this is not possible. Without multi-tasking capabilities, the handheld (or computer or iPad) running TI-BASIC will not show you any such changes until it finishes what it was working on! So the data will be happily captured and stored, as requested, but you will not see this on Calculator, or spreadsheet or graph screen until the collection process is complete. Frustrating.

This points to a key advantage of controlling your Innovator Hub using Lua - such processes are easily achieved. Properly scripted, your data can be collected and displayed at the same time. Likewise, if you are displaying changes on screen from the effects of some Innovator Hub activity, these can be shown live.


What is not so simple within Lua is to go backwards and recreate that simple sequencing that TI-BASIC delivers freely - and for that, we use a special tool called a oneShotTimer.

The oneShotTimer effectively hijacks the Lua script timer for its own purposes: the timer is linked to a period, and a function. The period will be the length of time that you wish the related function to run for. The function will be specific to the task that you wish to achieve while waiting.

Before we examine these elements in detail, we should first check on those changes to the script which are required here. Once again, the script is essentially the same as those we have studied previously, with just a few additions for the purposes of learning more about this oneShotTimer. These are highlighted in red in the script snippets on the right, and below.

  1. We add a new variable to those previously used (timeStr - to display information from the timer).

    Two lists (index and dataList) are created to store accumulated values of numeric results - this is now possible with a timer function, and offers a powerful advantage of using Lua over TI-BASIC. The lists are stored as TI-Nspire variables and so may be viewed (live) in graph or spreadsheet.

    You will notice in the menu that we have amalgamated the two sample menus from the previous lessons (SET Samples and READ Samples) and added a third, specific to this lesson - TIMER Samples.

    The three functions in the TIMER Samples menu will form the basis for this tutorial.


  3. In addition to displaying both rxValue and message lines, we have added a timerStr line to display information about the timer as it does its work. We also define initial values for the variables in on.resize.

  5. Here is where the action happens! Three sample functions are provided for you to study.

    • wait(input)

      This is a fairly generic oneShotTimer() function, deliberately styled after the TI-BASIC form - so we call, say, wait(5) and would like the process to pause for 5 seconds and then continue.

      Looking ahead into the function, find where the oneShotTimer is called, and look at its arguments: oneShotTimer(1000*time_step,wait) . The first calls a 1 second delay (or whatever your time_step is set to), and then the wait function is called again - but this time WITHOUT any input! This is important - it accepts a delay period only the first time it is called.

      This explains the first line of the function: we have set up a delay variable already, and the first time the wait function is called, the input is stored as delay, from which it may be accessed later. So for wait(5), on the first run, delay is set to the value 5. A time_step value is set - use 1 for 1 second intervals, but be careful when using smaller intervals as the read function needs time to gather and send data. Hence you will see an adjustment which forces a read only at 1 second intervals.

      While delay is greater than or equal to zero, the timeStr is displayed, the current hubStrOn command is run, and the oneShotTimer function is called. This serves to start the clock, a second passes (or whatever your time_step is set to), delay is decreased, and the wait function is called again. This happens each second until time catches up with the delay value, and then it stops.

      The nice feature of this function is that it takes whatever is the current hubStrOn command - SET LIGHT ON, or play sound, or run servo - and adds a 5 second timer. Try reading BRIGHTNESS to set that as the current command. Then run wait(5) and you will see values read at 1 second intervals for 5 seconds, as specified.

    • blink(rep,waitValue)

      Again, styled after a simple TI-BASIC function, this takes a value for number of blinks ('rep') and delay between blinks ('waitValue').

      As previously, we link these to existing defined variables (repeats and delay) so that the first time the function is run, these variables are defined with the values desired. So, for example, blink(5,0.5) will blink 5 times at half second intervals.

      I defined another variable called "lightState" for this function, since I want to know whether the light is ON or OFF, and switch between these values each time the function is run.

      The sequence of this function is similar to the wait function above, although this time it is the variable 'repeats' that serves as the count down, rather than 'delay' (which stays constant and serves the same role as time_step did previously.) This time, though, we step through TWICE the number of steps specified, so that we can turn the light ON and then OFF for each repetition.

    • alarm(rep,waitValue)

      Just to add a little variety, this function is a variation on blink to demonstrate how the same principle can be applied in different ways.

  7. No changes from that previously defined.


  9. These functions are as previously defined.



-- TI Innovator Init Values

platform.apilevel = '2.7'
local screen = platform.window
local date = "021517"

require 'color'
pcall(function() require 'asi' end)

local isASIavailable = false
local hubStrOn, hubStrOff, msgStr, rxValue, timeStr, delay, repeats
local index, dataList = {}, {}
var.store("index", index)
var.store("dataList", dataList)

function addMsg(input)

msgStr = input


function on.construction()

--the next three lines are required for Hub connection


{" ©2017 Compass Learning Technologies", function() end},
{" Version "..date, function() end},
{" Contact: steve@compasstech.com.au ", function() end},
{"Scan and Connect", function() TI_Innovator.connect() end},
{"Disconnect", function() TI_Innovator.disconnect() on.resize() end},
{"RESET", function() on.resize() end},
{"SET Samples",
{"SET LIGHT ON/OFF (default)", function() hubStrOn = "SET LIGHT ON" hubStrOff = "SET LIGHT OFF" TI_Innovator.Send(hubStrOn) end},
{"SET COLOR.RED ON/OFF", function() hubStrOn = "SET COLOR.RED ON" hubStrOff = "SET COLOR.RED OFF" TI_Innovator.Send(hubStrOn) end},
{"SET COLOR.GREEN ON/OFF", function() hubStrOn = "SET COLOR.GREEN ON" hubStrOff = "SET COLOR.GREEN OFF" TI_Innovator.Send(hubStrOn) end},
{"SET COLOR.BLUE ON/OFF", function() hubStrOn = "SET COLOR.BLUE ON" hubStrOff = "SET COLOR.BLUE OFF" TI_Innovator.Send(hubStrOn) end},
{"SET SOUND 220 5", function() hubStrOn = "SET SOUND 220 5" hubStrOff = "SET SOUND 0 1" TI_Innovator.Send(hubStrOn) end},
{"SET SPEAKER 1 220 5", function() hubStrOn = "SET SPEAKER 1 220 5" TI_Innovator.Send("CONNECT SPEAKER 1 OUT1") TI_Innovator.Send(hubStrOn) hubStrOff = "SET SPEAKER 1 0 1" end},
{"SET LED 1 ON", function() hubStrOn = "SET LED 1 ON" TI_Innovator.Send("CONNECT LED 1 OUT1") TI_Innovator.Send(hubStrOn) hubStrOff = "SET LED 1 0FF" end},
{"SET SERVO 1 CW 50 5", function() TI_Innovator.Send("CONNECT SERVO 1 OUT3") hubStrOn = "SET SERVO 1 CW 50 5" hubStrOff = "SET SERVO 1 CW 50 0" TI_Innovator.Send(hubStrOn) end},
{"SET ANALOG.OUT 1 100", function() hubStrOn = "SET ANALOG.OUT 1 100" TI_Innovator.Send("CONNECT ANALOG.OUT 1 OUT1") TI_Innovator.Send(hubStrOn) hubStrOff = "SET ANALOG.OUT 1 0" end},
{"SET DIGITAL.OUT 1 ON", function() hubStrOn = "SET DIGITAL.OUT 1 ON" TI_Innovator.Send("CONNECT DIGITAL.OUT 1 OUT1") TI_Innovator.Send(hubStrOn) hubStrOff = "SET DIGITAL.OUT 1 0FF" end},
}, {"READ Samples",
{"READ BRIGHTNESS (default)", function() hubStrOn = "READ BRIGHTNESS" TI_Innovator.Send(hubStrOn) TI_Innovator.Read() msgStr = rxValue or hubStrOn end},
{"READ RANGER 1", function() hubStrOn = "READ RANGER 1" TI_Innovator.Send("CONNECT RANGER 1 IN1") TI_Innovator.Send(hubStrOn) TI_Innovator.Read() msgStr = rxValue or hubStrOn end},
{"READ DHT 1", function() hubStrOn = "READ DHT 1" TI_Innovator.Send("CONNECT DHT 1 IN1") TI_Innovator.Send(hubStrOn) TI_Innovator.Read() msgStr = rxValue or hubStrOn end},
{"READ ACCEL 1", function() hubStrOn = "READ ACCEL 1" TI_Innovator.Send("CONNECT ACCEL 1 I2C") TI_Innovator.Send(hubStrOn) TI_Innovator.Read() msgStr = rxValue or hubStrOn end},
{"READ BAROMETER", function() hubStrOn = "READ BAROMETER" TI_Innovator.Send("CONNECT BAROMETER I2C") TI_Innovator.Send(hubStrOn) TI_Innovator.Read() msgStr = rxValue or hubStrOn end},
{"READ ANALOG.IN 1", function() hubStrOn = "READ ANALOG.IN 1" TI_Innovator.Send("CONNECT ANALOG.IN 1 IN1") TI_Innovator.Send(hubStrOn) TI_Innovator.Read() hubStrOff = hubStrOn end},
{"READ DIGITAL.IN 1", function() hubStrOn = "READ DIGITAL.IN 1" TI_Innovator.Send("CONNECT DIGITAL.IN 1 IN1") TI_Innovator.Send(hubStrOn) TI_Innovator.Read() hubStrOff = hubStrOn end},
{"TIMER Samples",
{"wait(5)", function() wait(5) end},
{"blink(5,1)", function() blink(5,1) end},
{"alarm(5,0.5)", function() alarm(5,0.5) end},



Our next lesson offers another nice application of the OneShotTimer - playing music from lists of tone values and times!


Back to Top


Home STEM HQ TI-Nspire Authoring TI-Nspire Scripting HQ Scripting Tutorial - Lesson 43