led_effects.cpp 8.13 KB
Newer Older
1
2
3
4
5
6
#include "led_effects.h"
/*****************************************************************
 * Configuration                                                 *
 *****************************************************************/
namespace config {
  const uint8_t max_brightness = MAX_BRIGHTNESS;
Eric Duminil's avatar
Eric Duminil committed
7
8
9
10
11
#if defined(MIN_BRIGHTNESS)
  const uint8_t min_brightness = MIN_BRIGHTNESS;
#else
  const uint8_t min_brightness = MAX_BRIGHTNESS;
#endif
12
  const uint8_t brightness_amplitude = config::max_brightness - config::min_brightness;
13
  const int kitt_tail = 3; // How many dimmer LEDs follow in K.I.T.T. wheel
14
  const uint16_t poor_air_quality_ppm = 1600; // Above this threshold, LED breathing effect is faster.
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
  bool night_mode = false; //NOTE: Use a class instead? NightMode could then be another state.

#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°),
  // last 4 LEDs will be pure red (hue angle 0°), LEDs in-between will be yellowish.
  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
  const uint16_t co2_ticks[led_count + 1] = { 0, 400, 500, 600, 700, 800, 900, 1000, 1100,
                                           1200, 1300, 1400, 1500, 1600, 1800, 2000, 2200 }; // [ppm]
  const uint16_t led_hues[led_count] = { 21845U, 20024U, 18204U, 16383U, 14563U, 12742U, 10922U, 9102U,
                                          7281U, 5461U, 3640U, 1820U, 0, 0, 0, 0 }; // [hue angle]
#else
#  error "Only 12 and 16 LEDs rings are currently supported."
#endif
37
38
}

39
40
#if defined(ESP8266)
// NeoPixels on GPIO05, aka D1 on ESP8266.
41
const int NEOPIXELS_PIN = 5;
42
43
44
45
46
#elif defined(ESP32)
// NeoPixels on GPIO23 on ESP32. To avoid conflict with LoRa_SCK on TTGO.
const int NEOPIXELS_PIN = 23;
#endif

47
Adafruit_NeoPixel pixels(config::led_count, NEOPIXELS_PIN, NEO_GRB + NEO_KHZ800);
48

49
namespace led_effects {
50
51
52
53
54
55
56
57
  //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
58
59
    //NOTE: OFF is LOW on ESP32 and HIGH on ESP8266 :-/
#ifdef ESP8266
60
    digitalWrite(ONBOARD_LED_PIN, HIGH);
Eric Duminil's avatar
Eric Duminil committed
61
62
63
#else
    digitalWrite(ONBOARD_LED_PIN, LOW);
#endif
64
65
66
  }

  void onBoardLEDOn() {
Eric Duminil's avatar
Eric Duminil committed
67
#ifdef ESP8266
68
    digitalWrite(ONBOARD_LED_PIN, LOW);
Eric Duminil's avatar
Eric Duminil committed
69
70
71
#else
    digitalWrite(ONBOARD_LED_PIN, HIGH);
#endif
72
73
  }

Eric Duminil's avatar
Eric Duminil committed
74
75
76
77
78
79
  void LEDsOff() {
    pixels.clear();
    pixels.show();
    onBoardLEDOff();
  }

Eric Duminil's avatar
Eric Duminil committed
80
81
82
83
84
85
86
  void showColor(int32_t color) {
    config::night_mode = true; // In order to avoid overwriting the desired color next time CO2 is displayed
    pixels.setBrightness(255);
    pixels.fill(color);
    pixels.show();
  }

