|
| 1 | +/* |
| 2 | + Esplora Table |
| 3 | +
|
| 4 | + Acts like a keyboard that prints some of its sensors' |
| 5 | + data in a table-like text, row by row. |
| 6 | + It is a sort of "data-logger". |
| 7 | + |
| 8 | + At startup, it does nothing. It just waits for you to open a |
| 9 | + spreadsheet (e.g. Google Drive spreadsheet) so it can put its |
| 10 | + data. Then, by pressing Switch 1, it starts printing the table |
| 11 | + headers and the first row of data. It waits a bit, then it |
| 12 | + will print another row, and so on. |
| 13 | + |
| 14 | + The amount of time between each row is given by the slider. |
| 15 | + If put to full left, the sketch will wait 10 seconds; at |
| 16 | + full right position, it will wait 5 minutes. An intermediate |
| 17 | + position will make the sketch wait for some time in-between. |
| 18 | + |
| 19 | + Clicking the Switch 1 at any time will stop the logging. |
| 20 | +
|
| 21 | + The color LED shows what the sketch is doing: |
| 22 | + blue = idle, waiting for you to press Switch 1 to start logging |
| 23 | + green = active; will print soon |
| 24 | + red = printing data to the PC |
| 25 | +
|
| 26 | + Created on 22 november 2012 |
| 27 | + By Enrico Gueli <enrico.gueli@gmail.com> |
| 28 | + modified 24 Nov 2012 |
| 29 | + by Tom Igoe |
| 30 | +*/ |
| 31 | + |
| 32 | +#include <Esplora.h> |
| 33 | + |
| 34 | +/* |
| 35 | + * this variable tells if the data-logging is currently active. |
| 36 | + */ |
| 37 | +boolean active = false; |
| 38 | + |
| 39 | +/* |
| 40 | + * this variable holds the time in the future when the sketch |
| 41 | + * will "sample" the data (sampling is the act of reading some |
| 42 | + * input at a known time). This variable is checked continuously |
| 43 | + * against millis() to know when it's time to sample. |
| 44 | + */ |
| 45 | +unsigned long nextSampleAt = 0; |
| 46 | + |
| 47 | +/* |
| 48 | + * This variable just holds the millis() value at the time the |
| 49 | + * logging was activated. This is needed to enter the correct |
| 50 | + * value in the "Time" column in the printed table. |
| 51 | + */ |
| 52 | +unsigned long startedAt = 0; |
| 53 | + |
| 54 | + |
| 55 | +/* |
| 56 | + * when the "active" variable is set to true, the same is done |
| 57 | + * with this variable. This is needed because the code that does |
| 58 | + * the "just-after-activation" stuff is run some time later than |
| 59 | + * the code that says "be active now". |
| 60 | + */ |
| 61 | +boolean justActivated = false; |
| 62 | + |
| 63 | + |
| 64 | +/* |
| 65 | + * this variable holds the last sensed status of the switch press |
| 66 | + * button. If the code sees a difference between the value of |
| 67 | + * this variable and the current status of the switch, it means |
| 68 | + * that the button was either pressed or released. |
| 69 | + */ |
| 70 | +boolean lastStartBtn = HIGH; |
| 71 | + |
| 72 | +/* |
| 73 | + * Initialization code. The virtual USB keyboard must be |
| 74 | + * initialized; the Serial class is needed just for debugging. |
| 75 | + */ |
| 76 | +void setup() { |
| 77 | + Keyboard.begin(); |
| 78 | + Serial.begin(9600); |
| 79 | +} |
| 80 | + |
| 81 | +/* |
| 82 | + * This code is run continuously. |
| 83 | + */ |
| 84 | +void loop() { |
| 85 | + /* |
| 86 | + * note: we don't use Arduino's delay() here, because we can't |
| 87 | + * normally do anything while delaying. Our own version lets us |
| 88 | + * check for button presses often enough to not miss any event. |
| 89 | + */ |
| 90 | + activeDelay(50); |
| 91 | + |
| 92 | + /* |
| 93 | + * the justActivated variable may be set to true in the |
| 94 | + * checkSwitchPress() function. Here we check its status to |
| 95 | + * print the table headers and configure what's needed to. |
| 96 | + */ |
| 97 | + if (justActivated == true) { |
| 98 | + justActivated = false; // do this just once |
| 99 | + printHeaders(); |
| 100 | + // do next sampling ASAP |
| 101 | + nextSampleAt = startedAt = millis(); |
| 102 | + } |
| 103 | + |
| 104 | + if (active == true) { |
| 105 | + if (nextSampleAt < millis()) { |
| 106 | + // it's time to sample! |
| 107 | + int slider = Esplora.readSlider(); |
| 108 | + // the row below maps the slider position to a range between |
| 109 | + // 10 and 290 seconds. |
| 110 | + int sampleInterval = map(slider, 0, 1023, 10, 290); |
| 111 | + nextSampleAt = millis() + sampleInterval * 1000; |
| 112 | + |
| 113 | + logAndPrint(); |
| 114 | + } |
| 115 | + |
| 116 | + // let the RGB led blink green once per second, for 200ms. |
| 117 | + unsigned int ms = millis() % 1000; |
| 118 | + if (ms < 200) |
| 119 | + Esplora.writeGreen(50); |
| 120 | + else |
| 121 | + Esplora.writeGreen(0); |
| 122 | + |
| 123 | + Esplora.writeBlue(0); |
| 124 | + } |
| 125 | + else |
| 126 | + // while not active, keep a reassuring blue color coming |
| 127 | + // from the Esplora... |
| 128 | + Esplora.writeBlue(20); |
| 129 | + |
| 130 | +} |
| 131 | + |
| 132 | +/* |
| 133 | + * Print the table headers. |
| 134 | + */ |
| 135 | +void printHeaders() { |
| 136 | + Keyboard.print("Time"); |
| 137 | + Keyboard.write(KEY_TAB); |
| 138 | + activeDelay(300); // Some spreadsheets are slow, e.g. Google |
| 139 | + // Drive that wants to save every edit. |
| 140 | + Keyboard.print("Accel X"); |
| 141 | + Keyboard.write(KEY_TAB); |
| 142 | + activeDelay(300); |
| 143 | + Keyboard.print("Accel Y"); |
| 144 | + Keyboard.write(KEY_TAB); |
| 145 | + activeDelay(300); |
| 146 | + Keyboard.print("Accel Z"); |
| 147 | + Keyboard.println(); |
| 148 | + activeDelay(300); |
| 149 | +} |
| 150 | + |
| 151 | +void logAndPrint() { |
| 152 | + // do all the samplings at once, because keystrokes have delays |
| 153 | + unsigned long timeSecs = (millis() - startedAt) /1000; |
| 154 | + int xAxis = Esplora.readAccelerometer(X_AXIS); |
| 155 | + int yAxis = Esplora.readAccelerometer(Y_AXIS); |
| 156 | + int zAxis = Esplora.readAccelerometer(Z_AXIS); |
| 157 | + |
| 158 | + Esplora.writeRed(100); |
| 159 | + |
| 160 | + Keyboard.print(timeSecs); |
| 161 | + Keyboard.write(KEY_TAB); |
| 162 | + activeDelay(300); |
| 163 | + Keyboard.print(xAxis); |
| 164 | + Keyboard.write(KEY_TAB); |
| 165 | + activeDelay(300); |
| 166 | + Keyboard.print(yAxis); |
| 167 | + Keyboard.write(KEY_TAB); |
| 168 | + activeDelay(300); |
| 169 | + Keyboard.print(zAxis); |
| 170 | + Keyboard.println(); |
| 171 | + activeDelay(300); |
| 172 | + Keyboard.write(KEY_HOME); |
| 173 | + |
| 174 | + Esplora.writeRed(0); |
| 175 | +} |
| 176 | + |
| 177 | +/** |
| 178 | + * Similar to delay(), but allows to do something else |
| 179 | + * in the meanwhile. In particular, it calls waitLoop(). |
| 180 | + * Note 1: it may wait longer than the specified amount, not less; |
| 181 | + * Note 2: beware of data synchronization issues, e.g. if the |
| 182 | + * whileWaiting() function alters some variables used by the |
| 183 | + * caller of this function. |
| 184 | + * |
| 185 | + * I discovered by chance that there's an ongoing discussion about |
| 186 | + * adding yield() in the Arduino API: |
| 187 | + * http://comments.gmane.org/gmane.comp.hardware.arduino.devel/1381 |
| 188 | + * The purpose is the same, but for now I'm using this implementation. |
| 189 | + */ |
| 190 | +void activeDelay(unsigned long amount) { |
| 191 | + unsigned long at = millis() + amount; |
| 192 | + while (millis() < at) { |
| 193 | + checkSwitchPress(); |
| 194 | + } |
| 195 | +} |
| 196 | + |
| 197 | +/* |
| 198 | + * This function reads the status of the switch; if it sees that |
| 199 | + * it was pressed, toggles the status of the "active" variable. |
| 200 | + * If it's set to true, also the justActivated variable is set to |
| 201 | + * true, so the loop() function above can do the right things. |
| 202 | + * This function should be called as often as possible and do as |
| 203 | + * little as possible, because it can be called while another |
| 204 | + * function is running. |
| 205 | + */ |
| 206 | +void checkSwitchPress() { |
| 207 | + boolean startBtn = Esplora.readButton(SWITCH_DOWN); |
| 208 | + |
| 209 | + if (startBtn != lastStartBtn) { |
| 210 | + if (startBtn == HIGH) { // button released |
| 211 | + active = !active; |
| 212 | + if (active) |
| 213 | + justActivated = true; |
| 214 | + } |
| 215 | + |
| 216 | + lastStartBtn = startBtn; |
| 217 | + } |
| 218 | +} |
| 219 | + |
0 commit comments