blob: 372dcf09f40dbfd80907207322070a424c5eaf0e [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001/*
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 */
reed@google.comdaaafa62014-04-29 15:20:16 +00007
reed@android.com8a1c16f2008-12-17 15:59:43 +00008#include "SkBlurDrawLooper.h"
robertphillips@google.com7ce661d2013-08-27 16:14:03 +00009#include "SkBlurMask.h" // just for SkBlurMask::ConvertRadiusToSigma
reed@android.com8a1c16f2008-12-17 15:59:43 +000010#include "SkBlurMaskFilter.h"
11#include "SkCanvas.h"
senorblanco@chromium.org4868e6b2011-02-18 19:03:01 +000012#include "SkColorFilter.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000013#include "SkReadBuffer.h"
14#include "SkWriteBuffer.h"
robertphillips@google.com4991b8f2013-01-28 20:21:59 +000015#include "SkMaskFilter.h"
16#include "SkPaint.h"
17#include "SkString.h"
18#include "SkStringUtils.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000019
commit-bot@chromium.org73cb1532014-04-15 15:48:36 +000020#ifdef SK_SUPPORT_LEGACY_BLURDRAWLOOPERCONSTRUCTORS
reed@android.com8a1c16f2008-12-17 15:59:43 +000021SkBlurDrawLooper::SkBlurDrawLooper(SkScalar radius, SkScalar dx, SkScalar dy,
robertphillips@google.com7ce661d2013-08-27 16:14:03 +000022 SkColor color, uint32_t flags) {
23 this->init(SkBlurMask::ConvertRadiusToSigma(radius), dx, dy, color, flags);
24}
commit-bot@chromium.org73cb1532014-04-15 15:48:36 +000025#endif
robertphillips@google.com7ce661d2013-08-27 16:14:03 +000026
skia.committer@gmail.com7bd141d2013-08-28 07:01:18 +000027SkBlurDrawLooper::SkBlurDrawLooper(SkColor color, SkScalar sigma,
robertphillips@google.com7ce661d2013-08-27 16:14:03 +000028 SkScalar dx, SkScalar dy, uint32_t flags) {
29 this->init(sigma, dx, dy, color, flags);
30}
31
reed@google.comdaaafa62014-04-29 15:20:16 +000032// only call from constructor
33void SkBlurDrawLooper::initEffects() {
34 SkASSERT(fBlurFlags <= kAll_BlurFlag);
35 if (fSigma > 0) {
36 uint32_t flags = fBlurFlags & kIgnoreTransform_BlurFlag ?
37 SkBlurMaskFilter::kIgnoreTransform_BlurFlag :
38 SkBlurMaskFilter::kNone_BlurFlag;
skia.committer@gmail.comeeaeafe2014-04-30 03:05:25 +000039
reed@google.comdaaafa62014-04-29 15:20:16 +000040 flags |= fBlurFlags & kHighQuality_BlurFlag ?
41 SkBlurMaskFilter::kHighQuality_BlurFlag :
42 SkBlurMaskFilter::kNone_BlurFlag;
skia.committer@gmail.comeeaeafe2014-04-30 03:05:25 +000043
reed@google.comdaaafa62014-04-29 15:20:16 +000044 fBlur = SkBlurMaskFilter::Create(kNormal_SkBlurStyle, fSigma, flags);
reed@google.com4e2b3d32011-04-07 14:18:59 +000045 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +000046 fBlur = NULL;
senorblanco@chromium.org4868e6b2011-02-18 19:03:01 +000047 }
skia.committer@gmail.comeeaeafe2014-04-30 03:05:25 +000048
reed@google.comdaaafa62014-04-29 15:20:16 +000049 if (fBlurFlags & kOverrideColor_BlurFlag) {
junov@google.coma6398912011-03-11 16:45:00 +000050 // Set alpha to 1 for the override since transparency will already
51 // be baked into the blurred mask.
reed@google.comdaaafa62014-04-29 15:20:16 +000052 SkColor opaqueColor = SkColorSetA(fBlurColor, 255);
senorblanco@chromium.org4868e6b2011-02-18 19:03:01 +000053 //The SrcIn xfer mode will multiply 'color' by the incoming alpha
reed@google.com4e2b3d32011-04-07 14:18:59 +000054 fColorFilter = SkColorFilter::CreateModeFilter(opaqueColor,
55 SkXfermode::kSrcIn_Mode);
56 } else {
senorblanco@chromium.org4868e6b2011-02-18 19:03:01 +000057 fColorFilter = NULL;
58 }
reed@android.com8a1c16f2008-12-17 15:59:43 +000059}
60
reed@google.comdaaafa62014-04-29 15:20:16 +000061void SkBlurDrawLooper::init(SkScalar sigma, SkScalar dx, SkScalar dy,
62 SkColor color, uint32_t flags) {
63 fSigma = sigma;
64 fDx = dx;
65 fDy = dy;
66 fBlurColor = color;
67 fBlurFlags = flags;
reed@google.com6bac9472011-06-21 19:24:00 +000068
reed@google.comdaaafa62014-04-29 15:20:16 +000069 this->initEffects();
70}
71
72SkBlurDrawLooper::SkBlurDrawLooper(SkReadBuffer& buffer) : INHERITED(buffer) {
73
74 fSigma = buffer.readScalar();
reed@android.com8a1c16f2008-12-17 15:59:43 +000075 fDx = buffer.readScalar();
76 fDy = buffer.readScalar();
djsollen@google.comc73dd5c2012-08-07 15:54:32 +000077 fBlurColor = buffer.readColor();
djsollen@google.comc73dd5c2012-08-07 15:54:32 +000078 fBlurFlags = buffer.readUInt() & kAll_BlurFlag;
reed@google.comdaaafa62014-04-29 15:20:16 +000079
80 this->initEffects();
81}
82
83void SkBlurDrawLooper::flatten(SkWriteBuffer& buffer) const {
84 this->INHERITED::flatten(buffer);
85 buffer.writeScalar(fSigma);
86 buffer.writeScalar(fDx);
87 buffer.writeScalar(fDy);
88 buffer.writeColor(fBlurColor);
89 buffer.write32(fBlurFlags);
reed@android.com8a1c16f2008-12-17 15:59:43 +000090}
91
reed@google.com4e2b3d32011-04-07 14:18:59 +000092SkBlurDrawLooper::~SkBlurDrawLooper() {
reed@google.com82065d62011-02-07 15:30:46 +000093 SkSafeUnref(fBlur);
senorblanco@chromium.org4868e6b2011-02-18 19:03:01 +000094 SkSafeUnref(fColorFilter);
reed@android.com8a1c16f2008-12-17 15:59:43 +000095}
96
reed@google.comdaaafa62014-04-29 15:20:16 +000097bool SkBlurDrawLooper::asABlurShadow(BlurShadowRec* rec) const {
98 if (fSigma <= 0 || (fBlurFlags & fBlurFlags & kIgnoreTransform_BlurFlag)) {
99 return false;
100 }
101
102 if (rec) {
103 rec->fSigma = fSigma;
104 rec->fColor = fBlurColor;
105 rec->fOffset.set(fDx, fDy);
106 rec->fStyle = kNormal_SkBlurStyle;
107 rec->fQuality = (fBlurFlags & kHighQuality_BlurFlag) ?
108 kHigh_SkBlurQuality : kLow_SkBlurQuality;
109 }
110 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000111}
112
reed@google.comdaaafa62014-04-29 15:20:16 +0000113////////////////////////////////////////////////////////////////////////////////////////
114
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000115SkDrawLooper::Context* SkBlurDrawLooper::createContext(SkCanvas*, void* storage) const {
116 return SkNEW_PLACEMENT_ARGS(storage, BlurDrawLooperContext, (this));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000117}
118
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000119SkBlurDrawLooper::BlurDrawLooperContext::BlurDrawLooperContext(
120 const SkBlurDrawLooper* looper)
121 : fLooper(looper), fState(SkBlurDrawLooper::kBeforeEdge) {}
122
123bool SkBlurDrawLooper::BlurDrawLooperContext::next(SkCanvas* canvas,
124 SkPaint* paint) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000125 switch (fState) {
reed@google.com4e2b3d32011-04-07 14:18:59 +0000126 case kBeforeEdge:
127 // we do nothing if a maskfilter is already installed
128 if (paint->getMaskFilter()) {
129 fState = kDone;
130 return false;
131 }
djsollen@google.com56c69772011-11-08 19:00:26 +0000132#ifdef SK_BUILD_FOR_ANDROID
djsollen@google.comf5dbe2f2011-04-15 13:41:26 +0000133 SkColor blurColor;
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000134 blurColor = fLooper->fBlurColor;
djsollen@google.comf5dbe2f2011-04-15 13:41:26 +0000135 if (SkColorGetA(blurColor) == 255) {
136 blurColor = SkColorSetA(blurColor, paint->getAlpha());
137 }
138 paint->setColor(blurColor);
139#else
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000140 paint->setColor(fLooper->fBlurColor);
djsollen@google.comf5dbe2f2011-04-15 13:41:26 +0000141#endif
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000142 paint->setMaskFilter(fLooper->fBlur);
143 paint->setColorFilter(fLooper->fColorFilter);
commit-bot@chromium.org74e608c2014-04-23 19:26:38 +0000144 canvas->save();
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000145 if (fLooper->fBlurFlags & kIgnoreTransform_BlurFlag) {
reed@google.com4e2b3d32011-04-07 14:18:59 +0000146 SkMatrix transform(canvas->getTotalMatrix());
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000147 transform.postTranslate(fLooper->fDx, fLooper->fDy);
reed@google.com4e2b3d32011-04-07 14:18:59 +0000148 canvas->setMatrix(transform);
149 } else {
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +0000150 canvas->translate(fLooper->fDx, fLooper->fDy);
reed@google.com4e2b3d32011-04-07 14:18:59 +0000151 }
152 fState = kAfterEdge;
153 return true;
154 case kAfterEdge:
155 canvas->restore();
156 fState = kDone;
157 return true;
158 default:
159 SkASSERT(kDone == fState);
160 return false;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000161 }
162}
robertphillips@google.com4991b8f2013-01-28 20:21:59 +0000163
commit-bot@chromium.org0f10f7b2014-03-13 18:02:17 +0000164#ifndef SK_IGNORE_TO_STRING
robertphillips@google.com4991b8f2013-01-28 20:21:59 +0000165void SkBlurDrawLooper::toString(SkString* str) const {
166 str->append("SkBlurDrawLooper: ");
167
168 str->append("dx: ");
169 str->appendScalar(fDx);
170
171 str->append(" dy: ");
172 str->appendScalar(fDy);
173
174 str->append(" color: ");
175 str->appendHex(fBlurColor);
176
177 str->append(" flags: (");
178 if (kNone_BlurFlag == fBlurFlags) {
179 str->append("None");
180 } else {
181 bool needsSeparator = false;
skia.committer@gmail.comcdcb2ce2013-01-29 07:05:52 +0000182 SkAddFlagToString(str, SkToBool(kIgnoreTransform_BlurFlag & fBlurFlags), "IgnoreTransform",
robertphillips@google.com4991b8f2013-01-28 20:21:59 +0000183 &needsSeparator);
184 SkAddFlagToString(str, SkToBool(kOverrideColor_BlurFlag & fBlurFlags), "OverrideColor",
185 &needsSeparator);
skia.committer@gmail.comcdcb2ce2013-01-29 07:05:52 +0000186 SkAddFlagToString(str, SkToBool(kHighQuality_BlurFlag & fBlurFlags), "HighQuality",
robertphillips@google.com4991b8f2013-01-28 20:21:59 +0000187 &needsSeparator);
188 }
189 str->append(")");
190
191 // TODO: add optional "fBlurFilter->toString(str);" when SkMaskFilter::toString is added
192 // alternatively we could cache the radius in SkBlurDrawLooper and just add it here
193}
194#endif