BangleApps/apps/BLEcontroller/app-ex2.js

451 lines
13 KiB
JavaScript
Raw Normal View History

2020-06-01 13:19:23 +00:00
/*
==========================================================
Simple event based robot controller that enables robot
to switch into automatic or manual control modes. Behaviours
are controlled via a simple finite state machine.
In automatic mode the
robot will look after itself. In manual mode, the watch
will provide simple forward, back, left and right commands.
The messages will be transmitted to a partner BLE Espruino
using BLE
Written by Richard Hopkins, May 2020
==========================================================
declare global variables for watch button statuses */
top_btn = false;
middle_btn = false;
left_btn= false; // the left side of the touch screen
right_btn = false; // the right side of the touch screen
bottom_btn = false;
msgNum = 0; // message number
/*
CONFIGURATION AREA - STATE VARIABLES
declare global variables for the toggle button
statuses; if you add an additional toggle button
you should declare it and initiase it here */
2020-06-01 18:36:21 +00:00
var status_spk = {value: true};
var status_face = {value: true};
var status_iris_light = {value: false};
var status_iris = {value: false};
var status_hover = {value: false};
var status_dome = {value: false};
2020-06-01 13:19:23 +00:00
/* trsnsmit message
where
s = first character of state,
o = first three character of object name
v = value of state.object
*/
const transmit = (state,object,status) => {
msgNum ++;
msg = {
n: msgNum.toString().slice(-4),
s: state.substr(0,4),
o: object.substr(0,4),
2020-06-01 17:11:18 +00:00
v: status.substr(0,4),
2020-06-01 13:19:23 +00:00
};
message= msg.n + "," + msg.s + "," + msg.o + "," + msg.v;
NRF.setAdvertising({},{
showName: false,
manufacturer: 0x0590,
manufacturerData: JSON.stringify(message)});
};
/*
CONFIGURATION AREA - ICON DEFINITIONS
Retrieve 30px PNG icons from:
https://icons8.com/icon/set/speak/ios-glyphs
Create icons using:
https://www.espruino.com/Image+Converter
Use compression: true
Transparency: true
Diffusion: flat
Colours: 16bit RGB
Ouput as: Image Object
Add an additional element to the icons array
with a unique name and the data from the Image Object
*/
const icons = [
{
2020-06-01 18:36:21 +00:00
name: "back",
data: "gEBAP4B/AP4B/AKgADHPI71HP45/HP45/HP45/HP45/Hf49/Hv49/Hv49/Hv49/Hv497He4B/AP4B/AJAA=="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "spk_on",
data: "gEBAP4B/AP4Bic/YAFPP4v1HrYZRVJo7ZDKp5jMJYvZHaYAHVL4LHACZrhADLBTJKI7dPLI7/Hf47/HeZBVFqZHZRJp1lAJ47LOtZTnHbIZDKLpHNAL69ZANp1tQbY5/AP4B/ANQ"
2020-06-01 13:19:23 +00:00
},
2020-06-01 18:36:21 +00:00
{
name: "spk_off",
data: "gEBAPhB7P/o9rFKI9pFKY9tXNYZNHrZXfMaoAHPOZhNF7LdXHpKpZEJpvPDZK1ZAB49NPLo9jHdI9NHd49PHebvxEJY9NI6I7dHpaDXcKqfPHLKjZHcpTjHbIZDKa73JHa4BXGY45xe5Y7zV+o9/Hv49JHe4BEA="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "facerecog",
data: "gEBAP4BSLuozNH9YpTHsolXPsYfdDraZhELIZhHeLtJELY1VC4Y7HHqoXJABYdNHa5bJDrLvfHfbrPZJI7nGZpdVNJ4lRIpaznRqp1hCq55ZC6IRPd8oPjW8Y5jSr45dEJppNHcIjLHZY5ja6rrhFK45pVqI5rGI4AHHNpx3ANA="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "sleep",
data: "gEBAP4B/AP4B2ACY7/Quq95HP45/HP4APOdY7fACZfnHcaZZAL45/HP45/E7YAHCaZFZHfbh/HP45/HOoAHHf4B/AP4B/AP4BIA="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "awake",
data: "gEBAP4B/AKyb7HfIAFHPI77Ov451Hf453Hf453HdoAbHf45/Hf5HrHNY7NHNo7/HO47/HO47HHPJ1/Heo51HfoB/ALg="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "happy",
data: "gEBAP4B/AP4BKa+oAXHNITfHK4ZtD5JZfHOojZaMYlXHMYnXHfI5nFaYPLaaIRNHf47/d/47/HtInTCZrfZHa4vNABYlVKLI3PbLrzfD7qTXDLaphHMIpLAB45hIKY1pAP4B/AMA"
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "sad",
data: "gEBAP4B/AP4BKa+oAXHNITfHK4ZtD5JZfHOojZaMYlXHMYnXHfI5nFaYPLaaIRNHf47/d/47/CK4njCZ4APHcIVJBbbdTecYjZHr4fdSa4ZbEZ4lNCaY9dAB45hIKY1pAP4B/AMA"
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "hover",
data: "gEBAP4B/AP7NedL4fZK7ojNHeJ35DJI7vC5Y7tVMI7XHNYnNYro7hHKI7lAK47/HdoAhHPI7/Hf47/Hf4AtHPI7/Hf47/Hd45LAP4B/ANwA="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "light",
data: "gEBAP4B/APi/Na67lfACZ/nNaI9lE6o9jEbI9hD7Y7dDsJZ3D6YRJHdIJHHfaz7Hf5Z/Hf4hZHMIjFEqIVVHsY5hDpI7TEqL1jVsqlTdM55THOJvHOuY7/HfI9JHOI9HHOoBgA=="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "speak",
data: "gEBAP4B/AP4BIbO4AXG+4/hAEY55HqoArHPI9PHfIAzHf47/Hf47/HeY9xHJI79Hto5NHtY5RHc45THco5VHcI3XHJpHRG7I7LEro5ZG+IB/AP4BwA=="
2020-06-01 13:19:23 +00:00
},
{
2020-06-01 18:36:21 +00:00
name: "dalek",
data: "gEBAP4B/AP4B/AJMQwQBBGucIoMAkADBhFhAoZBcAAQfJhEgB45BCHYMBjGiB4ZLCK5APDFpphBC5AbEJosY0YfCG4IAEJIYdGFYR5LHJYlEAI0Y4cY8YXMOpQBFlNFlMkOZA7MKII7JOAXkE4T1UERKtFHoxJBABY5QiGiD5kANYTnCiFiWIJVOgDZCOra3FoKxFDKI7hADQ7PkEIaoIHEaKYfJAoKPFAJcIGYIJHkI7UgMY8ZFHC5rVDKIZTCDIJhBA4ILBBoYFHC4QBEBogpBjHDdsJJEAoYAHKoTxWWb5tNWZOiHZRbBHbwtLF5ynBL7wtLjHjd6oAZkHkI5JJKAAZ3TkAjJhALBsJ5K0a/KkLvfkMEFpVhO8hrIU4QLGG4QAzkCdVAP4B/AP4Bb"
2020-06-01 13:19:23 +00:00
}
];
/* finds icon data by name in the icon array and returns an image object*/
const drawIcon = (name) => {
for (var icon of icons) {
if (icon.name == name) {
image = {
width : 30, height : 30, bpp : 16,
transparent : 1,
buffer: require("heatshrink").decompress(atob(icon.data))
};
return image;}
}
};
/*
CONFIGURATION AREA - BUTTON DEFINITIONS
for a simple button, just define a primary colour
and an icon name from the icon array and
the text to display beneath the button
for toggle buttons, additionally provide secondary
colours, icon name and text. Also provide a reference
to a global variable for the value of the button.
The global variable should be declared at the start of
the program and it may be adviable to use the 'status_name'
format to ensure it is clear.
*/
2020-06-01 16:30:19 +00:00
2020-06-01 18:36:21 +00:00
var happyBtn = {
2020-06-01 13:19:23 +00:00
primary_colour: 0x653E,
2020-06-01 18:36:21 +00:00
primary_text: 'Speak',
primary_icon: 'happy',
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
var sadBtn = {
2020-06-01 13:19:23 +00:00
primary_colour: 0x33F9,
2020-06-01 18:36:21 +00:00
primary_text: 'Speak',
primary_icon: 'sad',
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
var speakBtn = {
primary_colour: 0x33F9,
primary_text: 'Speak',
primary_icon: 'speak',
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
var faceBtn = {
primary_colour: 0xE9C7,
primary_text: 'Off',
primary_icon: 'facerecog',
toggle: true,
secondary_colour: 0x3F48,
secondary_text: 'On',
secondary_icon : 'facerecog',
value: status_face
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
var irisLightBtn = {
2020-06-01 13:19:23 +00:00
primary_colour: 0xE9C7,
primary_text: 'Off',
2020-06-01 18:36:21 +00:00
primary_icon: 'light',
2020-06-01 13:19:23 +00:00
toggle: true,
secondary_colour: 0x3F48,
secondary_text: 'On',
2020-06-01 18:36:21 +00:00
secondary_icon : 'light',
value: status_iris_light
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
var irisBtn = {
2020-06-01 13:19:23 +00:00
primary_colour: 0xE9C7,
2020-06-01 18:36:21 +00:00
primary_text: 'Closed',
2020-06-01 13:19:23 +00:00
primary_icon: 'sleep',
toggle: true,
secondary_colour: 0x3F48,
2020-06-01 18:36:21 +00:00
secondary_text: 'Open',
2020-06-01 13:19:23 +00:00
secondary_icon : 'awake',
2020-06-01 18:36:21 +00:00
value: status_iris
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
var hoverBtn = {
2020-06-01 13:19:23 +00:00
primary_colour: 0xE9C7,
2020-06-01 18:36:21 +00:00
primary_text: 'Off',
primary_icon: 'hover',
2020-06-01 13:19:23 +00:00
toggle: true,
secondary_colour: 0x3F48,
2020-06-01 18:36:21 +00:00
secondary_text: 'On',
secondary_icon : 'hover',
value: status_hover
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
var domeBtn = {
primary_colour: 0xE9C7,
primary_text: 'Off',
primary_icon: 'dalek',
toggle: true,
secondary_colour: 0x3F48,
secondary_text: 'On',
secondary_icon : 'dalek',
value: status_dome
};
2020-06-01 13:19:23 +00:00
/*
CONFIGURATION AREA - SCREEN DEFINITIONS
a screen can have a button (as defined above)
on the left and/or the right of the screen.
in adddition a screen can optionally have
an icon for each of the three buttons on
the left hand side of the screen. These
are defined as btn1, bt2 and bt3. The
values are names from the icon array.
*/
2020-06-01 18:36:21 +00:00
2020-06-01 13:19:23 +00:00
const menuScreen = {
2020-06-01 18:36:21 +00:00
left: faceBtn,
right: speakBtn,
btn1: "hover",
btn2: "light",
2020-06-01 13:19:23 +00:00
};
2020-06-01 18:36:21 +00:00
const speakScreen = {
left: happyBtn,
right: sadBtn,
2020-06-01 13:19:23 +00:00
btn3: "back"
};
2020-06-01 18:36:21 +00:00
const irisScreen = {
left: irisBtn,
right: irisLightBtn,
2020-06-01 13:19:23 +00:00
btn3: "back"
};
2020-06-01 18:36:21 +00:00
const lightsScreen = {
left: hoverBtn,
right: domeBtn,
2020-06-01 13:19:23 +00:00
btn3: "back"
};
/* base state definition
Each of the screens correspond to a state;
this class provides a constuctor for each
of the states
*/
class State {
constructor(params) {
this.state = params.state;
this.events = params.events;
this.screen = params.screen;
}
}
/*
CONFIGURATION AREA - BUTTON BEHAVIOURS/STATE TRANSITIONS
This area defines how each screen behaves.
Each screen corresponds to a different State of the
state machine. This makes it much easier to isolate
behaviours between screens.
The state value is transmitted whenever a button is pressed
to provide context (so the receiving device, knows which
button was pressed on which screen).
The screens are defined above.
The events section identifies if a particular button has been
pressed and released on the screen and an action can then be taken.
The events function receives a notification from a mySetWatch which
provides an event object that identifies which button and whether
it has been pressed down or released. Actions can then be taken.
The events function will always return a State object.
If the events function returns different State from the current
one, then the state machine will change to that new State and redrsw
the screen appropriately.
To add in additional capabilities for button presses, simply add
an additional 'if' statement.
2020-06-01 18:36:21 +00:00
For toggle buttons, the value of the sppropiate status object is
2020-06-01 13:19:23 +00:00
inversed and the new value transmitted.
*/
/* The Home State/Page is where the application beings */
2020-06-01 16:30:19 +00:00
const Home = new State({
2020-06-01 18:36:21 +00:00
state: "DalekMenu",
2020-06-01 16:30:19 +00:00
screen: menuScreen,
2020-06-01 13:19:23 +00:00
events: (event) => {
2020-06-01 18:36:21 +00:00
if ((event.object == "top") && (event.status == "end")) {
return Lights;
}
if ((event.object == "middle") && (event.status == "end")) {
return Iris;
}
2020-06-01 13:19:23 +00:00
if ((event.object == "right") && (event.status == "end")) {
2020-06-01 18:36:21 +00:00
return Speak;
2020-06-01 13:19:23 +00:00
}
if ((event.object == "left") && (event.status == "end")) {
2020-06-01 18:36:21 +00:00
status_face.value = !status_face.value;
transmit(this.state, "face", onOff(status_face.value));
2020-06-01 13:19:23 +00:00
return this;
}
transmit(this.state, event.object, event.status);
return this;
}
});
2020-06-01 18:36:21 +00:00
const Speak = new State({
state: "Speak",
screen: speakScreen,
2020-06-01 13:19:23 +00:00
events: (event) => {
if ((event.object == "bottom") && (event.status == "end")) {
2020-06-01 16:30:19 +00:00
return Home;
2020-06-01 13:19:23 +00:00
}
transmit(this.state, event.object, event.status);
return this;
}
});
2020-06-01 18:36:21 +00:00
const Iris = new State({
state: "Iris",
screen: irisScreen,
2020-06-01 13:19:23 +00:00
events: (event) => {
if ((event.object == "bottom") && (event.status == "end")) {
2020-06-01 16:30:19 +00:00
return Home;
2020-06-01 13:19:23 +00:00
}
2020-06-01 18:36:21 +00:00
if ((event.object == "right") && (event.status == "end")) {
status_iris_light.value = !status_iris_light.value;
transmit(this.state, "light", onOff(status_iris_light.value));
return this;
}
if ((event.object == "left") && (event.status == "end")) {
status_iris.value = !status_iris.value;
transmit(this.state, "servo", onOff(status_iris.value));
return this;
}
2020-06-01 13:19:23 +00:00
transmit(this.state, event.object, event.status);
return this;
}
});
2020-06-01 18:36:21 +00:00
const Lights = new State({
state: "Lights",
screen: lightsScreen,
2020-06-01 13:19:23 +00:00
events: (event) => {
if ((event.object == "bottom") && (event.status == "end")) {
2020-06-01 18:36:21 +00:00
return Home;
}
if ((event.object == "right") && (event.status == "end")) {
status_dome.value = !status_dome.value;
transmit(this.state, "dome", onOff(status_dome.value));
return this;
}
if ((event.object == "left") && (event.status == "end")) {
status_hover.value = !status_hover.value;
transmit(this.state, "hover", onOff(status_hover.value));
return this;
}
2020-06-01 13:19:23 +00:00
transmit(this.state, event.object, event.status);
return this;
2020-06-01 18:36:21 +00:00
}
2020-06-01 13:19:23 +00:00
});
/* translate button status into english */
const startEnd = status => status ? "start" : "end";
/* translate status into english */
const onOff= status => status ? "on" : "off";
/* create watching functions that will change the global
button status when pressed or released
This is actuslly the hesrt of the program. When a button
is not being pressed, nothing is happening (no loops).
This makes the progrsm more battery efficient.
When a setWatch event is raised, the custom callbacks defined
here will be called. These then fired as events to the current
state/screen of the state mschine.
Some events, will result in the stste of the state machine
chsnging, which is why the screen is redrswn after each
button press.
*/
const setMyWatch = (params) => {
setWatch(() => {
params.bool=!params.bool;
machine = machine.events({object: params.label, status: startEnd(params.bool)});
drawScreen(machine.screen);
}, params.btn, {repeat:true, edge:"both"});
};
/* object array used to set up the watching functions
*/
const buttons = [
{bool : bottom_btn, label : "bottom",btn : BTN3},
{bool : middle_btn, label : "middle",btn : BTN2},
{bool : top_btn, label : "top",btn : BTN1},
{bool : left_btn, label : "left",btn : BTN4},
{bool : right_btn, label : "right",btn : BTN5}
];
/* set up watchers for buttons */
for (var button of buttons)
{setMyWatch(button);}
/* Draw various kinds of buttons */
const drawButton = (params,side) => {
g.setFontAlign(0,1);
icon = drawIcon(params.primary_icon);
text = params.primary_text;
g.setColor(params.primary_colour);
const x = (side == "left") ? 0 : 120;
if ((params.toggle) && (params.value.value)) {
g.setColor(params.secondary_colour);
text = params.secondary_text;
icon = drawIcon(params.secondary_icon);
}
2020-06-01 18:36:21 +00:00
g.fillRect(0+x,24,119+x, 239);
2020-06-01 13:19:23 +00:00
g.setColor(0x000);
g.setFont("Vector",15);
g.setFontAlign(0,0.0);
g.drawString(text,60+x,160);
options = {rotate: 0, scale:2};
g.drawImage(icon,x+60,120,options);
};
/* Draw the pages corresponding to the states */
const drawScreen = (params) => {
drawButton(params.left,'left');
drawButton(params.right,'right');
g.setColor(0x000);
if (params.btn1) {g.drawImage(drawIcon(params.btn1),210,40);}
if (params.btn2) {g.drawImage(drawIcon(params.btn2),210,125);}
if (params.btn3) {g.drawImage(drawIcon(params.btn3),210,195);}
};
machine = Home; // instantiate the state machine at Home
Bangle.drawWidgets(); // draw active widgets
drawScreen(machine.screen); // draw the screen