blob: 1446e92117ca7a78a6da8008564e6a0ab790ca67 [file] [log] [blame]
senorblanco@chromium.org60014ca2011-11-09 16:05:58 +00001/*
2 * Copyright 2011 The Android Open Source Project
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "SkBlurImageFilter.h"
senorblanco@chromium.orgae814c72011-12-20 20:02:19 +00009#include "SkColorPriv.h"
senorblanco@chromium.org60014ca2011-11-09 16:05:58 +000010
senorblanco@chromium.org54e01b22011-11-16 18:20:47 +000011SkBlurImageFilter::SkBlurImageFilter(SkFlattenableReadBuffer& buffer)
12 : INHERITED(buffer) {
13 fSigma.fWidth = buffer.readScalar();
14 fSigma.fHeight = buffer.readScalar();
15}
16
senorblanco@chromium.org60014ca2011-11-09 16:05:58 +000017SkBlurImageFilter::SkBlurImageFilter(SkScalar sigmaX, SkScalar sigmaY)
18 : fSigma(SkSize::Make(sigmaX, sigmaY)) {
19 SkASSERT(sigmaX >= 0 && sigmaY >= 0);
20}
21
22bool SkBlurImageFilter::asABlur(SkSize* sigma) const {
23 *sigma = fSigma;
24 return true;
25}
senorblanco@chromium.org54e01b22011-11-16 18:20:47 +000026
27void SkBlurImageFilter::flatten(SkFlattenableWriteBuffer& buffer) {
28 this->INHERITED::flatten(buffer);
29 buffer.writeScalar(fSigma.fWidth);
30 buffer.writeScalar(fSigma.fHeight);
31}
32
senorblanco@chromium.orgae814c72011-12-20 20:02:19 +000033static void boxBlurX(const SkBitmap& src, SkBitmap* dst, int kernelSize,
34 int leftOffset, int rightOffset)
35{
36 int width = src.width(), height = src.height();
37 int rightBorder = SkMin32(rightOffset + 1, width);
38 for (int y = 0; y < height; ++y) {
39 int sumA = 0, sumR = 0, sumG = 0, sumB = 0;
40 SkPMColor* p = src.getAddr32(0, y);
41 for (int i = 0; i < rightBorder; ++i) {
42 sumA += SkGetPackedA32(*p);
43 sumR += SkGetPackedR32(*p);
44 sumG += SkGetPackedG32(*p);
45 sumB += SkGetPackedB32(*p);
46 p++;
47 }
48
49 const SkColor* sptr = src.getAddr32(0, y);
50 SkColor* dptr = dst->getAddr32(0, y);
51 for (int x = 0; x < width; ++x) {
52 *dptr = SkPackARGB32(sumA / kernelSize,
53 sumR / kernelSize,
54 sumG / kernelSize,
55 sumB / kernelSize);
56 if (x >= leftOffset) {
57 SkColor l = *(sptr - leftOffset);
58 sumA -= SkGetPackedA32(l);
59 sumR -= SkGetPackedR32(l);
60 sumG -= SkGetPackedG32(l);
61 sumB -= SkGetPackedB32(l);
62 }
63 if (x + rightOffset + 1 < width) {
64 SkColor r = *(sptr + rightOffset + 1);
65 sumA += SkGetPackedA32(r);
66 sumR += SkGetPackedR32(r);
67 sumG += SkGetPackedG32(r);
68 sumB += SkGetPackedB32(r);
69 }
70 sptr++;
71 dptr++;
72 }
73 }
74}
75
76static void boxBlurY(const SkBitmap& src, SkBitmap* dst, int kernelSize,
77 int topOffset, int bottomOffset)
78{
79 int width = src.width(), height = src.height();
80 int bottomBorder = SkMin32(bottomOffset + 1, height);
81 int srcStride = src.rowBytesAsPixels();
82 int dstStride = dst->rowBytesAsPixels();
83 for (int x = 0; x < width; ++x) {
84 int sumA = 0, sumR = 0, sumG = 0, sumB = 0;
85 SkColor* p = src.getAddr32(x, 0);
86 for (int i = 0; i < bottomBorder; ++i) {
87 sumA += SkGetPackedA32(*p);
88 sumR += SkGetPackedR32(*p);
89 sumG += SkGetPackedG32(*p);
90 sumB += SkGetPackedB32(*p);
91 p += srcStride;
92 }
93
94 const SkColor* sptr = src.getAddr32(x, 0);
95 SkColor* dptr = dst->getAddr32(x, 0);
96 for (int y = 0; y < height; ++y) {
97 *dptr = SkPackARGB32(sumA / kernelSize,
98 sumR / kernelSize,
99 sumG / kernelSize,
100 sumB / kernelSize);
101 if (y >= topOffset) {
102 SkColor l = *(sptr - topOffset * srcStride);
103 sumA -= SkGetPackedA32(l);
104 sumR -= SkGetPackedR32(l);
105 sumG -= SkGetPackedG32(l);
106 sumB -= SkGetPackedB32(l);
107 }
108 if (y + bottomOffset + 1 < height) {
109 SkColor r = *(sptr + (bottomOffset + 1) * srcStride);
110 sumA += SkGetPackedA32(r);
111 sumR += SkGetPackedR32(r);
112 sumG += SkGetPackedG32(r);
113 sumB += SkGetPackedB32(r);
114 }
115 sptr += srcStride;
116 dptr += dstStride;
117 }
118 }
119}
120
121static void getBox3Params(SkScalar s, int *kernelSize, int* kernelSize3, int *lowOffset, int *highOffset)
122{
schenney@chromium.org73f3ded2011-12-20 22:31:40 +0000123 float pi = SkScalarToFloat(SK_ScalarPI);
124 int d = static_cast<int>(floorf(SkScalarToFloat(s) * 3.0f * sqrtf(2.0f * pi) / 4.0f + 0.5f));
senorblanco@chromium.orgae814c72011-12-20 20:02:19 +0000125 *kernelSize = d;
126 if (d % 2 == 1) {
127 *lowOffset = *highOffset = (d - 1) / 2;
128 *kernelSize3 = d;
129 } else {
130 *highOffset = d / 2;
131 *lowOffset = *highOffset - 1;
132 *kernelSize3 = d + 1;
133 }
134}
135
reed@google.com76dd2772012-01-05 21:15:07 +0000136bool SkBlurImageFilter::onFilterImage(Proxy*,
137 const SkBitmap& src, const SkMatrix&,
senorblanco@chromium.orgae814c72011-12-20 20:02:19 +0000138 SkBitmap* dst, SkIPoint*) {
139 if (src.config() != SkBitmap::kARGB_8888_Config) {
140 return false;
141 }
142
reed@google.com76dd2772012-01-05 21:15:07 +0000143 SkAutoLockPixels alp(src);
144 if (!src.getPixels()) {
145 return false;
146 }
147
senorblanco@chromium.orgae814c72011-12-20 20:02:19 +0000148 dst->setConfig(src.config(), src.width(), src.height());
149 dst->allocPixels();
150 int kernelSizeX, kernelSizeX3, lowOffsetX, highOffsetX;
151 int kernelSizeY, kernelSizeY3, lowOffsetY, highOffsetY;
152 getBox3Params(fSigma.width(), &kernelSizeX, &kernelSizeX3, &lowOffsetX, &highOffsetX);
153 getBox3Params(fSigma.height(), &kernelSizeY, &kernelSizeY3, &lowOffsetY, &highOffsetY);
154
155 if (kernelSizeX < 0 || kernelSizeY < 0) {
156 return false;
157 }
158
159 if (kernelSizeX == 0 && kernelSizeY == 0) {
160 src.copyTo(dst, dst->config());
161 return true;
162 }
163
164 SkBitmap temp;
165 temp.setConfig(dst->config(), dst->width(), dst->height());
166 if (!temp.allocPixels()) {
167 return false;
168 }
169
170 if (kernelSizeX > 0 && kernelSizeY > 0) {
171 boxBlurX(src, &temp, kernelSizeX, lowOffsetX, highOffsetX);
172 boxBlurY(temp, dst, kernelSizeY, lowOffsetY, highOffsetY);
173 boxBlurX(*dst, &temp, kernelSizeX, highOffsetX, lowOffsetX);
174 boxBlurY(temp, dst, kernelSizeY, highOffsetY, lowOffsetY);
175 boxBlurX(*dst, &temp, kernelSizeX3, highOffsetX, highOffsetX);
176 boxBlurY(temp, dst, kernelSizeY3, highOffsetY, highOffsetY);
177 } else if (kernelSizeX > 0) {
178 boxBlurX(src, dst, kernelSizeX, lowOffsetX, highOffsetX);
179 boxBlurX(*dst, &temp, kernelSizeX, highOffsetX, lowOffsetX);
180 boxBlurX(temp, dst, kernelSizeX3, highOffsetX, highOffsetX);
181 } else if (kernelSizeY > 0) {
182 boxBlurY(src, dst, kernelSizeY, lowOffsetY, highOffsetY);
183 boxBlurY(*dst, &temp, kernelSizeY, highOffsetY, lowOffsetY);
184 boxBlurY(temp, dst, kernelSizeY3, highOffsetY, highOffsetY);
185 }
186 return true;
187}
188
caryclark@google.comd26147a2011-12-15 14:16:43 +0000189SK_DEFINE_FLATTENABLE_REGISTRAR(SkBlurImageFilter)