87
88
89
  void setupRing() {
    pixels.begin();
    pixels.setBrightness(config::max_brightness);
Eric Duminil's avatar
Eric Duminil committed
90
    LEDsOff();
91
92
    sensor_console::defineCommand("night_mode", toggleNightMode, F("(Toggles night mode on/off)"));
    sensor_console::defineIntCommand("color", showColor, F("0xFF0015 (Shows color, specified as RGB, for debugging)"));
93
94
95
96
97
98
  }

  void toggleNightMode() {
    config::night_mode = !config::night_mode;
    if (config::night_mode) {
      Serial.println(F("NIGHT MODE!"));
Eric Duminil's avatar
Eric Duminil committed
99
      LEDsOff();
100
101
102
103
104
105
106
    } else {
      Serial.println(F("DAY MODE!"));
    }
  }

  //NOTE: basically one iteration of KITT wheel
  void showWaitingLED(uint32_t color) {
107
    using namespace config;
108
    delay(80);
109
    if (night_mode) {
110
111
      return;
    }
Eric Duminil's avatar
Eric Duminil committed
112
    static uint16_t kitt_offset = 0;
113
    pixels.clear();
114
115
    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
116
117
118
      pixels.setPixelColor(ledNumber, color * pixels.gamma8(255 - j * 76) / 255);
    }
    pixels.show();
Eric Duminil's avatar
Eric Duminil committed
119
    kitt_offset++;
120
121
122
123
124
125
126
  }

  // 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);
127
    for (int i = 0; i < duration_s * config::led_count; ++i) {
128
129
130
131
132
133
134
135
136
      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) {
137
    if (co2 >= config::co2_ticks[ledId + 1]) {
138
139
      return 255;
    } else {
140
      if (2 * co2 >= config::co2_ticks[ledId] + config::co2_ticks[ledId + 1]) {
141
142
143
144
145
146
147
148
149
        // 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;
      }
    }
  }

150
151
152
153
154
  /**
   * 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
155
    uint16_t brightness = config::min_brightness + pixels.sine8(breathing_offset) * config::brightness_amplitude / 255;
156
157
    pixels.setBrightness(brightness);
    pixels.show();
158
    breathing_offset += co2 > config::poor_air_quality_ppm ? 6 : 3; // breathing speed. +3 looks like slow human breathing.
159
160
  }

161
162
163
164
165
166
167
168
  /**
   * Fills the whole ring with green, yellow, orange or black, depending on co2 input and CO2_TICKS.
   */
  void displayCO2color(uint16_t co2) {
    if (config::night_mode) {
      return;
    }
    pixels.setBrightness(config::max_brightness);
169
    for (int ledId = 0; ledId < config::led_count; ++ledId) {
170
      uint8_t brightness = getLedBrightness(co2, ledId);
171
      pixels.setPixelColor(ledId, pixels.ColorHSV(config::led_hues[ledId], 255, brightness));
172
173
    }
    pixels.show();
Eric Duminil's avatar
Eric Duminil committed
174
    if (config::brightness_amplitude > 0) {
175
176
      breathe(co2);
    }
177
178
  }

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

  void redAlert() {
    if (config::night_mode) {
      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();
    }
  }

213
214
215
216
217
218
219
220
221
  void greenAlert() {
    static uint8_t i;
    i = (i + 1) % 10;
    pixels.setBrightness(static_cast<int>(config::max_brightness * (1 - i * 0.1)));
    delay(50);
    pixels.fill(color::green);
    pixels.show();
  }

222
223
224
225
226
227
  /**
   * Displays a complete blue circle, and starts removing LEDs one by one. Returns the number of remaining LEDs.
   * Can be used for calibration, e.g. when countdown is 0. Does not work in night mode.
   */
  int countdownToZero() {
    if (config::night_mode) {
Eric Duminil's avatar
Eric Duminil committed
228
      Serial.println(F("Night mode. Not doing anything."));
229
230
231
232
233
234
      delay(1000); // Wait for a while, to avoid coming back to this function too many times when button is pressed.
      return 1;
    }
    pixels.fill(color::blue);
    pixels.show();
    int countdown;
235
    for (countdown = config::led_count; countdown >= 0 && !digitalRead(0); countdown--) {
236
237
238
239
240
241
242
243
      pixels.setPixelColor(countdown, color::black);
      pixels.show();
      Serial.println(countdown);
      delay(500);
    }
    return countdown;
  }
}