led_effects.cpp 8.46 KB
Newer Older
1
#include "led_effects.h"
2
3

#include "config.h"
Eric Duminil's avatar
Eric Duminil committed
4
#include "web_config.h"
5
6
7
8
9
10
11
#include "sensor_console.h"

// Adafruit NeoPixel (Arduino library for controlling single-wire-based LED pixels and strip)
// https://github.com/adafruit/Adafruit_NeoPixel
// Documentation : http://adafruit.github.io/Adafruit_NeoPixel/html/class_adafruit___neo_pixel.html
#include "src/lib/Adafruit_NeoPixel/Adafruit_NeoPixel.h"

12
13
14
15
/*****************************************************************
 * Configuration                                                 *
 *****************************************************************/
namespace config {
16
  const int kitt_tail = 3; // How many dimmer LEDs follow in K.I.T.T. wheel
17
  const uint16_t poor_air_quality_ppm = 1600; // Above this threshold, LED breathing effect is faster.
18
  bool display_led = true; // Will be set to false during "night mode".
19
20
21
22
23
24
25
26
27
28
29

#if !defined(LED_COUNT)
#  define LED_COUNT 12
#endif

  const uint16_t led_count = LED_COUNT;

#if LED_COUNT == 12
  //NOTE: One value has been prepended, to make calculations easier and avoid out of bounds index.
  const uint16_t co2_ticks[led_count + 1] = { 0, 500, 600, 700, 800, 900, 1000, 1200, 1400, 1600, 1800, 2000, 2200 }; // [ppm]
  // For a given LED, which color should be displayed? First LED will be pure green (hue angle 120°),
30
  // LEDs >= 1600ppm will be pure red (hue angle 0°), LEDs in-between will be yellowish.
31
32
  const uint16_t led_hues[led_count] = { 21845U, 19114U, 16383U, 13653U, 10922U, 8191U, 5461U, 2730U, 0, 0, 0, 0 }; // [hue angle]
#elif LED_COUNT == 16
Eric Duminil's avatar
Eric Duminil committed
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
  const uint16_t co2_ticks[led_count + 1] = {
      0,
      500,
      600,
      700,
      800,
      900,
      1000,
      1100,
      1200,
      1300,
      1400,
      1500,
      1600,
      1700,
      1800,
      2000,
      2200 }; // [ppm]
  const uint16_t led_hues[led_count] = {
      21845U,
      19859U,
      17873U,
      15887U,
      13901U,
      11915U,
      9929U,
      7943U,
      5957U,
      3971U,
      1985U,
      0,
      0,
      0,
      0,
      0 }; // [hue angle]
68
69
70
#else
#  error "Only 12 and 16 LEDs rings are currently supported."
#endif
71
72
}

73
74
#if defined(ESP8266)
// NeoPixels on GPIO05, aka D1 on ESP8266.
75
const int NEOPIXELS_PIN = 5;
76
77
78
79
80
#elif defined(ESP32)
// NeoPixels on GPIO23 on ESP32. To avoid conflict with LoRa_SCK on TTGO.
const int NEOPIXELS_PIN = 23;
#endif

81
Adafruit_NeoPixel pixels(config::led_count, NEOPIXELS_PIN, NEO_GRB + NEO_KHZ800);
82

83
namespace led_effects {
84
85
86
87
88
89
90
91
  //On-board LED on D4, aka GPIO02
  const int ONBOARD_LED_PIN = 2;

  void setupOnBoardLED() {
    pinMode(ONBOARD_LED_PIN, OUTPUT);
  }

  void onBoardLEDOff() {
Eric Duminil's avatar
Eric Duminil committed
92
93
    //NOTE: OFF is LOW on ESP32 and HIGH on ESP8266 :-/
#ifdef ESP8266
94
    digitalWrite(ONBOARD_LED_PIN, HIGH);
Eric Duminil's avatar
Eric Duminil committed
95
96
97
#else
    digitalWrite(ONBOARD_LED_PIN, LOW);
#endif
98
99
100
  }

  void onBoardLEDOn() {
Eric Duminil's avatar
Eric Duminil committed
101
#ifdef ESP8266
102
    digitalWrite(ONBOARD_LED_PIN, LOW);
Eric Duminil's avatar
Eric Duminil committed
103
104
105
#else
    digitalWrite(ONBOARD_LED_PIN, HIGH);
#endif
106
107
  }

Eric Duminil's avatar
Eric Duminil committed
108
109
110
111
112
113
  void LEDsOff() {
    pixels.clear();
    pixels.show();
    onBoardLEDOff();
  }

Eric Duminil's avatar
Eric Duminil committed
114
  void showColor(int32_t color) {
115
    config::display_led = false; // In order to avoid overwriting the desired color next time CO2 is displayed
Eric Duminil's avatar
Eric Duminil committed
116
117
118
119
120
    pixels.setBrightness(255);
    pixels.fill(color);
    pixels.show();
  }

121
122
  void setupRing() {
    pixels.begin();
Eric Duminil's avatar
Eric Duminil committed
123
    pixels.setBrightness(*config::max_brightness);
Eric Duminil's avatar
Eric Duminil committed
124
    LEDsOff();
125
    sensor_console::defineIntCommand("led", turnLEDsOnOff, F("0/1 (Turns LEDs on/off)"));
126
    sensor_console::defineIntCommand("color", showColor, F("0xFF0015 (Shows color, specified as RGB, for debugging)"));
127
128
129
  }

  void toggleNightMode() {
130
131
132
133
134
135
136
    turnLEDsOnOff(!config::display_led);
  }

  void turnLEDsOnOff(int32_t display_led) {
    config::display_led = display_led;
    if (config::display_led) {
      Serial.println(F("LEDs are on!"));
137
    } else {
138
139
      Serial.println(F("Night mode!"));
      LEDsOff();
140
141
142
143
144
    }
  }

  //NOTE: basically one iteration of KITT wheel
  void showWaitingLED(uint32_t color) {
145
    using namespace config;
146
    delay(80);
147
    if (!display_led) {
148
149
      return;
    }
Eric Duminil's avatar
Eric Duminil committed
150
    static uint16_t kitt_offset = 0;
151
    pixels.clear();
152
153
    for (int j = kitt_tail; j >= 0; j--) {
      int ledNumber = abs((kitt_offset - j + led_count) % (2 * led_count) - led_count) % led_count; // Triangular function
154
155
156
      pixels.setPixelColor(ledNumber, color * pixels.gamma8(255 - j * 76) / 255);
    }
    pixels.show();
Eric Duminil's avatar
Eric Duminil committed
157
    kitt_offset++;
158
159
160
161
162
163
  }

  // Start K.I.T.T. led effect. Red color as default.
  // Simulate a moving LED with tail. First LED starts at 0, and moves along a triangular function. The tail follows, with decreasing brightness.
  // Takes approximately 1s for each direction.
  void showKITTWheel(uint32_t color, uint16_t duration_s) {
Eric Duminil's avatar
Eric Duminil committed
164
    pixels.setBrightness(*config::max_brightness);
165
    for (int i = 0; i < duration_s * config::led_count; ++i) {
166
167
168
169
170
171
172
173
174
      showWaitingLED(color);
    }
  }

