Include TheThingsNode class and create a pointer to the singleton instance like this:
#include <TheThingsNode.h>
TheThingsNode *node;
Use the static setup()
method in your sketch's setup()
function to create or get the class singleton.
void setup() {
node = TheThingsNode::setup();
}
In your loop()
function, call the singleton's loop()
method.
void loop() {
node->loop();
}
This will:
- Restore USB communication, if the Node has a USB data connection.
- Call the function set via
onWake()
. - Find out why the device woke up and call the function set via the related
on*()
method. - Call the function set via
onSleep()
. - Sleep the Node, unless the Node has a USB data connection and hasn't been configured to also sleep in that case.
Don't add any other code in your
loop()
function, but useonWake()
,onInterval()
andonSleep()
instead, to prevent unpredictable behavior.
Set a callback that will run first thing every time the Node wakes up, which is when an interrupt changes because of interaction with the button, motion sensor, temperature sensor or LoRa device wake.
Depending on configInterval()
setup, this method can be called:
- When you instructed the LoRa RN2xxx device to wake up after the specific interval.
- Every 8 seconds (Watchdog), which is the longest we can make it sleep.
the parameter wakeReason is indicating what waked the node and can be a bit field of the following description
Bit | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
---|---|---|---|---|---|---|---|---|
Wake by | Interval | Watchdog | LoRa | Btn Release | Btn Press | Motion Stop | Motion Start | Temperature |
Is declared into the library as follow:
#define TTN_WAKE_TEMPERATURE 0x01
#define TTN_WAKE_MOTION_START 0x02
#define TTN_WAKE_MOTION_STOP 0x04
#define TTN_WAKE_BTN_PRESS 0x08
#define TTN_WAKE_BTN_RELEASE 0x10
#define TTN_WAKE_LORA 0x20
#define TTN_WAKE_WATCHDOG 0x40
#define TTN_WAKE_INTERVAL 0x80
Note that of course, wake by TTN_WAKE_LORA also setup the TTN_WAKE_INTERVAL because we program the LoRa module to wake at the specified interval.
void onWake(void(*callback)(uint8_t wakeReason));
Usage:
void setup() {
node = TheThingsNode::setup();
node->onWake(wake);
}
void wake(uint8_t wakeReason) {
debugSerial.print(F("Wake Reason: 0x"));
debugSerial.println(wakeReason, HEX);
if (wakeReason & TTN_WAKE_WATCHDOG) {
debugSerial.print(F("Watchdog "));
}
if (wakeReason & TTN_WAKE_INTERVAL) {
debugSerial.print(F("Interval "));
}
if (wakeReason & TTN_WAKE_LORA) {
debugSerial.print(F("LoRa_RN2xxx "));
}
if (wakeReason & TTN_WAKE_BTN_PRESS) {
debugSerial.print(F("Button_Press "));
}
if (wakeReason & TTN_WAKE_BTN_RELEASE) {
debugSerial.print(F("Button_release "));
}
if (wakeReason & TTN_WAKE_MOTION_START) {
debugSerial.print(F("Motion_Start "));
}
if (wakeReason & TTN_WAKE_MOTION_STOP) {
debugSerial.print(F("Motion_Stop "));
}
if (wakeReason & TTN_WAKE_TEMPERATURE) {
debugSerial.print(F("Temperature "));
}
}
Set a callback that will run last thing before the Node goes back to sleep for 8 seconds or until an interrupt changes because of interaction with the button, motion sensor or temperature sensor.
void onSleep(void(*callback)(void));
void(*callback)(void)
: Function to be called, with no arguments nor return value.
Usage:
void setup() {
node = TheThingsNode::setup();
node->onSleep(sleep);
}
void sleep() {
node->setColor(TTN_BLACK);
}
Instead of using your loop()
function, use configInterval()
and onInterval()
to set a function to be called on a certain interval:
void setup() {
node = TheThingsNode::setup();
// Call every 30s instead of the default 60
node->configInterval(true, 30000);
node->onInterval(interval);
}
void interval(uint8_t wakeReason) {
node->showStatus();
}
See onWake()
for definition of wakeReason:
Set a callback that will run on a certain interval. This will automatically enable the interval.
void onInterval(void(*callback)(uint8_t wakeReason));
void(*callback)(uint8_t wakeReason)
: Function to be called.
See onWake()
for definition of wakeReason:
void configInterval(bool enabled, uint32_t ms);
void configInterval(bool enabled);
void configInterval(TheThingsNetwork *pttn, uint32_t ms);
bool enabled
: Enable or disable the interval callback. Enabled automatically byonInterval()
, but you can use this method to temporarily disable it. Defaults tofalse
.TheThingsNetwork * pttn
: This enable the interval callback but in this mode, the interval is passed to RN2483 or RN2903 module (this is why we need to pass pointer to object) with the commandsys sleep ms
and then it's the LoRa module that wake up the node. This is the most advanced Low Power Mode. In this mode, the watchdog is disabled and consumption again reduced.uint32_t ms
: Minimal time between calling the interval callback. Defaults to60000
or 60 seconds.
If the Node has no USB data connection or is configured to also sleep in that case, it will only wake up every 8 seconds to check if the interval callback should run. This means setting
ms
to less than8000
makes no sense. It also means that the maximum time between calls can be up to 8 seconds longer thanms
if it wakes up to handle button or sensor interaction in between. This does not apply when wake up by LoRa module.
Writes information about the device and sensors to Serial
.
void showStatus();
Will write something like:
Light: 25
Temperature: 23.16 C
Temperature alert: No
Moving: No
Button pressed: No
Color: Yellow
USB connected: Yes
Battery voltage: 4704 MV
Configure and read the light sensor.
Configure the light sensor.
void configLight(bool enabled, uint8_t gain);
void configLight(bool enabled);
bool enabled
: Enable or disable the light sensor. Defaults tofalse
.uint8_t gain
: TODO. Value between0
and3
. Defaults to1
.
Get the current value of the light sensor as an unsigned integer of 2 bytes.
uint16_t getLight();
Make sure you have enabled the light sensor or the value will be
0
.
Use and configure the temperature sensor.
Get the current value of the temperature sensor in Celsius, rounded to a signed integer of 1 byte.
int8_t getTemperatureAsInt();
You can call this method even if the temperature alert has been disabled. The sensor will be woken up shortly to get the current temperature.
Get the current value of the temperature sensor in Celsius as signed float of 4 bytes.
float getTemperatureAsFloat();
You can call this method even if the temperature alert has been disabled. The sensor will be woken up shortly to get the current temperature.
Configure the temperature sensor.
void configTemperature(bool enabled, MCP9804_Resolution resolution, int8_t lower, int8_t upper, int8_t critical, MCP9804_Hysteresis hysteresis);
void configTemperature(bool enabled, MCP9804_Resolution resolution);
void configTemperature(bool enabled);
-
bool enabled
: Enable or disable the temperature sensor ALERTS. Enabled automatically byonTemperature()
, but you can use this method to temporarily disable the alerts. Defaults tofalse
. -
MCP9804_Resolution resolution = R_DEGREES_0_0625
: Set the resolution (precision) of the sensor. One of:R_DEGREES_0_5000
: +0.5 CR_DEGREES_0_2500
: +0.25 CR_DEGREES_0_1250
: +0.125 CR_DEGREES_0_0625
: +0.0625 (default)
See the sensor's data sheet, section 5.2.4 for more details.
-
int8_t lower
: Alert when temperature in Celsius is lower. Defaults to0
. -
int8_t upper
: Alert when temperature in Celsius is higher. Defaults to30
. -
int8_t critical = 55
: Alert when temperature in Celsius is higher. Defaults to55
, above which you run the risk of battery failure. -
MCP9804_Hysteresis hysteresis = H_DEGREES_0_0
: Set the margin for the bounds, before an alert is triggered. One of:H_DEGREES_0_0
: 0 C (default)H_DEGREES_1_5
: +1.5 CH_DEGREES_3_0
: +3 CH_DEGREES_6_0
: +6 C
See the sensor's data sheet, section 5.2.2 for more details.
Register a function to be called when temperature exceeds the configured bounds. This will automatically enable the temperature alerts.
void onTemperature(void(*callback)(void));
void(*callback)(void)
: Function to be called, with no arguments nor return value.
Returns true
if the current temperature exceeds the configured bounds for the alert. This also works if the alert itself is temporarily disabled.
bool hasTemperatureAlert();
Use and configure the motion sensor.
Set a function to be called when motion starts. This will automatically enable the motion sensor.
void onMotionStart(void(*callback)(void));
void(*callback)(void)
: Function to be called, with no arguments nor return value.
Set a function to be called when motions stops. This will automatically enable the motion sensor.
void onMotionStop(void(*callback)(unsigned long duration));
void(*callback)
: Function to be called.unsigned long duration
: Time since the motion began, in ms.
Returns true
if the device is currently moving. Requires the sensor to be enabled via onMotionStart()
, onMotionStop()
or setMotion()
.
Returns the acceleration measured by the motion sensor in units of g-force. Each of the values x, y and z falls in the range -2g to +2g. This function is useful to determine the orientation of the sensor relative to the earth, as gravity causes a 1g acceleration towards the centre of the earth. Requires the sensor to be enabled via onMotionStart()
, onMotionStop()
or setMotion()
.
void getAcceleration(float *x, float *y, float *z);
float *x
: Float to store the acceleration along the x-axis.float *y
: Float to store the acceleration along the y-axis.float *z
: Float to store the acceleration along the z-axis.
Enable or disable the motion sensor. The sensor will be enabled automatically by onMotionStart()
and onMotionStop()
, but you can use this method to temporarily disable the sensor.
void configMotion(bool enabled);
bool enabled
: Enable or disable the sensor.
Respond to interaction with the button.
Set a function to be called when the button is pressed down.
void onButtonPress(void(*callback)(void));
void(*callback)(void)
: Function to be called, with no arguments nor return value.
Set a function to be called when the button is released.
void onButtonRelease(void(*callback)(unsigned long duration));
void(*callback)
: Function to be called.unsigned long duration
: Time since the button was pressed, in ms.
Returns true
if the button is currently pressed.
bool isButtonPressed();
Enable or disable the button callback. Will be enabled automatically by onButtonPress()
and onButtonRelease()
, but you can use this method to temporarily disable it.
void configButton(bool enabled);
bool enabled
: Enable or disable the button callback.
Configure the LED.
Combine one or more LEDs to get a primary or additive color.
setColor(ttn_color color);
ttn_color color
: Set to one of:TTN_RED
TTN_GREEN
TTN_BLUE
TTN_YELLOW
: Red and greenTTN_CYAN
: Green and blueTTN_MAGENTA
: Red and blueTTN_WHITE
: Red, green and blueTTN_BLACK
: None
Turn one or more LEDs on or off.
void setRGB(bool red = false, bool green = false, bool blue = false);
bool red = false
: Set totrue
to turn the red LED on. Defaults tofalse
.bool green = false
: Set totrue
to turn the green LED on. Defaults tofalse
.bool blue = false
: Set totrue
to turn the blue LED on. Defaults tofalse
.
Combine colors to create additives, e.g. red plus green makes yellow.
Turn the red LED on or off.
void setRed(bool on = true);
bool on = true
: Set totrue
to turn the red LED on. Defaults totrue
.
Turn the red LED on or off.
void setGreen(bool on = true);
bool on = true
: Set totrue
to turn the green LED on. Defaults totrue
.
Turn the red LED on or off.
void setBlue(bool on = true);
bool on = true
: Set totrue
to turn the blue LED on. Defaults totrue
.
Returns the current color of the LED.
ttn_color getColor();
Returns one of:
TTN_RED
TTN_GREEN
TTN_BLUE
TTN_YELLOW
: Red and greenTTN_CYAN
: Green and blueTTN_MAGENTA
: Red and blueTTN_WHITE
: Red, green and blueTTN_BLACK
: None
Returns a string for the color passed to it.
String colorToString(ttn_color color);
ttn_color color
: Color to return a string for.
Returns one of:
"Red"
:TTN_RED
"Green"
:TTN_GREEN
"Blue"
:TTN_BLUE
"Yellow"
:TTN_YELLOW
"Cyan"
:TTN_CYAN
"Magenta"
:TTN_MAGENTA
"White"
:TTN_WHITE
"Black"
:TTN_BLACK
Returns true
if the red LED is currently on.
bool getRed();
Returns true
if the green LED is currently on.
bool getGreen();
Returns `true if the blue LED is currently on.
bool getBlue();
Monitor the USB data connection.
Returns true
if the Node has a data connection over USB.
bool isUSBConnected();
Enable or disable deep sleep when a USB data connection is active.
void configUSB(bool deepSleep);
bool deepSleep
: Set totrue
to sleep the Node even if a USB data connection is active. Defaults tofalse
.
When disabled, the loop()
method will delay 100ms until an interrupt has changed or configured interval has been reached. When enabled, it will sleep until woken up by an interrupt change or every 8s.
When the Node goes into sleep, the Serial Monitor will loose its connection. The Node will try to reopen the connection when it wakes up, but Serial Monitor might not always be able to pick it up.
Different methods to get battery level
Returns the battery level in millivolt (mV) as a unsigned integer of 2 bytes. This method measure battery with a resistor voltage divider (located before 3.3V ob board regulator.) and connected to an analog input pin. The ADC pin as voltage reference of 2.52V
uint16_t getBattery();
Returns the AT32U4 chip measured VCC voltage millivolt (mV) using the 1.1V internal Analog Reference voltage. This method is prefered when voltage before regulator comes to and below 3.3V. Moreover, if voltage is below 2.52V, using the method getBattery
could return wrong values due to the main voltage < ADC reference voltage.
uint16_t getVCC();
You can also use the getVDD from TTN Device Library.
Returns the voltage in millivolt (mV) measured by the RN2xxx LoRa module. It's for information only since we don't know how it's measured.
uint16_t bat = ttn.getVDD();