Blending
This commit is contained in:
		| @@ -382,10 +382,8 @@ typedef struct Segment { | ||||
|     static uint16_t maxWidth, maxHeight;  // these define matrix width & height (max. segment dimensions) | ||||
|  | ||||
|     typedef struct TemporarySegmentData { | ||||
|       uint16_t _optionsT; | ||||
|       uint32_t _colorT[NUM_COLORS]; | ||||
|       //uint8_t  _opacityT; | ||||
|       //uint8_t  _cctT;        // temporary CCT | ||||
|       //uint16_t _optionsT; | ||||
|       uint8_t  _speedT; | ||||
|       uint8_t  _intensityT; | ||||
|       uint8_t  _custom1T, _custom2T;   // custom FX parameters/sliders | ||||
| @@ -560,8 +558,8 @@ typedef struct Segment { | ||||
|     // transition functions | ||||
|     void     startTransition(uint16_t dur); // transition has to start before actual segment values change | ||||
|     void     handleTransition(void); | ||||
|     void     saveSegenv(tmpsegd_t *tmpSegD); | ||||
|     void     restoreSegenv(tmpsegd_t *tmpSegD); | ||||
|     void     saveSegenv(tmpsegd_t *tmpSegD = nullptr); | ||||
|     void     restoreSegenv(tmpsegd_t *tmpSegD = nullptr); | ||||
|     uint16_t progress(void); //transition progression between 0-65535 | ||||
|     uint8_t  currentBri(uint8_t briNew, bool useCct = false); | ||||
|     uint8_t  currentMode(uint8_t modeNew); | ||||
|   | ||||
| @@ -216,23 +216,27 @@ void /*IRAM_ATTR*/ Segment::setPixelColorXY(int x, int y, uint32_t col) | ||||
|   y *= groupLength(); // expand to physical pixels | ||||
|   if (x >= width() || y >= height()) return;  // if pixel would fall out of segment just exit | ||||
|  | ||||
|   uint32_t tmpCol = col; | ||||
|   for (int j = 0; j < grouping; j++) {   // groupping vertically | ||||
|     for (int g = 0; g < grouping; g++) { // groupping horizontally | ||||
|       uint16_t xX = (x+g), yY = (y+j); | ||||
|       if (xX >= width() || yY >= height()) continue; // we have reached one dimension's end | ||||
|  | ||||
|       strip.setPixelColorXY(start + xX, startY + yY, col); | ||||
|       // if blending modes, blend with underlying pixel | ||||
|       if (transitional && _t && currentMode(mode) != _t->_modeT) tmpCol = color_blend(strip.getPixelColorXY(start + xX, startY + yY), col, progress(), true); | ||||
|  | ||||
|       strip.setPixelColorXY(start + xX, startY + yY, tmpCol); | ||||
|  | ||||
|       if (mirror) { //set the corresponding horizontally mirrored pixel | ||||
|         if (transpose) strip.setPixelColorXY(start + xX, startY + height() - yY - 1, col); | ||||
|         else           strip.setPixelColorXY(start + width() - xX - 1, startY + yY, col); | ||||
|         if (transpose) strip.setPixelColorXY(start + xX, startY + height() - yY - 1, tmpCol); | ||||
|         else           strip.setPixelColorXY(start + width() - xX - 1, startY + yY, tmpCol); | ||||
|       } | ||||
|       if (mirror_y) { //set the corresponding vertically mirrored pixel | ||||
|         if (transpose) strip.setPixelColorXY(start + width() - xX - 1, startY + yY, col); | ||||
|         else           strip.setPixelColorXY(start + xX, startY + height() - yY - 1, col); | ||||
|         if (transpose) strip.setPixelColorXY(start + width() - xX - 1, startY + yY, tmpCol); | ||||
|         else           strip.setPixelColorXY(start + xX, startY + height() - yY - 1, tmpCol); | ||||
|       } | ||||
|       if (mirror_y && mirror) { //set the corresponding vertically AND horizontally mirrored pixel | ||||
|         strip.setPixelColorXY(width() - xX - 1, height() - yY - 1, col); | ||||
|         strip.setPixelColorXY(width() - xX - 1, height() - yY - 1, tmpCol); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -288,9 +288,7 @@ void Segment::startTransition(uint16_t dur) { | ||||
|   _t = new Transition(dur); // no previous transition running | ||||
|   if (!_t) return; // failed to allocate data | ||||
|  | ||||
|   DEBUG_PRINT(F("-- Saving transition environment. ")); | ||||
|   DEBUG_PRINTLN(on ? opacity : 0); | ||||
|   saveSegenv(&(_t->_tmpSeg)); | ||||
|   saveSegenv(); | ||||
|   CRGBPalette16 _palT = CRGBPalette16(DEFAULT_COLOR); loadPalette(_palT, palette); | ||||
|   _t->_palT             = _palT; | ||||
|   _t->_modeT            = mode; | ||||
| @@ -328,18 +326,14 @@ uint8_t Segment::currentBri(uint8_t briNew, bool useCct) { | ||||
| } | ||||
|  | ||||
| uint8_t Segment::currentMode(uint8_t newMode) { | ||||
|   uint16_t prog = progress(); | ||||
|   if (prog < 0xFFFFU) { // implicit check for transitional & _t in progress() | ||||
|     restoreSegenv(&(_t->_tmpSeg)); | ||||
|     opacity  -= (uint32_t)opacity * prog / 0xFFFFU; | ||||
|     return _t->_modeT; | ||||
|   } | ||||
|   uint16_t prog = progress(); // implicit check for transitional & _t in progress() | ||||
|   if (prog < 0xFFFFU) return _t->_modeT; | ||||
|   return newMode; | ||||
| } | ||||
|  | ||||
| void Segment::saveSegenv(tmpsegd_t *tmpSeg) { | ||||
|   //tmpSeg._opacityT   = on ? opacity : 0; | ||||
|   //tmpSeg._optionsT   = options; | ||||
|   if (tmpSeg == nullptr) { if (_t) tmpSeg = &(_t->_tmpSeg); else return; } | ||||
|   tmpSeg->_optionsT   = options; | ||||
|   for (size_t i=0; i<NUM_COLORS; i++) tmpSeg->_colorT[i] = colors[i]; | ||||
|   tmpSeg->_speedT     = speed; | ||||
|   tmpSeg->_intensityT = intensity; | ||||
| @@ -358,17 +352,19 @@ void Segment::saveSegenv(tmpsegd_t *tmpSeg) { | ||||
| } | ||||
|  | ||||
| void Segment::restoreSegenv(tmpsegd_t *tmpSeg) { | ||||
|   if (&(_t->_tmpSeg) != tmpSeg) { | ||||
|     DEBUG_PRINTF("Temp: %p != %p\n", &(_t->_tmpSeg), tmpSeg); | ||||
|     DEBUG_PRINTLN(F("-- Restoring OLD environment.")); | ||||
|     // update possibly changed variables to keep old effect running correctly | ||||
|     _t->_tmpSeg._aux0T = aux0; | ||||
|     _t->_tmpSeg._aux1T = aux1; | ||||
|     _t->_tmpSeg._stepT = step; | ||||
|     _t->_tmpSeg._callT = call; | ||||
|   if (tmpSeg == nullptr) { | ||||
|     if (_t) tmpSeg = &(_t->_tmpSeg); | ||||
|     else return; | ||||
|   } else { | ||||
|     if (&(_t->_tmpSeg) != tmpSeg) { | ||||
|       // update possibly changed variables to keep old effect running correctly | ||||
|       _t->_tmpSeg._aux0T = aux0; | ||||
|       _t->_tmpSeg._aux1T = aux1; | ||||
|       _t->_tmpSeg._stepT = step; | ||||
|       _t->_tmpSeg._callT = call; | ||||
|     } | ||||
|   } | ||||
|   //opacity   = tmpSeg._opacityT; | ||||
|   //options   = tmpSeg._optionsT; | ||||
|   options   = tmpSeg->_optionsT; | ||||
|   for (size_t i=0; i<NUM_COLORS; i++) colors[i] = tmpSeg->_colorT[i]; | ||||
|   speed     = tmpSeg->_speedT; | ||||
|   intensity = tmpSeg->_intensityT; | ||||
| @@ -703,6 +699,7 @@ void IRAM_ATTR Segment::setPixelColor(int i, uint32_t col) | ||||
|   } | ||||
|   i += start; // starting pixel in a group | ||||
|  | ||||
|   uint32_t tmpCol = col; | ||||
|   // set all the pixels in the group | ||||
|   for (int j = 0; j < grouping; j++) { | ||||
|     uint16_t indexSet = i + ((reverse) ? -j : j); | ||||
| @@ -711,11 +708,13 @@ void IRAM_ATTR Segment::setPixelColor(int i, uint32_t col) | ||||
|         uint16_t indexMir = stop - indexSet + start - 1; | ||||
|         indexMir += offset; // offset/phase | ||||
|         if (indexMir >= stop) indexMir -= len; // wrap | ||||
|         strip.setPixelColor(indexMir, col); | ||||
|         if (transitional && _t && currentMode(mode) != _t->_modeT) tmpCol = color_blend(strip.getPixelColor(indexMir), col, progress(), true); | ||||
|         strip.setPixelColor(indexMir, tmpCol); | ||||
|       } | ||||
|       indexSet += offset; // offset/phase | ||||
|       if (indexSet >= stop) indexSet -= len; // wrap | ||||
|       strip.setPixelColor(indexSet, col); | ||||
|       if (transitional && _t && currentMode(mode) != _t->_modeT) tmpCol = color_blend(strip.getPixelColor(indexSet), col, progress(), true); | ||||
|       strip.setPixelColor(indexSet, tmpCol); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1176,24 +1175,22 @@ void WS2812FX::service() { | ||||
|         for (uint8_t c = 0; c < NUM_COLORS; c++) _colors_t[c] = gamma32(_colors_t[c]); | ||||
|  | ||||
|         // Effect blending (execute previous effect then new effect while in transition) | ||||
|         // WARNING: seg.currentMode(mode) (while in transition) will overwrite SEGENV variables!!! | ||||
|         // so they need to be saved first and then restored before running new mode. | ||||
|         // If two effects are being blended (each may have different segment settings), those | ||||
|         // settings need to be saved first and then restored before running new mode. | ||||
|         // The seetings need to be applied from transition buffer for previous effect. | ||||
|         // The blending will largely depend on the effect behaviour since actual output (LEDs) may be | ||||
|         // overwritten by later effect. To enable seamless blending for every effect, additional LED buffer | ||||
|         // would need to be allocated for each effect and then blended together for each pixel. | ||||
|         Segment::tmpsegd_t _tmpSegData; | ||||
|         seg.saveSegenv(&_tmpSegData); | ||||
|         uint8_t newMode    = seg.mode; | ||||
|         uint8_t newOpacity = seg.opacity; | ||||
|         uint8_t tmpMode    = seg.currentMode(seg.mode); | ||||
|         uint8_t tmpMode = seg.currentMode(seg.mode); | ||||
|         if (seg.mode != tmpMode) seg.restoreSegenv(nullptr); // restore transition data (including temporary opacity) | ||||
|         delay = (*_mode[tmpMode])(); // run old mode | ||||
|         if (newMode != tmpMode) { | ||||
|         if (seg.mode != tmpMode) { | ||||
|           if (tmpMode != FX_MODE_HALLOWEEN_EYES) seg.call++; | ||||
|           seg.restoreSegenv(&_tmpSegData); // restore mode state | ||||
|           seg.opacity = (uint32_t)newOpacity * seg.progress() / 0xFFFFU; | ||||
|           delay += (*_mode[seg.mode])();  // run new mode | ||||
|           delay /= 2;                     // average the delay | ||||
|           seg.opacity = newOpacity; | ||||
|           seg.restoreSegenv(&_tmpSegData);    // restore mode state | ||||
|           uint16_t d2 = (*_mode[seg.mode])(); // run new mode | ||||
|           delay = MIN(delay,d2);              // use shortest delay | ||||
|         } | ||||
|         if (seg.mode != FX_MODE_HALLOWEEN_EYES) seg.call++; | ||||
|         if (seg.transitional && delay > FRAMETIME) delay = FRAMETIME; // force faster updates during transition | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  */ | ||||
|  | ||||
| // version code in format yymmddb (b = daily build) | ||||
| #define VERSION 2308030 | ||||
| #define VERSION 2308050 | ||||
|  | ||||
| //uncomment this if you have a "my_config.h" file you'd like to use | ||||
| //#define WLED_USE_MY_CONFIG | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Blaz Kristan
					Blaz Kristan