Skip to content

Instantly share code, notes, and snippets.

@morningdew76
Last active March 17, 2023 20:52
Show Gist options
  • Star 3 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save morningdew76/8326eba7353d21ccdfc5a0c4b5cfecd6 to your computer and use it in GitHub Desktop.
Save morningdew76/8326eba7353d21ccdfc5a0c4b5cfecd6 to your computer and use it in GitHub Desktop.
ColorFromPalette12, a 12-bit ColorFromPalette function
CRGB ColorFromPalette12(CRGBPalette16 pal, uint16_t index, uint8_t brightness, TBlendType blendType) {
//index is a 12-bit number, range 0-4095
//uint8_t hi4 = lsrX4(index);
//uint8_t lo4 = index & 0x0F;
uint8_t hi4 = (index & 0x0F00) >> 8; //***shift 8 bits to the right instead of 4
uint8_t lo8 = index & 0xFF; //***keep right 8 bytes instead of right 4 bytes
// const CRGB* entry = &(pal[0]) + hi4;
// since hi4 is always 0..15, hi4 * sizeof(CRGB) can be a single-byte value,
// instead of the two byte 'int' that avr-gcc defaults to.
// So, we multiply hi4 X sizeof(CRGB), giving hi4XsizeofCRGB;
uint8_t hi4XsizeofCRGB = hi4 * sizeof(CRGB);
// We then add that to a base array pointer.
const CRGB* entry = (CRGB*)( (uint8_t*)(&(pal[0])) + hi4XsizeofCRGB);
//uint8_t blend = lo4 && (blendType != NOBLEND);
uint8_t blend = lo8 && (blendType != NOBLEND); //***8-bit instead of 4-bit
uint8_t red1 = entry->red;
uint8_t green1 = entry->green;
uint8_t blue1 = entry->blue;
if( blend ) {
if( hi4 == 15 ) {
entry = &(pal[0]);
} else {
entry++;
}
//uint8_t f2 = lo4 << 4;
uint8_t f2 = lo8; //***no shift required, 8-bit precision here
uint8_t f1 = 255 - f2;
// rgb1.nscale8(f1);
uint8_t red2 = entry->red;
red1 = scale8_LEAVING_R1_DIRTY( red1, f1);
red2 = scale8_LEAVING_R1_DIRTY( red2, f2);
red1 += red2;
uint8_t green2 = entry->green;
green1 = scale8_LEAVING_R1_DIRTY( green1, f1);
green2 = scale8_LEAVING_R1_DIRTY( green2, f2);
green1 += green2;
uint8_t blue2 = entry->blue;
blue1 = scale8_LEAVING_R1_DIRTY( blue1, f1);
blue2 = scale8_LEAVING_R1_DIRTY( blue2, f2);
blue1 += blue2;
cleanup_R1();
}
if( brightness != 255) {
if( brightness ) {
brightness++; // adjust for rounding
// Now, since brightness is nonzero, we don't need the full scale8_video logic;
// we can just to scale8 and then add one (unless scale8 fixed) to all nonzero inputs.
if( red1 ) {
red1 = scale8_LEAVING_R1_DIRTY( red1, brightness);
#if !(FASTLED_SCALE8_FIXED==1)
red1++;
#endif
}
if( green1 ) {
green1 = scale8_LEAVING_R1_DIRTY( green1, brightness);
#if !(FASTLED_SCALE8_FIXED==1)
green1++;
#endif
}
if( blue1 ) {
blue1 = scale8_LEAVING_R1_DIRTY( blue1, brightness);
#if !(FASTLED_SCALE8_FIXED==1)
blue1++;
#endif
}
cleanup_R1();
} else {
red1 = 0;
green1 = 0;
blue1 = 0;
}
}
return CRGB( red1, green1, blue1);
}
/*
Usage:
12-bit function-
leds[XY(x, y)] = ColorFromPalette12(myPal, noise[0][x][y], 255, LINEARBLEND);
where noise[0][x][y] is a uint16_t, constrained to 0-4095
Normal 8-bit function call-
leds[XY(x, y)] = ColorFromPalette(myPal, noise[0][x][y], 255, LINEARBLEND);
where noise[0][x][y] is a uint8_it constrained to 0-255
*/
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment