21 fl::u16 dstWidth = srcWidth / 2;
22 fl::u16 dstHeight = srcHeight / 2;
24 for (fl::u16
y = 0;
y < dstHeight; ++
y) {
25 for (fl::u16
x = 0;
x < dstWidth; ++
x) {
31 const CRGB &p00 = src[(srcY)*srcWidth + (srcX)];
32 const CRGB &p10 = src[(srcY)*srcWidth + (srcX + 1)];
33 const CRGB &p01 = src[(srcY + 1) * srcWidth + (srcX)];
34 const CRGB &p11 = src[(srcY + 1) * srcWidth + (srcX + 1)];
38 (p00.r + p10.r + p01.r + p11.r + 2) / 4;
39 fl::u16 g = (p00.g + p10.g + p01.g + p11.g + 2) / 4;
40 fl::u16 b = (p00.b + p10.b + p01.b + p11.b + 2) / 4;
50 fl::u16 dstWidth = dstXY.getWidth();
51 fl::u16 dstHeight = dstXY.getHeight();
53 FASTLED_ASSERT(srcXY.getWidth() == dstXY.getWidth() * 2,
54 "Source width must be double the destination width");
55 FASTLED_ASSERT(srcXY.getHeight() == dstXY.getHeight() * 2,
56 "Source height must be double the destination height");
58 for (fl::u16
y = 0;
y < dstHeight; ++
y) {
59 for (fl::u16
x = 0;
x < dstWidth; ++
x) {
65 const CRGB &p00 = src[srcXY.mapToIndex(srcX, srcY)];
66 const CRGB &p10 = src[srcXY.mapToIndex(srcX + 1, srcY)];
67 const CRGB &p01 = src[srcXY.mapToIndex(srcX, srcY + 1)];
68 const CRGB &p11 = src[srcXY.mapToIndex(srcX + 1, srcY + 1)];
72 (p00.r + p10.r + p01.r + p11.r + 2) / 4;
73 fl::u16 g = (p00.g + p10.g + p01.g + p11.g + 2) / 4;
74 fl::u16 b = (p00.b + p10.b + p01.b + p11.b + 2) / 4;
77 dst[dstXY.mapToIndex(
x,
y)] =
85 const fl::u16 srcWidth = srcXY.getWidth();
86 const fl::u16 srcHeight = srcXY.getHeight();
87 const fl::u16 dstWidth = dstXY.getWidth();
88 const fl::u16 dstHeight = dstXY.getHeight();
90 const fl::u32
FP_ONE = 256;
92 FASTLED_ASSERT(dstWidth <= srcWidth,
93 "Destination width must be <= source width");
94 FASTLED_ASSERT(dstHeight <= srcHeight,
95 "Destination height must be <= source height");
97 for (fl::u16 dy = 0; dy < dstHeight; ++dy) {
99 fl::u32 dstY0 = (dy * srcHeight *
FP_ONE) / dstHeight;
100 fl::u32 dstY1 = ((dy + 1) * srcHeight *
FP_ONE) / dstHeight;
102 for (fl::u16 dx = 0; dx < dstWidth; ++dx) {
103 fl::u32 dstX0 = (dx * srcWidth *
FP_ONE) / dstWidth;
104 fl::u32 dstX1 = ((dx + 1) * srcWidth *
FP_ONE) / dstWidth;
106 fl::u64 rSum = 0, gSum = 0, bSum = 0;
107 fl::u32 totalWeight = 0;
110 fl::u16 srcY_start = dstY0 /
FP_ONE;
113 fl::u16 srcX_start = dstX0 /
FP_ONE;
116 for (fl::u16 sy = srcY_start; sy < srcY_end; ++sy) {
118 fl::u32 sy0 = sy *
FP_ONE;
119 fl::u32 sy1 = (sy + 1) *
FP_ONE;
124 for (fl::u16 sx = srcX_start; sx < srcX_end; ++sx) {
125 fl::u32 sx0 = sx *
FP_ONE;
126 fl::u32 sx1 = (sx + 1) *
FP_ONE;
131 fl::u32 weight = (x_overlap * y_overlap + (
FP_ONE >> 1)) >>
134 const CRGB &p = src[srcXY.mapToIndex(sx, sy)];
135 rSum += p.r * weight;
136 gSum += p.g * weight;
137 bSum += p.b * weight;
138 totalWeight += weight;
144 totalWeight ? (rSum + (totalWeight >> 1)) / totalWeight : 0;
146 totalWeight ? (gSum + (totalWeight >> 1)) / totalWeight : 0;
148 totalWeight ? (bSum + (totalWeight >> 1)) / totalWeight : 0;
150 dst[dstXY.mapToIndex(dx, dy)] =
CRGB(r, g, b);
156 const XYMap &dstXY) {
157 fl::u16 srcWidth = srcXY.getWidth();
158 fl::u16 srcHeight = srcXY.getHeight();
159 fl::u16 dstWidth = dstXY.getWidth();
160 fl::u16 dstHeight = dstXY.getHeight();
162 FASTLED_ASSERT(dstWidth <= srcWidth,
163 "Destination width must be <= source width");
164 FASTLED_ASSERT(dstHeight <= srcHeight,
165 "Destination height must be <= source height");
166 const bool destination_is_half_of_source =
167 (dstWidth * 2 == srcWidth) && (dstHeight * 2 == srcHeight);
170 if (destination_is_half_of_source) {
173 if (both_rectangles) {