blob: fb3ea319c642f660ab742ba722dbd3ca401adc2d [file] [log] [blame]
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +00001/*
2 * Copyright 2013 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 "SkDropShadowImageFilter.h"
9
10#include "SkBitmap.h"
11#include "SkBlurImageFilter.h"
12#include "SkCanvas.h"
13#include "SkColorMatrixFilter.h"
14#include "SkDevice.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000015#include "SkReadBuffer.h"
16#include "SkWriteBuffer.h"
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000017
commit-bot@chromium.orgf7efa502014-04-11 18:57:00 +000018SkDropShadowImageFilter::SkDropShadowImageFilter(SkScalar dx, SkScalar dy,
19 SkScalar sigmaX, SkScalar sigmaY, SkColor color,
sugoi234f0362014-10-23 13:59:52 -070020 ShadowMode shadowMode, SkImageFilter* input,
senorblanco24e06d52015-03-18 12:11:33 -070021 const CropRect* cropRect)
22 : INHERITED(1, &input, cropRect)
rmistry@google.comd6bab022013-12-02 13:50:38 +000023 , fDx(dx)
24 , fDy(dy)
25 , fSigmaX(sigmaX)
26 , fSigmaY(sigmaY)
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000027 , fColor(color)
sugoi234f0362014-10-23 13:59:52 -070028 , fShadowMode(shadowMode)
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000029{
30}
31
reed9fa60da2014-08-21 07:59:51 -070032SkFlattenable* SkDropShadowImageFilter::CreateProc(SkReadBuffer& buffer) {
33 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1);
34 SkScalar dx = buffer.readScalar();
35 SkScalar dy = buffer.readScalar();
36 SkScalar sigmaX = buffer.readScalar();
37 SkScalar sigmaY = buffer.readScalar();
38 SkColor color = buffer.readColor();
sugoi234f0362014-10-23 13:59:52 -070039 ShadowMode shadowMode = buffer.isVersionLT(SkReadBuffer::kDropShadowMode_Version) ?
40 kDrawShadowAndForeground_ShadowMode :
41 static_cast<ShadowMode>(buffer.readInt());
42 return Create(dx, dy, sigmaX, sigmaY, color, shadowMode, common.getInput(0),
senorblanco24e06d52015-03-18 12:11:33 -070043 &common.cropRect());
reed9fa60da2014-08-21 07:59:51 -070044}
45
46void SkDropShadowImageFilter::flatten(SkWriteBuffer& buffer) const {
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000047 this->INHERITED::flatten(buffer);
48 buffer.writeScalar(fDx);
49 buffer.writeScalar(fDy);
rmistry@google.comd6bab022013-12-02 13:50:38 +000050 buffer.writeScalar(fSigmaX);
51 buffer.writeScalar(fSigmaY);
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000052 buffer.writeColor(fColor);
sugoi234f0362014-10-23 13:59:52 -070053 buffer.writeInt(static_cast<int>(fShadowMode));
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000054}
55
commit-bot@chromium.orgf7efa502014-04-11 18:57:00 +000056bool SkDropShadowImageFilter::onFilterImage(Proxy* proxy, const SkBitmap& source,
57 const Context& ctx,
58 SkBitmap* result, SkIPoint* offset) const
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000059{
60 SkBitmap src = source;
senorblanco@chromium.org6776b822014-01-03 21:48:22 +000061 SkIPoint srcOffset = SkIPoint::Make(0, 0);
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +000062 if (getInput(0) && !getInput(0)->filterImage(proxy, source, ctx, &src, &srcOffset))
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000063 return false;
64
rmistry@google.comd6bab022013-12-02 13:50:38 +000065 SkIRect bounds;
senorblanco@chromium.org11825292014-03-14 17:44:41 +000066 if (!this->applyCropRect(ctx, src, srcOffset, &bounds)) {
rmistry@google.comd6bab022013-12-02 13:50:38 +000067 return false;
68 }
69
70 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
commit-bot@chromium.orgcd3b15c2013-12-04 17:06:49 +000071 if (NULL == device.get()) {
72 return false;
73 }
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000074 SkCanvas canvas(device.get());
75
senorblanco@chromium.orgba31f1d2014-05-07 20:56:08 +000076 SkVector sigma = SkVector::Make(fSigmaX, fSigmaY);
77 ctx.ctm().mapVectors(&sigma, 1);
senorblanco@chromium.org74bdde02014-01-28 17:57:26 +000078 sigma.fX = SkMaxScalar(0, sigma.fX);
79 sigma.fY = SkMaxScalar(0, sigma.fY);
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +000080 SkAutoTUnref<SkImageFilter> blurFilter(SkBlurImageFilter::Create(sigma.fX, sigma.fY));
commit-bot@chromium.orgf7efa502014-04-11 18:57:00 +000081 SkAutoTUnref<SkColorFilter> colorFilter(
82 SkColorFilter::CreateModeFilter(fColor, SkXfermode::kSrcIn_Mode));
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000083 SkPaint paint;
84 paint.setImageFilter(blurFilter.get());
85 paint.setColorFilter(colorFilter.get());
86 paint.setXfermodeMode(SkXfermode::kSrcOver_Mode);
senorblanco@chromium.orgba31f1d2014-05-07 20:56:08 +000087 SkVector offsetVec = SkVector::Make(fDx, fDy);
88 ctx.ctm().mapVectors(&offsetVec, 1);
senorblanco@chromium.org11825292014-03-14 17:44:41 +000089 canvas.translate(SkIntToScalar(srcOffset.fX - bounds.fLeft),
90 SkIntToScalar(srcOffset.fY - bounds.fTop));
senorblanco@chromium.org2bfe36b2014-01-20 19:58:28 +000091 canvas.drawBitmap(src, offsetVec.fX, offsetVec.fY, &paint);
sugoi234f0362014-10-23 13:59:52 -070092 if (fShadowMode == kDrawShadowAndForeground_ShadowMode) {
93 canvas.drawBitmap(src, 0, 0);
94 }
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000095 *result = device->accessBitmap(false);
senorblanco@chromium.org6776b822014-01-03 21:48:22 +000096 offset->fX = bounds.fLeft;
97 offset->fY = bounds.fTop;
junov@chromium.orgf44fcdc2013-08-07 20:00:55 +000098 return true;
99}
senorblanco@chromium.org336d1d72014-01-27 21:03:17 +0000100
101void SkDropShadowImageFilter::computeFastBounds(const SkRect& src, SkRect* dst) const {
102 if (getInput(0)) {
103 getInput(0)->computeFastBounds(src, dst);
104 } else {
105 *dst = src;
106 }
107
108 SkRect shadowBounds = *dst;
109 shadowBounds.offset(fDx, fDy);
110 shadowBounds.outset(SkScalarMul(fSigmaX, SkIntToScalar(3)),
111 SkScalarMul(fSigmaY, SkIntToScalar(3)));
sugoi234f0362014-10-23 13:59:52 -0700112 if (fShadowMode == kDrawShadowAndForeground_ShadowMode) {
113 dst->join(shadowBounds);
114 } else {
115 *dst = shadowBounds;
116 }
senorblanco@chromium.org336d1d72014-01-27 21:03:17 +0000117}
senorblanco@chromium.orgc4b12f12014-02-05 17:51:22 +0000118
119bool SkDropShadowImageFilter::onFilterBounds(const SkIRect& src, const SkMatrix& ctm,
120 SkIRect* dst) const {
121 SkIRect bounds = src;
senorblanco@chromium.orgba31f1d2014-05-07 20:56:08 +0000122 SkVector offsetVec = SkVector::Make(fDx, fDy);
123 ctm.mapVectors(&offsetVec, 1);
senorblanco@chromium.orgc4b12f12014-02-05 17:51:22 +0000124 bounds.offset(-SkScalarCeilToInt(offsetVec.x()),
125 -SkScalarCeilToInt(offsetVec.y()));
senorblanco@chromium.orgba31f1d2014-05-07 20:56:08 +0000126 SkVector sigma = SkVector::Make(fSigmaX, fSigmaY);
127 ctm.mapVectors(&sigma, 1);
senorblanco@chromium.orgc4b12f12014-02-05 17:51:22 +0000128 bounds.outset(SkScalarCeilToInt(SkScalarMul(sigma.x(), SkIntToScalar(3))),
129 SkScalarCeilToInt(SkScalarMul(sigma.y(), SkIntToScalar(3))));
sugoi234f0362014-10-23 13:59:52 -0700130 if (fShadowMode == kDrawShadowAndForeground_ShadowMode) {
131 bounds.join(src);
132 }
senorblanco1150a6d2014-08-25 12:46:58 -0700133 if (getInput(0) && !getInput(0)->filterBounds(bounds, ctm, &bounds)) {
134 return false;
135 }
senorblanco@chromium.orgc4b12f12014-02-05 17:51:22 +0000136 *dst = bounds;
137 return true;
138}
robertphillipsf3f5bad2014-12-19 13:49:15 -0800139
140#ifndef SK_IGNORE_TO_STRING
141void SkDropShadowImageFilter::toString(SkString* str) const {
142 str->appendf("SkDropShadowImageFilter: (");
143
144 str->appendf("dX: %f ", fDx);
145 str->appendf("dY: %f ", fDy);
146 str->appendf("sigmaX: %f ", fSigmaX);
147 str->appendf("sigmaY: %f ", fSigmaY);
148
149 str->append("Color: ");
150 str->appendHex(fColor);
151
152 static const char* gModeStrings[] = {
153 "kDrawShadowAndForeground", "kDrawShadowOnly"
154 };
155
bungeman99fe8222015-08-20 07:57:51 -0700156 static_assert(kShadowModeCount == SK_ARRAY_COUNT(gModeStrings), "enum_mismatch");
robertphillipsf3f5bad2014-12-19 13:49:15 -0800157
158 str->appendf(" mode: %s", gModeStrings[fShadowMode]);
159
160 str->append(")");
161}
162#endif