led_effects.cpp 8.49 KB
Newer Older
1
#include "led_effects.h"
2
3
4
5
6
7
8
9
10

#include "config.h"
#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"

11
12
13
14
15
/*****************************************************************
 * Configuration                                                 *
 *****************************************************************/
namespace config {
  const uint8_t max_brightness = MAX_BRIGHTNESS;
Eric Duminil's avatar
Eric Duminil committed
16
17
18
19
20
#if defined(MIN_BRIGHTNESS)
  const uint8_t min_brightness = MIN_BRIGHTNESS;
#else
  const uint8_t min_brightness = MAX_BRIGHTNESS;
#endif
21
  const uint8_t brightness_amplitude = config::max_brightness - config::min_brightness;
22
  const int kitt_tail = 3; // How many dimmer LEDs follow in K.I.T.T. wheel
23
  const uint16_t poor_air_quality_ppm = 1600; // Above this threshold, LED breathing effect is faster.
24
  bool display_led = true; // Will be set to false during "night mode".
25
26
27
28
29
30
31
32
33
34
35

#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°),
36
  // LEDs >= 1600ppm will be pure red (hue angle 0°), LEDs in-between will be yellowish.
37
38
  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
39
40
41
42
  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]
43
44
45
#else
#  error "Only 12 and 16 LEDs rings are currently supported."
#endif
46
47
}

48
49
#if defined(ESP8266)
// NeoPixels on GPIO05, aka D1 on ESP8266.
50
const int NEOPIXELS_PIN = 5;
51
52
53
54
55
#elif defined(ESP32)
// NeoPixels on GPIO23 on ESP32. To avoid conflict with LoRa_SCK on TTGO.
const int NEOPIXELS_PIN = 23;
#endif

56
Adafruit_NeoPixel pixels(config::led_count, NEOPIXELS_PIN, NEO_GRB + NEO_KHZ800);
57

58
namespace led_effects {
59
60
61
62
63
64
65
66
  //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
67
68
    //NOTE: OFF is LOW on ESP32 and HIGH on ESP8266 :-/
#ifdef ESP8266
69
    digitalWrite(ONBOARD_LED_PIN, HIGH);
Eric Duminil's avatar
Eric Duminil committed
70
71
72
#else
    digitalWrite(ONBOARD_LED_PIN, LOW);
#endif
73
74
75
  }

  void onBoardLEDOn() {
Eric Duminil's avatar
Eric Duminil committed
76
#ifdef ESP8266
77
    digitalWrite(ONBOARD_LED_PIN, LOW);
Eric Duminil's avatar
Eric Duminil committed
78
79
80
#else
    digitalWrite(ONBOARD_LED_PIN, HIGH);
#endif
81
82
  }

Eric Duminil's avatar
Eric Duminil committed
83
84
85
86
87
88
  void LEDsOff() {
    pixels.clear();
    pixels.show();
    onBoardLEDOff();
  }

Eric Duminil's avatar
Eric Duminil committed
89
  void showColor(int32_t color) {
90
    config::display_led = false; // In order to avoid overwriting the desired color next time CO2 is displayed
Eric Duminil's avatar
Eric Duminil committed
91
92
93
94
95
    pixels.setBrightness(255);
    pixels.fill(color);
    pixels.show();
  }

96
97
98
  void setupRing() {
    pixels.begin();
    pixels.setBrightness(config::max_brightness);
Eric Duminil's avatar
Eric Duminil committed
99
    LEDsOff();
100
    sensor_console::defineIntCommand("led", turnLEDsOnOff, F("0/1 (Turns LEDs on/off)"));
101
    sensor_console::defineIntCommand("color", showColor, F("0xFF0015 (Shows color, specified as RGB, for debugging)"));
102
103
104
  }

  void toggleNightMode() {
105
106
107
108
109
110
111
    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!"));
112
    } else {
113
114
      Serial.println(F("Night mode!"));
      LEDsOff();
115
116
117
118
119
    }
  }

  //NOTE: basically one iteration of KITT wheel
  void showWaitingLED(uint32_t color) {
120
    using namespace config;
121
    delay(80);
122
    if (!display_led) {
123
124
      return;
    }
Eric Duminil's avatar
Eric Duminil committed
125
    static uint16_t kitt_offset = 0;
126
    pixels.clear();
127
128
    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
129
130
131
      pixels.setPixelColor(ledNumber, color * pixels.gamma8(255 - j * 76) / 255);
    }
    pixels.show();
