led_effects.cpp 8.09 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
  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°),
27
  // LEDs >= 1600ppm will be pure red (hue angle 0°), LEDs in-between will be yellowish.
28
29
  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
30
31
32
33
  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]
34
35
36
#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
213
      }
      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();
    }
  }

  /**
214
215
216
217
218
   * 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.
219
   */
220
  bool countdownToZero() {
221
    if (config::night_mode) {
Eric Duminil's avatar
Eric Duminil committed
222
      Serial.println(F("Night mode. Not doing anything."));
223
      delay(1000); // Wait for a while, to avoid coming back to this function too many times when button is pressed.
224
      return false;
225
226
227
228
    }
    pixels.fill(color::blue);
    pixels.show();
    int countdown;
229
    for (countdown = config::led_count; countdown >= 0 && !digitalRead(0); countdown--) {
230
231
232
233
234
      pixels.setPixelColor(countdown, color::black);
      pixels.show();
      Serial.println(countdown);
      delay(500);
    }
235
    return countdown < 0;
236
237
  }
}