  /*
   * For a given CO2 level and ledId, which brightness should be displayed? 0 for off, 255 for on. Something in-between for partial LED.
   * For example, for 1500ppm, every LED between 0 and 7 (500 -> 1400ppm) should be on, LED at 8 (1600ppm) should be half-on.
   */
  uint8_t getLedBrightness(uint16_t co2, int ledId) {
175
    if (co2 >= config::co2_ticks[ledId + 1]) {
176
177
      return 255;
    } else {
178
      if (2 * co2 >= config::co2_ticks[ledId] + config::co2_ticks[ledId + 1]) {
179
180
181
182
183
184
185
186
187
        // Show partial LED if co2 more than halfway between ticks.
        return 27; // Brightness isn't linear, so 27 / 255 looks much brighter than 10%
      } else {
        // LED off because co2 below previous tick
        return 0;
      }
    }
  }

188
189
190
191
192
  /**
   * If enabled, slowly varies the brightness between MAX_BRIGHTNESS & MIN_BRIGHTNESS.
   */
  void breathe(int16_t co2) {
    static uint8_t breathing_offset = 0;
Eric Duminil's avatar
Eric Duminil committed
193
194
    uint8_t brightness_amplitude = *config::max_brightness - *config::min_brightness;
    uint16_t brightness = *config::min_brightness + pixels.sine8(breathing_offset) * brightness_amplitude / 255;
195
196
    pixels.setBrightness(brightness);
    pixels.show();
197
    breathing_offset += co2 > config::poor_air_quality_ppm ? 6 : 3; // breathing speed. +3 looks like slow human breathing.
198
199
  }

200
201
202
203
  /**
   * Fills the whole ring with green, yellow, orange or black, depending on co2 input and CO2_TICKS.
   */
  void displayCO2color(uint16_t co2) {
204
    if (!config::display_led) {
205
206
      return;
    }
Eric Duminil's avatar
Eric Duminil committed
207
    pixels.setBrightness(*config::max_brightness);
208
    for (int ledId = 0; ledId < config::led_count; ++ledId) {
209
      uint8_t brightness = getLedBrightness(co2, ledId);
210
      pixels.setPixelColor(ledId, pixels.ColorHSV(config::led_hues[ledId], 255, brightness));
211
212
    }
    pixels.show();
Eric Duminil's avatar
Eric Duminil committed
213
    if (*config::max_brightness > *config::min_brightness) {
214
215
      breathe(co2);
    }
216
217
  }

218
  void showRainbowWheel(uint16_t duration_ms) {
219
    if (!config::display_led) {
220
221
      return;
    }
Eric Duminil's avatar
Eric Duminil committed
222
    static uint16_t wheel_offset = 0;
223
    static uint16_t sine_offset = 0;
Eric Duminil's avatar
Eric Duminil committed
224
    unsigned long t0 = millis();
Eric Duminil's avatar
Eric Duminil committed
225
    pixels.setBrightness(*config::max_brightness);
Eric Duminil's avatar
Eric Duminil committed
226
    while (millis() - t0 < duration_ms) {
227
228
      for (int i = 0; i < config::led_count; i++) {
        pixels.setPixelColor(i, pixels.ColorHSV(i * 65535 / config::led_count + wheel_offset));
229
        wheel_offset += (pixels.sine8(sine_offset++ / 50) - 127) / 2;
230
231
232
233
234
235
236
      }
      pixels.show();
      delay(10);
    }
  }

  void redAlert() {
237
    if (!config::display_led) {
238
239
240
241
242
243
244
      onBoardLEDOn();
      delay(500);
      onBoardLEDOff();
      delay(500);
      return;
    }
    for (int i = 0; i < 10; i++) {
Eric Duminil's avatar
Eric Duminil committed
245
      pixels.setBrightness(static_cast<int>(*config::max_brightness * (1 - i * 0.1)));
246
247
248
249
250
251
252
      delay(50);
      pixels.fill(color::red);
      pixels.show();
    }
  }

  /**
253
254
255
256
257
   * Displays a complete blue circle, and starts removing LEDs one by one.
   * Does nothing in night mode and returns false then. Returns true if
   * the countdown has finished. Can be used for calibration, e.g. when countdown is 0.
   * NOTE: This function is blocking and returns only after the button has
   * been released or after every LED has been turned off.
258
   */
259
  bool countdownToZero() {
260
    if (!config::display_led) {
Eric Duminil's avatar
Eric Duminil committed
261
      Serial.println(F("Night mode. Not doing anything."));
262
      delay(1000); // Wait for a while, to avoid coming back to this function too many times when button is pressed.
263
      return false;
264
265
266
267
    }
    pixels.fill(color::blue);
    pixels.show();
    int countdown;
268
    for (countdown = config::led_count; countdown >= 0 && !digitalRead(0); countdown--) {
269
270
271
272
273
      pixels.setPixelColor(countdown, color::black);
      pixels.show();
      Serial.println(countdown);
      delay(500);
    }
274
    return countdown < 0;
275
276
  }
}