I have a “meteor shower” animation where whenever an event happens, a comet will run the length of the entire strip.
Here is the code for that:
/* ======================= includes ================================= */
#include "Particle.h"
#include <neopixel.h>
SYSTEM_MODE(AUTOMATIC);
#define PIXEL_COUNT 864
#define PIXEL_PIN D1
#define PIXEL_TYPE SK6812RGBW
Adafruit_NeoPixel strip(PIXEL_COUNT, PIXEL_PIN, PIXEL_TYPE);
struct ANIMATION_DATA {
uint8_t g;
uint8_t r;
uint8_t b;
int pos;
int len;
};
// circular buffer with 100 slots will overwrite unfinished animations when full
const int maxAnim = 100;
int animHead = 0;
int animTail = 0;
ANIMATION_DATA anim[maxAnim];
const uint8_t cometAnim[] = { 200, 150, 100, 75, 60, 50, 35, 25, 25, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
const uint8_t brightAnim[] = { 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 225, 225, 220, 200, 200, 200, 200, 190, 185, 185, 175, 165, 150, 140, 125, 120, 115, 100, 100, 90, 80, 75};
const int cometLen = sizeof(cometAnim) / sizeof(cometAnim[0]);
const uint32_t msRefresh = 0;
const int stepSpeed = 10;
void setup() {
memset(anim, 0, sizeof(anim));
strip.begin();
strip.show(); // Initialize all pixels to 'off'
Particle.function("led", addComet);
}
void loop() {
static uint32_t msDelay = 0;
if (millis() - msDelay < msRefresh) return;
msDelay = millis();
if (animHead != animTail) {
ANIMATION_DATA *a;
for (int i = animTail; i != animHead; ) {
a = &anim[i];
if (a->len) {
int px;
//Draw the Comet
for (int p = a->len-1; p >= 0; p--) {
px = constrain(a->pos - p, 0, strip.numPixels()-1);
//strip.setPixelColor(px, a->g, a->r, a->b, cometAnim[p]);
strip.setColorDimmed(px, a->g, a->r, a->b, cometAnim[p], brightAnim[p]);
}
//Erase the pixels behind comet as it moves forward
if (a->pos - a->len >= 0) {
for (int k = 0; k < stepSpeed; k++) {
strip.setPixelColor(a->pos-k - a->len, 0);
}
}
//Make comet jump forward stepSpeed pixels at a time to make comet run seemingly faster
if (a->pos < strip.numPixels()) {
a->pos+=stepSpeed;
}
else
a->len-=stepSpeed;
if (a->len <= 0) {
*a = { 0, 0, 0, 0, 0};
animTail++;
animTail %= maxAnim;
}
}
i++;
i %= maxAnim;
}
strip.show();
}
}
int addComet(String command) {
int retVal = -1;
if (command=="login") {
anim[animHead] = { 200, 200, 200, 0, cometLen }; // white
retVal = 0;
Particle.publish("login");
}
else if (command=="idea") {
anim[animHead] = { 255, 255, 0, 0, cometLen }; // yellow
retVal = 1;
Particle.publish("idea");
}
else if (command=="comment") {
anim[animHead] = { 0, 0, 255, 0, cometLen }; // blue
retVal = 2;
Particle.publish("comment");
}
else if (command=="outcome") {
anim[animHead] = { 255, 0, 0, 0, cometLen }; // green
retVal = 3;
Particle.publish("outcome");
}
else if (command=="project") {
anim[animHead] = { 255, 0, 0, 0, cometLen }; // green
retVal = 4;
Particle.publish("project");
}
else if (command=="status") {
anim[animHead] = { 100, 255, 0, 0, cometLen }; // orange
retVal = 5;
Particle.publish("status");
}
else if (command=="step") {
anim[animHead] = { 0, 75, 255, 0, cometLen }; // purple
retVal = 6;
Particle.publish("step");
}
else if (command=="vote") {
anim[animHead] = { 0, 255, 0, 0, cometLen }; // red
retVal = 7;
Particle.publish("vote");
}
else if (command=="view") {
anim[animHead] = { 105, 255, 200, 0, cometLen }; // pink
retVal = 7;
Particle.publish("view");
}
else {
return -1;
}
animHead++;
animHead %= maxAnim;
return retVal;
}
While this animation is running, I want to run a Sparkle animation in the background.
Here’s simple code I found for the Sparkle animation:
void Sparkle(byte green, byte red, byte blue, int wait) {
int Pixel = random(PIXEL_COUNT+1);
strip.setPixelColor(Pixel,green,red,blue);
strip.show();
delay(wait);
strip.setPixelColor(Pixel,0,0,0);
}
When I put the two codes together, it slows down the comet animation. Any way, to keep the two animations running at optimal speed?
Complete code with two animations in same program. (I just added the call to Sparkle after the calls at the end of the loop.):
/* ======================= includes ================================= */
#include "Particle.h"
#include <neopixel.h>
SYSTEM_MODE(AUTOMATIC);
#define PIXEL_COUNT 864
#define PIXEL_PIN D1
#define PIXEL_TYPE SK6812RGBW
Adafruit_NeoPixel strip(PIXEL_COUNT, PIXEL_PIN, PIXEL_TYPE);
struct ANIMATION_DATA {
uint8_t g;
uint8_t r;
uint8_t b;
int pos;
int len;
};
// circular buffer with 100 slots will overwrite unfinished animations when full
const int maxAnim = 100;
int animHead = 0;
int animTail = 0;
ANIMATION_DATA anim[maxAnim];
const uint8_t cometAnim[] = { 200, 150, 100, 75, 60, 50, 35, 25, 25, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
const uint8_t brightAnim[] = { 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 225, 225, 220, 200, 200, 200, 200, 190, 185, 185, 175, 165, 150, 140, 125, 120, 115, 100, 100, 90, 80, 75};
const int cometLen = sizeof(cometAnim) / sizeof(cometAnim[0]);
const uint32_t msRefresh = 0;
const int stepSpeed = 10;
void setup() {
memset(anim, 0, sizeof(anim));
strip.begin();
strip.show(); // Initialize all pixels to 'off'
Particle.function("led", addComet);
}
void loop() {
static uint32_t msDelay = 0;
if (millis() - msDelay < msRefresh) return;
msDelay = millis();
if (animHead != animTail) {
ANIMATION_DATA *a;
for (int i = animTail; i != animHead; ) {
a = &anim[i];
if (a->len) {
int px;
//Draw the Comet
for (int p = a->len-1; p >= 0; p--) {
px = constrain(a->pos - p, 0, strip.numPixels()-1);
//strip.setPixelColor(px, a->g, a->r, a->b, cometAnim[p]);
strip.setColorDimmed(px, a->g, a->r, a->b, cometAnim[p], brightAnim[p]);
}
//Erase the pixels behind comet as it moves forward
if (a->pos - a->len >= 0) {
for (int k = 0; k < stepSpeed; k++) {
strip.setPixelColor(a->pos-k - a->len, 0);
}
}
//Make comet jump forward stepSpeed pixels at a time to make comet run seemingly faster
if (a->pos < strip.numPixels()) {
a->pos+=stepSpeed;
}
else
a->len-=stepSpeed;
if (a->len <= 0) {
*a = { 0, 0, 0, 0, 0};
animTail++;
animTail %= maxAnim;
}
}
i++;
i %= maxAnim;
}
strip.show();
}
Sparkle(0xff, 0xff, 0xff, 0);
}
int addComet(String command) {
int retVal = -1;
if (command=="login") {
anim[animHead] = { 200, 200, 200, 0, cometLen }; // white
retVal = 0;
Particle.publish("login");
}
else if (command=="idea") {
anim[animHead] = { 255, 255, 0, 0, cometLen }; // yellow
retVal = 1;
Particle.publish("idea");
}
else if (command=="comment") {
anim[animHead] = { 0, 0, 255, 0, cometLen }; // blue
retVal = 2;
Particle.publish("comment");
}
else if (command=="outcome") {
anim[animHead] = { 255, 0, 0, 0, cometLen }; // green
retVal = 3;
Particle.publish("outcome");
}
else if (command=="project") {
anim[animHead] = { 255, 0, 0, 0, cometLen }; // green
retVal = 4;
Particle.publish("project");
}
else if (command=="status") {
anim[animHead] = { 100, 255, 0, 0, cometLen }; // orange
retVal = 5;
Particle.publish("status");
}
else if (command=="step") {
anim[animHead] = { 0, 75, 255, 0, cometLen }; // purple
retVal = 6;
Particle.publish("step");
}
else if (command=="vote") {
anim[animHead] = { 0, 255, 0, 0, cometLen }; // red
retVal = 7;
Particle.publish("vote");
}
else if (command=="view") {
anim[animHead] = { 105, 255, 200, 0, cometLen }; // pink
retVal = 7;
Particle.publish("view");
}
else {
return -1;
}
animHead++;
animHead %= maxAnim;
return retVal;
}
void Sparkle(byte green, byte red, byte blue, int SpeedDelay) {
int Pixel = random(PIXEL_COUNT);
strip.setPixelColor(Pixel,green,red,blue);
strip.show();
delay(SpeedDelay);
strip.setPixelColor(Pixel,0,0,0);
}