Eric Duminil's avatar
Eric Duminil committed
132
    kitt_offset++;
133
134
135
136
137
138
139
  }

  // 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) {
    pixels.setBrightness(config::max_brightness);
140
    for (int i = 0; i < duration_s * config::led_count; ++i) {
141
142
143
144
145
146
147
148
149
      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) {
150
    if (co2 >= config::co2_ticks[ledId + 1]) {
151
152
      return 255;
    } else {
153
      if (2 * co2 >= config::co2_ticks[ledId] + config::co2_ticks[ledId + 1]) {
154
155
156
157
158
159
160
161
162
        // 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;
      }
    }
  }

163
164
165
166
167
  /**
   * 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
168
    uint16_t brightness = config::min_brightness + pixels.sine8(breathing_offset) * config::brightness_amplitude / 255;
169
170
    pixels.setBrightness(brightness);
    pixels.show();
171
    breathing_offset += co2 > config::poor_air_quality_ppm ? 6 : 3; // breathing speed. +3 looks like slow human breathing.
172
173
  }

174
175
176
177
  /**
   * Fills the whole ring with green, yellow, orange or black, depending on co2 input and CO2_TICKS.
   */
  void displayCO2color(uint16_t co2) {
178
    if (!config::display_led) {
179
180
181
      return;
    }
    pixels.setBrightness(config::max_brightness);
182
    for (int ledId = 0; ledId < config::led_count; ++ledId) {
183
      uint8_t brightness = getLedBrightness(co2, ledId);
184
      pixels.setPixelColor(ledId, pixels.ColorHSV(config::led_hues[ledId], 255, brightness));
185
186
    }
    pixels.show();
Eric Duminil's avatar
Eric Duminil committed
187
    if (config::brightness_amplitude > 0) {
188
189
      breathe(co2);
    }
190
191
  }

192
  void showRainbowWheel(uint16_t duration_ms) {
193
    if (!config::display_led) {
194
195
      return;
    }
Eric Duminil's avatar
Eric Duminil committed
196
    static uint16_t wheel_offset = 0;
197
    static uint16_t sine_offset = 0;
Eric Duminil's avatar
Eric Duminil committed
198
    unsigned long t0 = millis();
199
    pixels.setBrightness(config::max_brightness);
Eric Duminil's avatar
Eric Duminil committed
200
    while (millis() - t0 < duration_ms) {
201
202
      for (int i = 0; i < config::led_count; i++) {
        pixels.setPixelColor(i, pixels.ColorHSV(i * 65535 / config::led_count + wheel_offset));
203
        wheel_offset += (pixels.sine8(sine_offset++ / 50) - 127) / 2;
204
205
206
207
208
209
210
      }
      pixels.show();
      delay(10);
    }
  }

  void redAlert() {
211
    if (!config::display_led) {
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
      onBoardLEDOn();
      delay(500);
      onBoardLEDOff();
      delay(500);
      return;
    }
    for (int i = 0; i < 10; i++) {
      pixels.setBrightness(static_cast<int>(config::max_brightness * (1 - i * 0.1)));
      delay(50);
      pixels.fill(color::red);
      pixels.show();
    }
  }

  /**
227
228
229
230
231
   * 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.
232
   */
233
  bool countdownToZero() {
234
    if (!config::display_led) {
Eric Duminil's avatar
Eric Duminil committed
235
      Serial.println(F("Night mode. Not doing anything."));
236
      delay(1000); // Wait for a while, to avoid coming back to this function too many times when button is pressed.
237
      return false;
238
239
240
241
    }
    pixels.fill(color::blue);
    pixels.show();
    int countdown;
242
    for (countdown = config::led_count; countdown >= 0 && !digitalRead(0); countdown--) {
243
244
245
246
247
      pixels.setPixelColor(countdown, color::black);
      pixels.show();
      Serial.println(countdown);
      delay(500);
    }
248
    return countdown < 0;
249
250
  }
}