How to make two animations run at the same time

One of the problems - as already mentioned in this thread - is that you are calling strip.show() more than once per loop() iteration.
strip.show() is the most time consuming function in your whole project.

The logic of my original code is to setup all the LEDs according to all concurrent animations in the pixel buffer only and only after all pixels have been set to the correct combination of values call strip.show() once and only once per visit to loop().

The other thing why you are not seeing the concurrent animations is that we got rid of the for() loop that took care of multiple animations - for testing purposes only tho'


Have a try with this then

#include <Particle.h>
#include <neopixel.h>

SYSTEM_MODE(AUTOMATIC)

const int PIXEL_COUNT = 864;
const int PIXEL_PIN   = D1;
const int 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  whiteAnim[] = 
{ 255, 200, 175, 160, 155, 150, 145, 140, 135, 130
, 125, 120, 115, 110, 105, 100,  95,  90,  85,  80
,  75,  70,  65,  60,  55,  50,  45,  40,  35,  30
,  25,  20,  15,  10,   5,   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,   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, 255, 255, 255, 255, 255
, 255, 255, 255, 255, 255, 255, 255, 255, 245, 240
, 240, 240, 240, 235, 230, 230, 230, 230, 225, 220
, 220, 220, 220, 215, 210, 210, 210, 210, 205, 200
, 200, 200, 200, 200, 195, 195, 190, 190, 190, 190
, 190, 195, 180, 180, 180, 175, 170, 165, 160, 155
, 150, 145, 140, 135, 130, 125, 120, 115, 110, 105
, 100,  95,  90,  85,  80,  75,  70 
};
const int      cometLen = sizeof(whiteAnim) / sizeof(whiteAnim[0]);
const uint32_t msRefresh = 0;
const int stepSpeed = 10;
static int animationStep = 0;

void setup()
{
  memset(anim, 0, sizeof(anim));

  strip.begin();
  strip.show(); // Initialize all pixels to 'off'
  Particle.function("led", addComet);
  //strip.setBrightness(30);
}

void loop()
{
  static uint32_t msDelay = 0;
  if (millis() - msDelay < msRefresh) return;
  msDelay = millis();
  
  static bool sparkleState = false;
  static int  sparklePixel = 0;

  if (animHead != animTail)
  {
    ANIMATION_DATA *a;
    for (int animationStep = animTail; animationStep != animHead; ) 
    {
      a = &anim[animationStep];
      if (a->len)
      {
        int px;

        //Draw the Comet
        for (int p = a->len - 1; p >= 0; p--)
        {
          px = constrain(a->pos - p, 0, PIXEL_COUNT - 1);
          strip.setColorDimmed(px, a->g, a->r, a->b, whiteAnim[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 < PIXEL_COUNT + stepSpeed)
          a->pos += stepSpeed;
        else
          a->len -= stepSpeed;

        if (a->len <= 0)
        {
          *a = { 0, 0, 0, 0, 0 };
          animTail++;
          animTail %= maxAnim;
        }
      }

      animationStep++;
      animationStep %= maxAnim;
    }
  }

  sparkleState = !sparkleState;  // flip state
  if (sparkleState)
  {
    sparklePixel = random(PIXEL_COUNT);
    strip.setPixelColor(sparklePixel, random(255), random(255), random(255), 100);
  }
  else
    strip.setPixelColor(sparklePixel, 0, 0, 0, 0);

  strip.show();
}

struct topic {
  char    title[16];
  uint8_t rgb[3];
};

const topic topicList[] = 
{ { "login"  , { 200, 200, 200 } }
, { "idea"   , { 255, 255,   0 } }
, { "comment", {   0,   0, 255 } }
, { "outcome", { 255,   0,   0 } }
, { "project", { 255,   0,   0 } }
, { "status" , { 255,   0,   0 } }
, { "step"   , {   0,  75, 255 } }
, { "vote"   , {   0, 255,   0 } }
, { "view"   , { 105, 255, 200 } }
};
const int   topics = sizeof(topicList)/sizeof(topicList[0]);

int addComet(String command)
{
  int retVal = -1;

  for (int i = 0; i < topics; i++)
  {
    if (strcmp((const char*)command, topicList[i].title) == 0)
    {
      anim[animHead] = { topicList[i].rgb[0], topicList[i].rgb[1], topicList[i].rgb[2], 0, cometLen }; // purple
      Particle.publish(command);
      animHead++;
      animHead %= maxAnim;
      retVal = i;
      break;
    }
  }
  return retVal;
}
1 Like