SkBlurDrawLooper.cpp revision eeaeafebdeec2c546134b741ab8b3c6b7c5190ab
1ec3ed6a5ebf6f2c406d7bcf94b6bc34fcaeb976eepoger@google.com/*
2ec3ed6a5ebf6f2c406d7bcf94b6bc34fcaeb976eepoger@google.com * Copyright 2011 Google Inc.
3ec3ed6a5ebf6f2c406d7bcf94b6bc34fcaeb976eepoger@google.com *
4ec3ed6a5ebf6f2c406d7bcf94b6bc34fcaeb976eepoger@google.com * Use of this source code is governed by a BSD-style license that can be
5ec3ed6a5ebf6f2c406d7bcf94b6bc34fcaeb976eepoger@google.com * found in the LICENSE file.
6ec3ed6a5ebf6f2c406d7bcf94b6bc34fcaeb976eepoger@google.com */
7daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com
88a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com#include "SkBlurDrawLooper.h"
97ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com#include "SkBlurMask.h"     // just for SkBlurMask::ConvertRadiusToSigma
108a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com#include "SkBlurMaskFilter.h"
118a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com#include "SkCanvas.h"
124991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com#include "SkColorFilter.h"
138b0e8ac5f582de80356019406e2975079bf0829dcommit-bot@chromium.org#include "SkReadBuffer.h"
148b0e8ac5f582de80356019406e2975079bf0829dcommit-bot@chromium.org#include "SkWriteBuffer.h"
158a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com#include "SkMaskFilter.h"
164991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com#include "SkPaint.h"
174991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com#include "SkString.h"
184991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com#include "SkStringUtils.h"
198a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com
2073cb15351f33459e0c861a96135c634dec77ef9dcommit-bot@chromium.org#ifdef SK_SUPPORT_LEGACY_BLURDRAWLOOPERCONSTRUCTORS
218a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.comSkBlurDrawLooper::SkBlurDrawLooper(SkScalar radius, SkScalar dx, SkScalar dy,
227ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com                                   SkColor color, uint32_t flags) {
237ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com    this->init(SkBlurMask::ConvertRadiusToSigma(radius), dx, dy, color, flags);
247ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com}
2573cb15351f33459e0c861a96135c634dec77ef9dcommit-bot@chromium.org#endif
267ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com
277bd141dce43ea3405bc60c9c84e6f910b851b079skia.committer@gmail.comSkBlurDrawLooper::SkBlurDrawLooper(SkColor color, SkScalar sigma,
287ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com                                   SkScalar dx, SkScalar dy, uint32_t flags) {
297ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com    this->init(sigma, dx, dy, color, flags);
307ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com}
317ce661d19c5cf4484305a1b20c44bd111f129847robertphillips@google.com
32daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com// only call from constructor
33daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.comvoid SkBlurDrawLooper::initEffects() {
34daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    SkASSERT(fBlurFlags <= kAll_BlurFlag);
35daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    if (fSigma > 0) {
36daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        uint32_t flags = fBlurFlags & kIgnoreTransform_BlurFlag ?
37daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com                            SkBlurMaskFilter::kIgnoreTransform_BlurFlag :
38daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com                            SkBlurMaskFilter::kNone_BlurFlag;
39eeaeafebdeec2c546134b741ab8b3c6b7c5190abskia.committer@gmail.com
40daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        flags |= fBlurFlags & kHighQuality_BlurFlag ?
41daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com                    SkBlurMaskFilter::kHighQuality_BlurFlag :
42daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com                    SkBlurMaskFilter::kNone_BlurFlag;
43eeaeafebdeec2c546134b741ab8b3c6b7c5190abskia.committer@gmail.com
44daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        fBlur = SkBlurMaskFilter::Create(kNormal_SkBlurStyle, fSigma, flags);
454e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com    } else {
468a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com        fBlur = NULL;
474868e6b221a4a98e40f977851af5fcf09631ea15senorblanco@chromium.org    }
48eeaeafebdeec2c546134b741ab8b3c6b7c5190abskia.committer@gmail.com
49daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    if (fBlurFlags & kOverrideColor_BlurFlag) {
50a6398911174d5445456ecb2f5f4f0565db2f100bjunov@google.com        // Set alpha to 1 for the override since transparency will already
51a6398911174d5445456ecb2f5f4f0565db2f100bjunov@google.com        // be baked into the blurred mask.
52daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        SkColor opaqueColor = SkColorSetA(fBlurColor, 255);
534868e6b221a4a98e40f977851af5fcf09631ea15senorblanco@chromium.org        //The SrcIn xfer mode will multiply 'color' by the incoming alpha
544e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com        fColorFilter = SkColorFilter::CreateModeFilter(opaqueColor,
554e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com                                                       SkXfermode::kSrcIn_Mode);
564e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com    } else {
574868e6b221a4a98e40f977851af5fcf09631ea15senorblanco@chromium.org        fColorFilter = NULL;
584868e6b221a4a98e40f977851af5fcf09631ea15senorblanco@chromium.org    }
598a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com}
608a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com
61daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.comvoid SkBlurDrawLooper::init(SkScalar sigma, SkScalar dx, SkScalar dy,
62daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com                            SkColor color, uint32_t flags) {
63daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    fSigma = sigma;
64daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    fDx = dx;
65daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    fDy = dy;
66daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    fBlurColor = color;
67daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    fBlurFlags = flags;
68daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com
69daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    this->initEffects();
70daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com}
71daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com
72daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.comSkBlurDrawLooper::SkBlurDrawLooper(SkReadBuffer& buffer) : INHERITED(buffer) {
736bac947cd5bc460dd9166ada6310d678fd2e39f8reed@google.com
74daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    fSigma = buffer.readScalar();
758a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com    fDx = buffer.readScalar();
768a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com    fDy = buffer.readScalar();
77c73dd5c6880739f26216f198c757028fd28df1a4djsollen@google.com    fBlurColor = buffer.readColor();
78c73dd5c6880739f26216f198c757028fd28df1a4djsollen@google.com    fBlurFlags = buffer.readUInt() & kAll_BlurFlag;
798a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com
80daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    this->initEffects();
818a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com}
828a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com
838b0e8ac5f582de80356019406e2975079bf0829dcommit-bot@chromium.orgvoid SkBlurDrawLooper::flatten(SkWriteBuffer& buffer) const {
8454924243c1b65b3ee6d8fa064b50a9b1bb2a19a5djsollen@google.com    this->INHERITED::flatten(buffer);
85daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    buffer.writeScalar(fSigma);
868a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com    buffer.writeScalar(fDx);
878a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com    buffer.writeScalar(fDy);
88c73dd5c6880739f26216f198c757028fd28df1a4djsollen@google.com    buffer.writeColor(fBlurColor);
89daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    buffer.write32(fBlurFlags);
90daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com}
91daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com
92daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.comSkBlurDrawLooper::~SkBlurDrawLooper() {
93daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    SkSafeUnref(fBlur);
94daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    SkSafeUnref(fColorFilter);
958a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com}
968a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com
97daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.combool SkBlurDrawLooper::asABlurShadow(BlurShadowRec* rec) const {
98daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    if (fSigma <= 0 || (fBlurFlags & fBlurFlags & kIgnoreTransform_BlurFlag)) {
99daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        return false;
100daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    }
101daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com
102daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    if (rec) {
103daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        rec->fSigma = fSigma;
104daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        rec->fColor = fBlurColor;
105daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        rec->fOffset.set(fDx, fDy);
106daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        rec->fStyle = kNormal_SkBlurStyle;
107daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com        rec->fQuality = (fBlurFlags & kHighQuality_BlurFlag) ?
108daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com                        kHigh_SkBlurQuality : kLow_SkBlurQuality;
109daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    }
110daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com    return true;
111daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com}
112daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com
113daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com////////////////////////////////////////////////////////////////////////////////////////
114daaafa6e81860e3dc52660ba019c336f0a43f1e7reed@google.com
11579fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.orgSkDrawLooper::Context* SkBlurDrawLooper::createContext(SkCanvas*, void* storage) const {
11679fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org    return SkNEW_PLACEMENT_ARGS(storage, BlurDrawLooperContext, (this));
1178a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com}
1188a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com
11979fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.orgSkBlurDrawLooper::BlurDrawLooperContext::BlurDrawLooperContext(
12079fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org        const SkBlurDrawLooper* looper)
12179fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org    : fLooper(looper), fState(SkBlurDrawLooper::kBeforeEdge) {}
12279fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org
12379fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.orgbool SkBlurDrawLooper::BlurDrawLooperContext::next(SkCanvas* canvas,
12479fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org                                                   SkPaint* paint) {
1258a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com    switch (fState) {
1264e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com        case kBeforeEdge:
1274e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            // we do nothing if a maskfilter is already installed
1284e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            if (paint->getMaskFilter()) {
1294e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com                fState = kDone;
1304e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com                return false;
1314e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            }
13256c69773aea56c6c6bd47bc7e7970dd081205184djsollen@google.com#ifdef SK_BUILD_FOR_ANDROID
133f5dbe2f00f853c6a1719924bdd0c33335a53423adjsollen@google.com            SkColor blurColor;
13479fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org            blurColor = fLooper->fBlurColor;
135f5dbe2f00f853c6a1719924bdd0c33335a53423adjsollen@google.com            if (SkColorGetA(blurColor) == 255) {
136f5dbe2f00f853c6a1719924bdd0c33335a53423adjsollen@google.com                blurColor = SkColorSetA(blurColor, paint->getAlpha());
137f5dbe2f00f853c6a1719924bdd0c33335a53423adjsollen@google.com            }
138f5dbe2f00f853c6a1719924bdd0c33335a53423adjsollen@google.com            paint->setColor(blurColor);
139f5dbe2f00f853c6a1719924bdd0c33335a53423adjsollen@google.com#else
14079fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org            paint->setColor(fLooper->fBlurColor);
141f5dbe2f00f853c6a1719924bdd0c33335a53423adjsollen@google.com#endif
14279fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org            paint->setMaskFilter(fLooper->fBlur);
14379fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org            paint->setColorFilter(fLooper->fColorFilter);
14474e608cdc1e13593b54d54f11e65c1c309d19370commit-bot@chromium.org            canvas->save();
14579fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org            if (fLooper->fBlurFlags & kIgnoreTransform_BlurFlag) {
1464e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com                SkMatrix transform(canvas->getTotalMatrix());
14779fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org                transform.postTranslate(fLooper->fDx, fLooper->fDy);
1484e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com                canvas->setMatrix(transform);
1494e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            } else {
15079fbb40bca9d815ef79b896b31ba6ee736817e0fcommit-bot@chromium.org                canvas->translate(fLooper->fDx, fLooper->fDy);
1514e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            }
1524e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            fState = kAfterEdge;
1534e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            return true;
1544e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com        case kAfterEdge:
1554e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            canvas->restore();
1564e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            fState = kDone;
1574e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            return true;
1584e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com        default:
1594e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            SkASSERT(kDone == fState);
1604e2b3d3fb1288c6dc0f3ea1c0aa4a0d7c603bd7breed@google.com            return false;
1618a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com    }
1628a1c16ff38322f0210116fa7293eb8817c7e477ereed@android.com}
1634991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com
1640f10f7bf1fb43ca6346dc220a076773b1f19a367commit-bot@chromium.org#ifndef SK_IGNORE_TO_STRING
1654991b8f23482afc1494fd17647421ce68de53331robertphillips@google.comvoid SkBlurDrawLooper::toString(SkString* str) const {
1664991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->append("SkBlurDrawLooper: ");
1674991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com
1684991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->append("dx: ");
1694991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->appendScalar(fDx);
1704991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com
1714991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->append(" dy: ");
1724991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->appendScalar(fDy);
1734991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com
1744991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->append(" color: ");
1754991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->appendHex(fBlurColor);
1764991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com
1774991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->append(" flags: (");
1784991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    if (kNone_BlurFlag == fBlurFlags) {
1794991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com        str->append("None");
1804991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    } else {
1814991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com        bool needsSeparator = false;
182cdcb2ce2744c7e5c47453328dbf292edee79ab37skia.committer@gmail.com        SkAddFlagToString(str, SkToBool(kIgnoreTransform_BlurFlag & fBlurFlags), "IgnoreTransform",
1834991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com                          &needsSeparator);
1844991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com        SkAddFlagToString(str, SkToBool(kOverrideColor_BlurFlag & fBlurFlags), "OverrideColor",
1854991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com                          &needsSeparator);
186cdcb2ce2744c7e5c47453328dbf292edee79ab37skia.committer@gmail.com        SkAddFlagToString(str, SkToBool(kHighQuality_BlurFlag & fBlurFlags), "HighQuality",
1874991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com                          &needsSeparator);
1884991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    }
1894991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    str->append(")");
1904991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com
1914991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    // TODO: add optional "fBlurFilter->toString(str);" when SkMaskFilter::toString is added
1924991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com    // alternatively we could cache the radius in SkBlurDrawLooper and just add it here
1934991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com}
1944991b8f23482afc1494fd17647421ce68de53331robertphillips@google.com#endif
195