1 /*
2  * Copyright 2011 Google Inc.
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 "SkBlurDrawLooper.h"
9 #include "SkBlurMask.h"     // just for SkBlurMask::ConvertRadiusToSigma
10 #include "SkBlurMaskFilter.h"
11 #include "SkCanvas.h"
12 #include "SkColorFilter.h"
13 #include "SkReadBuffer.h"
14 #include "SkWriteBuffer.h"
15 #include "SkMaskFilter.h"
16 #include "SkPaint.h"
17 #include "SkString.h"
18 #include "SkStringUtils.h"
19 
SkBlurDrawLooper(SkColor color,SkScalar sigma,SkScalar dx,SkScalar dy,uint32_t flags)20 SkBlurDrawLooper::SkBlurDrawLooper(SkColor color, SkScalar sigma,
21                                    SkScalar dx, SkScalar dy, uint32_t flags) {
22     this->init(sigma, dx, dy, color, flags);
23 }
24 
25 // only call from constructor
initEffects()26 void SkBlurDrawLooper::initEffects() {
27     SkASSERT(fBlurFlags <= kAll_BlurFlag);
28     if (fSigma > 0) {
29         uint32_t flags = fBlurFlags & kIgnoreTransform_BlurFlag ?
30                             SkBlurMaskFilter::kIgnoreTransform_BlurFlag :
31                             SkBlurMaskFilter::kNone_BlurFlag;
32 
33         flags |= fBlurFlags & kHighQuality_BlurFlag ?
34                     SkBlurMaskFilter::kHighQuality_BlurFlag :
35                     SkBlurMaskFilter::kNone_BlurFlag;
36 
37         fBlur = SkBlurMaskFilter::Create(kNormal_SkBlurStyle, fSigma, flags);
38     } else {
39         fBlur = nullptr;
40     }
41 
42     if (fBlurFlags & kOverrideColor_BlurFlag) {
43         // Set alpha to 1 for the override since transparency will already
44         // be baked into the blurred mask.
45         SkColor opaqueColor = SkColorSetA(fBlurColor, 255);
46         //The SrcIn xfer mode will multiply 'color' by the incoming alpha
47         fColorFilter = SkColorFilter::CreateModeFilter(opaqueColor,
48                                                        SkXfermode::kSrcIn_Mode);
49     } else {
50         fColorFilter = nullptr;
51     }
52 }
53 
init(SkScalar sigma,SkScalar dx,SkScalar dy,SkColor color,uint32_t flags)54 void SkBlurDrawLooper::init(SkScalar sigma, SkScalar dx, SkScalar dy,
55                             SkColor color, uint32_t flags) {
56     fSigma = sigma;
57     fDx = dx;
58     fDy = dy;
59     fBlurColor = color;
60     fBlurFlags = flags;
61 
62     this->initEffects();
63 }
64 
CreateProc(SkReadBuffer & buffer)65 SkFlattenable* SkBlurDrawLooper::CreateProc(SkReadBuffer& buffer) {
66     const SkColor color = buffer.readColor();
67     const SkScalar sigma = buffer.readScalar();
68     const SkScalar dx = buffer.readScalar();
69     const SkScalar dy = buffer.readScalar();
70     const uint32_t flags = buffer.read32();
71     return Create(color, sigma, dx, dy, flags);
72 }
73 
flatten(SkWriteBuffer & buffer) const74 void SkBlurDrawLooper::flatten(SkWriteBuffer& buffer) const {
75     buffer.writeColor(fBlurColor);
76     buffer.writeScalar(fSigma);
77     buffer.writeScalar(fDx);
78     buffer.writeScalar(fDy);
79     buffer.write32(fBlurFlags);
80 }
81 
~SkBlurDrawLooper()82 SkBlurDrawLooper::~SkBlurDrawLooper() {
83     SkSafeUnref(fBlur);
84     SkSafeUnref(fColorFilter);
85 }
86 
asABlurShadow(BlurShadowRec * rec) const87 bool SkBlurDrawLooper::asABlurShadow(BlurShadowRec* rec) const {
88     if (fSigma <= 0 || (fBlurFlags & fBlurFlags & kIgnoreTransform_BlurFlag)) {
89         return false;
90     }
91 
92     if (rec) {
93         rec->fSigma = fSigma;
94         rec->fColor = fBlurColor;
95         rec->fOffset.set(fDx, fDy);
96         rec->fStyle = kNormal_SkBlurStyle;
97         rec->fQuality = (fBlurFlags & kHighQuality_BlurFlag) ?
98                         kHigh_SkBlurQuality : kLow_SkBlurQuality;
99     }
100     return true;
101 }
102 
103 ////////////////////////////////////////////////////////////////////////////////////////
104 
createContext(SkCanvas *,void * storage) const105 SkDrawLooper::Context* SkBlurDrawLooper::createContext(SkCanvas*, void* storage) const {
106     return new (storage) BlurDrawLooperContext(this);
107 }
108 
BlurDrawLooperContext(const SkBlurDrawLooper * looper)109 SkBlurDrawLooper::BlurDrawLooperContext::BlurDrawLooperContext(
110         const SkBlurDrawLooper* looper)
111     : fLooper(looper), fState(SkBlurDrawLooper::kBeforeEdge) {}
112 
next(SkCanvas * canvas,SkPaint * paint)113 bool SkBlurDrawLooper::BlurDrawLooperContext::next(SkCanvas* canvas,
114                                                    SkPaint* paint) {
115     switch (fState) {
116         case kBeforeEdge:
117             // we do nothing if a maskfilter is already installed
118             if (paint->getMaskFilter()) {
119                 fState = kDone;
120                 return false;
121             }
122 #ifdef SK_BUILD_FOR_ANDROID
123             SkColor blurColor;
124             blurColor = fLooper->fBlurColor;
125             if (SkColorGetA(blurColor) == 255) {
126                 blurColor = SkColorSetA(blurColor, paint->getAlpha());
127             }
128             paint->setColor(blurColor);
129 #else
130             paint->setColor(fLooper->fBlurColor);
131 #endif
132             paint->setMaskFilter(fLooper->fBlur);
133             paint->setColorFilter(fLooper->fColorFilter);
134             canvas->save();
135             if (fLooper->fBlurFlags & kIgnoreTransform_BlurFlag) {
136                 SkMatrix transform(canvas->getTotalMatrix());
137                 transform.postTranslate(fLooper->fDx, fLooper->fDy);
138                 canvas->setMatrix(transform);
139             } else {
140                 canvas->translate(fLooper->fDx, fLooper->fDy);
141             }
142             fState = kAfterEdge;
143             return true;
144         case kAfterEdge:
145             canvas->restore();
146             fState = kDone;
147             return true;
148         default:
149             SkASSERT(kDone == fState);
150             return false;
151     }
152 }
153 
154 #ifndef SK_IGNORE_TO_STRING
toString(SkString * str) const155 void SkBlurDrawLooper::toString(SkString* str) const {
156     str->append("SkBlurDrawLooper: ");
157 
158     str->append("dx: ");
159     str->appendScalar(fDx);
160 
161     str->append(" dy: ");
162     str->appendScalar(fDy);
163 
164     str->append(" color: ");
165     str->appendHex(fBlurColor);
166 
167     str->append(" flags: (");
168     if (kNone_BlurFlag == fBlurFlags) {
169         str->append("None");
170     } else {
171         bool needsSeparator = false;
172         SkAddFlagToString(str, SkToBool(kIgnoreTransform_BlurFlag & fBlurFlags), "IgnoreTransform",
173                           &needsSeparator);
174         SkAddFlagToString(str, SkToBool(kOverrideColor_BlurFlag & fBlurFlags), "OverrideColor",
175                           &needsSeparator);
176         SkAddFlagToString(str, SkToBool(kHighQuality_BlurFlag & fBlurFlags), "HighQuality",
177                           &needsSeparator);
178     }
179     str->append(")");
180 
181     // TODO: add optional "fBlurFilter->toString(str);" when SkMaskFilter::toString is added
182     // alternatively we could cache the radius in SkBlurDrawLooper and just add it here
183 }
184 #endif
185