blob: 14cf0c3a422c8bc87ce62947c50e15485299da56 [file] [log] [blame]
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +00001/*
2 * Copyright 2013 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 "SkXfermodeImageFilter.h"
Brian Salomon89cb8212017-01-09 10:48:23 -05009#include "SkArithmeticImageFilter.h"
reed374772b2016-10-05 17:33:02 -070010#include "SkArithmeticModePriv.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000011#include "SkCanvas.h"
12#include "SkColorPriv.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000013#include "SkReadBuffer.h"
robertphillipsf230c6d2016-04-15 12:47:42 -070014#include "SkSpecialImage.h"
15#include "SkSpecialSurface.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000016#include "SkWriteBuffer.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000017#if SK_SUPPORT_GPU
Brian Salomonc65aec92017-03-09 09:03:58 -050018#include "GrClip.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000019#include "GrContext.h"
Brian Osman11052242016-10-27 14:47:55 -040020#include "GrRenderTargetContext.h"
Robert Phillips7f6cd902016-11-10 17:03:43 -050021#include "GrTextureProxy.h"
22
robertphillips4f037942016-02-09 05:09:27 -080023#include "effects/GrConstColorProcessor.h"
senorblancod762ca22015-04-07 12:16:55 -070024#include "effects/GrTextureDomain.h"
bsalomonae4738f2015-09-15 15:33:27 -070025#include "effects/GrSimpleTextureEffect.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000026#include "SkGr.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000027#endif
Mike Reedebfce6d2016-12-12 10:02:12 -050028#include "SkClipOpPriv.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000029
Mike Reed403d1542016-09-26 18:51:24 -040030class SkXfermodeImageFilter_Base : public SkImageFilter {
Mike Reed667a0072016-09-23 11:43:54 -040031public:
reed374772b2016-10-05 17:33:02 -070032 SkXfermodeImageFilter_Base(SkBlendMode mode, sk_sp<SkImageFilter> inputs[2],
Mike Reed667a0072016-09-23 11:43:54 -040033 const CropRect* cropRect);
34
35 SK_TO_STRING_OVERRIDE()
36 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkXfermodeImageFilter_Base)
37
38protected:
39 sk_sp<SkSpecialImage> onFilterImage(SkSpecialImage* source, const Context&,
40 SkIPoint* offset) const override;
Matt Sarett6d72ed92017-04-10 16:35:33 -040041 sk_sp<SkImageFilter> onMakeColorSpace(SkColorSpaceXformer*) const override;
Mike Reed667a0072016-09-23 11:43:54 -040042
43#if SK_SUPPORT_GPU
44 sk_sp<SkSpecialImage> filterImageGPU(SkSpecialImage* source,
45 sk_sp<SkSpecialImage> background,
46 const SkIPoint& backgroundOffset,
47 sk_sp<SkSpecialImage> foreground,
48 const SkIPoint& foregroundOffset,
49 const SkIRect& bounds,
50 const OutputProperties& outputProperties) const;
51#endif
52
53 void flatten(SkWriteBuffer&) const override;
54
Brian Salomon89cb8212017-01-09 10:48:23 -050055 void drawForeground(SkCanvas* canvas, SkSpecialImage*, const SkIRect&) const;
Mike Reed403d1542016-09-26 18:51:24 -040056#if SK_SUPPORT_GPU
Brian Salomon89cb8212017-01-09 10:48:23 -050057 sk_sp<GrFragmentProcessor> makeFGFrag(sk_sp<GrFragmentProcessor> bgFP) const;
Mike Reed403d1542016-09-26 18:51:24 -040058#endif
Mike Reed667a0072016-09-23 11:43:54 -040059
60private:
Brian Salomon89cb8212017-01-09 10:48:23 -050061 static sk_sp<SkFlattenable> LegacyArithmeticCreateProc(SkReadBuffer& buffer);
62
reed374772b2016-10-05 17:33:02 -070063 SkBlendMode fMode;
Mike Reed667a0072016-09-23 11:43:54 -040064
65 friend class SkXfermodeImageFilter;
66
67 typedef SkImageFilter INHERITED;
68};
69
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000070///////////////////////////////////////////////////////////////////////////////
71
reed374772b2016-10-05 17:33:02 -070072sk_sp<SkImageFilter> SkXfermodeImageFilter::Make(SkBlendMode mode,
robertphillips8c0326d2016-04-05 12:48:34 -070073 sk_sp<SkImageFilter> background,
74 sk_sp<SkImageFilter> foreground,
Mike Reed667a0072016-09-23 11:43:54 -040075 const SkImageFilter::CropRect* cropRect) {
robertphillips8c0326d2016-04-05 12:48:34 -070076 sk_sp<SkImageFilter> inputs[2] = { std::move(background), std::move(foreground) };
Mike Reed667a0072016-09-23 11:43:54 -040077 return sk_sp<SkImageFilter>(new SkXfermodeImageFilter_Base(mode, inputs, cropRect));
reedcfb6bdf2016-03-29 11:32:50 -070078}
79
reed374772b2016-10-05 17:33:02 -070080SkXfermodeImageFilter_Base::SkXfermodeImageFilter_Base(SkBlendMode mode,
81 sk_sp<SkImageFilter> inputs[2],
82 const CropRect* cropRect)
Mike Reed18342422016-10-04 10:06:20 -040083 : INHERITED(inputs, 2, cropRect)
reed374772b2016-10-05 17:33:02 -070084 , fMode(mode)
85{}
86
87static int unflatten_blendmode(SkReadBuffer& buffer, SkArithmeticParams* arith) {
88 if (buffer.isVersionLT(SkReadBuffer::kXfermodeToBlendMode_Version)) {
89 sk_sp<SkXfermode> xfer = buffer.readXfermode();
90 if (xfer) {
91 if (xfer->isArithmetic(arith)) {
92 return -1;
93 }
94 return (int)xfer->blend();
95 } else {
96 return (int)SkBlendMode::kSrcOver;
97 }
98 } else {
99 uint32_t mode = buffer.read32();
100 (void)buffer.validate(mode <= (unsigned)SkBlendMode::kLastMode);
101 return mode;
102 }
Mike Reed3ed485f2016-10-04 21:35:32 +0000103}
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000104
Mike Reed667a0072016-09-23 11:43:54 -0400105sk_sp<SkFlattenable> SkXfermodeImageFilter_Base::CreateProc(SkReadBuffer& buffer) {
reed9fa60da2014-08-21 07:59:51 -0700106 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 2);
reed374772b2016-10-05 17:33:02 -0700107 SkArithmeticParams arith;
108 int mode = unflatten_blendmode(buffer, &arith);
109 if (mode >= 0) {
110 return SkXfermodeImageFilter::Make((SkBlendMode)mode, common.getInput(0),
111 common.getInput(1), &common.cropRect());
112 } else {
Brian Salomon89cb8212017-01-09 10:48:23 -0500113 return SkArithmeticImageFilter::Make(arith.fK[0], arith.fK[1], arith.fK[2], arith.fK[3],
114 arith.fEnforcePMColor, common.getInput(0),
115 common.getInput(1), &common.cropRect());
reed374772b2016-10-05 17:33:02 -0700116 }
reed9fa60da2014-08-21 07:59:51 -0700117}
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000118
Mike Reed667a0072016-09-23 11:43:54 -0400119void SkXfermodeImageFilter_Base::flatten(SkWriteBuffer& buffer) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000120 this->INHERITED::flatten(buffer);
reed374772b2016-10-05 17:33:02 -0700121 buffer.write32((unsigned)fMode);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000122}
123
Mike Reed667a0072016-09-23 11:43:54 -0400124sk_sp<SkSpecialImage> SkXfermodeImageFilter_Base::onFilterImage(SkSpecialImage* source,
robertphillipsf230c6d2016-04-15 12:47:42 -0700125 const Context& ctx,
126 SkIPoint* offset) const {
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000127 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
robertphillipsf230c6d2016-04-15 12:47:42 -0700128 sk_sp<SkSpecialImage> background(this->filterInput(0, source, ctx, &backgroundOffset));
129
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000130 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
robertphillipsf230c6d2016-04-15 12:47:42 -0700131 sk_sp<SkSpecialImage> foreground(this->filterInput(1, source, ctx, &foregroundOffset));
132
133 SkIRect foregroundBounds = SkIRect::EmptyIRect();
134 if (foreground) {
135 foregroundBounds = SkIRect::MakeXYWH(foregroundOffset.x(), foregroundOffset.y(),
136 foreground->width(), foreground->height());
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000137 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000138
robertphillipsf230c6d2016-04-15 12:47:42 -0700139 SkIRect srcBounds = SkIRect::EmptyIRect();
140 if (background) {
141 srcBounds = SkIRect::MakeXYWH(backgroundOffset.x(), backgroundOffset.y(),
142 background->width(), background->height());
143 }
144
senorblanco9db04272016-03-31 08:24:29 -0700145 srcBounds.join(foregroundBounds);
robertphillipsf230c6d2016-04-15 12:47:42 -0700146 if (srcBounds.isEmpty()) {
147 return nullptr;
148 }
149
senorblanco9db04272016-03-31 08:24:29 -0700150 SkIRect bounds;
151 if (!this->applyCropRect(ctx, srcBounds, &bounds)) {
robertphillipsf230c6d2016-04-15 12:47:42 -0700152 return nullptr;
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000153 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000154
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000155 offset->fX = bounds.left();
156 offset->fY = bounds.top();
robertphillipsf230c6d2016-04-15 12:47:42 -0700157
158#if SK_SUPPORT_GPU
159 if (source->isTextureBacked()) {
160 return this->filterImageGPU(source,
161 background, backgroundOffset,
162 foreground, foregroundOffset,
brianosman2a75e5d2016-09-22 07:15:37 -0700163 bounds, ctx.outputProperties());
robertphillipsf230c6d2016-04-15 12:47:42 -0700164 }
165#endif
166
brianosmaneed6b0e2016-09-23 13:04:05 -0700167 sk_sp<SkSpecialSurface> surf(source->makeSurface(ctx.outputProperties(), bounds.size()));
robertphillipsf230c6d2016-04-15 12:47:42 -0700168 if (!surf) {
169 return nullptr;
170 }
171
172 SkCanvas* canvas = surf->getCanvas();
173 SkASSERT(canvas);
174
robertphillips3806b8f2016-05-12 10:46:39 -0700175 canvas->clear(0x0); // can't count on background to fully clear the background
robertphillipsf230c6d2016-04-15 12:47:42 -0700176 canvas->translate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bounds.top()));
177
robertphillipsf230c6d2016-04-15 12:47:42 -0700178 if (background) {
Mike Reed403d1542016-09-26 18:51:24 -0400179 SkPaint paint;
reed374772b2016-10-05 17:33:02 -0700180 paint.setBlendMode(SkBlendMode::kSrc);
robertphillipsf230c6d2016-04-15 12:47:42 -0700181 background->draw(canvas,
182 SkIntToScalar(backgroundOffset.fX), SkIntToScalar(backgroundOffset.fY),
183 &paint);
184 }
185
Mike Reed403d1542016-09-26 18:51:24 -0400186 this->drawForeground(canvas, foreground.get(), foregroundBounds);
Mike Reed10ff5bf2016-09-26 20:40:57 +0000187
188 return surf->makeImageSnapshot();
Mike Reed0530c882016-09-26 15:33:00 -0400189}
190
Matt Sarett6d72ed92017-04-10 16:35:33 -0400191sk_sp<SkImageFilter> SkXfermodeImageFilter_Base::onMakeColorSpace(SkColorSpaceXformer* xformer)
192const {
193 SkASSERT(2 == this->countInputs());
194 if (!this->getInput(0) && !this->getInput(1)) {
195 return sk_ref_sp(const_cast<SkXfermodeImageFilter_Base*>(this));
196 }
197
198 sk_sp<SkImageFilter> background =
199 this->getInput(0) ? this->getInput(0)->makeColorSpace(xformer) : nullptr;
200 sk_sp<SkImageFilter> foreground =
201 this->getInput(1) ? this->getInput(1)->makeColorSpace(xformer) : nullptr;
202
203 return SkXfermodeImageFilter::Make(fMode, std::move(background), std::move(foreground),
204 this->getCropRectIfSet());
205}
206
Mike Reed403d1542016-09-26 18:51:24 -0400207void SkXfermodeImageFilter_Base::drawForeground(SkCanvas* canvas, SkSpecialImage* img,
208 const SkIRect& fgBounds) const {
209 SkPaint paint;
reed374772b2016-10-05 17:33:02 -0700210 paint.setBlendMode(fMode);
Mike Reed403d1542016-09-26 18:51:24 -0400211 if (img) {
212 img->draw(canvas, SkIntToScalar(fgBounds.fLeft), SkIntToScalar(fgBounds.fTop), &paint);
213 }
214
215 SkAutoCanvasRestore acr(canvas, true);
Mike Reedc1f77742016-12-09 09:00:50 -0500216 canvas->clipRect(SkRect::Make(fgBounds), kDifference_SkClipOp);
Mike Reed403d1542016-09-26 18:51:24 -0400217 paint.setColor(0);
218 canvas->drawPaint(paint);
219}
220
robertphillipsf3f5bad2014-12-19 13:49:15 -0800221#ifndef SK_IGNORE_TO_STRING
Mike Reed667a0072016-09-23 11:43:54 -0400222void SkXfermodeImageFilter_Base::toString(SkString* str) const {
robertphillipsf3f5bad2014-12-19 13:49:15 -0800223 str->appendf("SkXfermodeImageFilter: (");
reed374772b2016-10-05 17:33:02 -0700224 str->appendf("blendmode: (%d)", (int)fMode);
robertphillips63195182015-06-08 06:21:14 -0700225 if (this->getInput(0)) {
226 str->appendf("foreground: (");
227 this->getInput(0)->toString(str);
228 str->appendf(")");
229 }
230 if (this->getInput(1)) {
231 str->appendf("background: (");
232 this->getInput(1)->toString(str);
233 str->appendf(")");
234 }
235 str->append(")");
robertphillipsf3f5bad2014-12-19 13:49:15 -0800236}
237#endif
238
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000239#if SK_SUPPORT_GPU
240
robertphillips4f037942016-02-09 05:09:27 -0800241#include "SkXfermode_proccoeff.h"
242
Mike Reed667a0072016-09-23 11:43:54 -0400243sk_sp<SkSpecialImage> SkXfermodeImageFilter_Base::filterImageGPU(
brianosman2a75e5d2016-09-22 07:15:37 -0700244 SkSpecialImage* source,
245 sk_sp<SkSpecialImage> background,
246 const SkIPoint& backgroundOffset,
247 sk_sp<SkSpecialImage> foreground,
248 const SkIPoint& foregroundOffset,
249 const SkIRect& bounds,
250 const OutputProperties& outputProperties) const {
robertphillipsf230c6d2016-04-15 12:47:42 -0700251 SkASSERT(source->isTextureBacked());
252
253 GrContext* context = source->getContext();
254
Robert Phillips8e1c4e62017-02-19 12:27:01 -0500255 sk_sp<GrTextureProxy> backgroundProxy, foregroundProxy;
256
robertphillipsf230c6d2016-04-15 12:47:42 -0700257 if (background) {
Robert Phillips8e1c4e62017-02-19 12:27:01 -0500258 backgroundProxy = background->asTextureProxyRef(context);
egdaniel38cd0552015-01-14 08:05:11 -0800259 }
260
robertphillipsf230c6d2016-04-15 12:47:42 -0700261 if (foreground) {
Robert Phillips8e1c4e62017-02-19 12:27:01 -0500262 foregroundProxy = foreground->asTextureProxyRef(context);
senorblanco06d54ad2016-01-13 13:48:54 -0800263 }
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000264
joshualittb2456052015-07-08 09:36:59 -0700265 GrPaint paint;
bungeman06ca8ec2016-06-09 08:01:03 -0700266 sk_sp<GrFragmentProcessor> bgFP;
robertphillips4f037942016-02-09 05:09:27 -0800267
Robert Phillips8e1c4e62017-02-19 12:27:01 -0500268 if (backgroundProxy) {
269 SkMatrix bgMatrix = SkMatrix::MakeTrans(-SkIntToScalar(backgroundOffset.fX),
270 -SkIntToScalar(backgroundOffset.fY));
Brian Osman7dc54522016-12-20 10:36:35 -0500271 sk_sp<GrColorSpaceXform> bgXform = GrColorSpaceXform::Make(background->getColorSpace(),
272 outputProperties.colorSpace());
bungeman06ca8ec2016-06-09 08:01:03 -0700273 bgFP = GrTextureDomainEffect::Make(
Robert Phillips296b1cc2017-03-15 10:42:12 -0400274 context->resourceProvider(), std::move(backgroundProxy),
275 std::move(bgXform), bgMatrix,
Robert Phillipse98234f2017-01-09 14:23:59 -0500276 GrTextureDomain::MakeTexelDomain(background->subset()),
robertphillips4f037942016-02-09 05:09:27 -0800277 GrTextureDomain::kDecal_Mode,
Brian Salomon514baff2016-11-17 15:17:07 -0500278 GrSamplerParams::kNone_FilterMode);
robertphillips4f037942016-02-09 05:09:27 -0800279 } else {
Brian Osman618d3042016-10-25 10:51:28 -0400280 bgFP = GrConstColorProcessor::Make(GrColor4f::TransparentBlack(),
281 GrConstColorProcessor::kIgnore_InputMode);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000282 }
283
Robert Phillips8e1c4e62017-02-19 12:27:01 -0500284 if (foregroundProxy) {
285 SkMatrix fgMatrix = SkMatrix::MakeTrans(-SkIntToScalar(foregroundOffset.fX),
286 -SkIntToScalar(foregroundOffset.fY));
Brian Osman7dc54522016-12-20 10:36:35 -0500287 sk_sp<GrColorSpaceXform> fgXform = GrColorSpaceXform::Make(foreground->getColorSpace(),
288 outputProperties.colorSpace());
bungeman06ca8ec2016-06-09 08:01:03 -0700289 sk_sp<GrFragmentProcessor> foregroundFP;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000290
bungeman06ca8ec2016-06-09 08:01:03 -0700291 foregroundFP = GrTextureDomainEffect::Make(
Robert Phillips296b1cc2017-03-15 10:42:12 -0400292 context->resourceProvider(), std::move(foregroundProxy),
293 std::move(fgXform), fgMatrix,
Robert Phillipse98234f2017-01-09 14:23:59 -0500294 GrTextureDomain::MakeTexelDomain(foreground->subset()),
robertphillips4f037942016-02-09 05:09:27 -0800295 GrTextureDomain::kDecal_Mode,
Brian Salomon514baff2016-11-17 15:17:07 -0500296 GrSamplerParams::kNone_FilterMode);
senorblancod762ca22015-04-07 12:16:55 -0700297
bungeman06ca8ec2016-06-09 08:01:03 -0700298 paint.addColorFragmentProcessor(std::move(foregroundFP));
robertphillips4f037942016-02-09 05:09:27 -0800299
Mike Reed403d1542016-09-26 18:51:24 -0400300 sk_sp<GrFragmentProcessor> xferFP = this->makeFGFrag(bgFP);
robertphillips4f037942016-02-09 05:09:27 -0800301
302 // A null 'xferFP' here means kSrc_Mode was used in which case we can just proceed
303 if (xferFP) {
bungeman06ca8ec2016-06-09 08:01:03 -0700304 paint.addColorFragmentProcessor(std::move(xferFP));
robertphillips4f037942016-02-09 05:09:27 -0800305 }
306 } else {
bungeman06ca8ec2016-06-09 08:01:03 -0700307 paint.addColorFragmentProcessor(std::move(bgFP));
bsalomonae4738f2015-09-15 15:33:27 -0700308 }
robertphillips4f037942016-02-09 05:09:27 -0800309
reed374772b2016-10-05 17:33:02 -0700310 paint.setPorterDuffXPFactory(SkBlendMode::kSrc);
robertphillipsea461502015-05-26 11:38:03 -0700311
Robert Phillips7f6cd902016-11-10 17:03:43 -0500312 sk_sp<GrRenderTargetContext> renderTargetContext(context->makeDeferredRenderTargetContext(
313 SkBackingFit::kApprox, bounds.width(), bounds.height(),
314 GrRenderableConfigForColorSpace(outputProperties.colorSpace()),
315 sk_ref_sp(outputProperties.colorSpace())));
Brian Osman11052242016-10-27 14:47:55 -0400316 if (!renderTargetContext) {
robertphillipsf230c6d2016-04-15 12:47:42 -0700317 return nullptr;
robertphillipsea461502015-05-26 11:38:03 -0700318 }
Brian Osman11052242016-10-27 14:47:55 -0400319 paint.setGammaCorrect(renderTargetContext->isGammaCorrect());
robertphillipsea461502015-05-26 11:38:03 -0700320
senorblanco06d54ad2016-01-13 13:48:54 -0800321 SkMatrix matrix;
322 matrix.setTranslate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bounds.top()));
Brian Salomon82f44312017-01-11 13:42:54 -0500323 renderTargetContext->drawRect(GrNoClip(), std::move(paint), GrAA::kNo, matrix,
324 SkRect::Make(bounds));
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000325
Robert Phillips7f6cd902016-11-10 17:03:43 -0500326 return SkSpecialImage::MakeDeferredFromGpu(context,
327 SkIRect::MakeWH(bounds.width(), bounds.height()),
328 kNeedNewImageUniqueID_SpecialImage,
Robert Phillipsf200a902017-01-30 13:27:37 -0500329 renderTargetContext->asTextureProxyRef(),
Robert Phillips75a475c2017-01-13 09:18:59 -0500330 renderTargetContext->refColorSpace());
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000331}
332
Mike Reed403d1542016-09-26 18:51:24 -0400333sk_sp<GrFragmentProcessor>
334SkXfermodeImageFilter_Base::makeFGFrag(sk_sp<GrFragmentProcessor> bgFP) const {
335 // A null fMode is interpreted to mean kSrcOver_Mode (to match raster).
reed374772b2016-10-05 17:33:02 -0700336 SkXfermode* xfer = SkXfermode::Peek(fMode);
337 sk_sp<SkXfermode> srcover;
338 if (!xfer) {
Mike Reed403d1542016-09-26 18:51:24 -0400339 // It would be awesome to use SkXfermode::Create here but it knows better
340 // than us and won't return a kSrcOver_Mode SkXfermode. That means we
341 // have to get one the hard way.
342 struct ProcCoeff rec;
Mike Reed6a015542016-11-09 10:38:09 -0500343 rec.fProc = SkXfermode::GetProc(SkBlendMode::kSrcOver);
344 SkXfermode::ModeAsCoeff(SkBlendMode::kSrcOver, &rec.fSC, &rec.fDC);
Mike Reed403d1542016-09-26 18:51:24 -0400345
Mike Reed7d954ad2016-10-28 15:42:34 -0400346 srcover.reset(new SkProcCoeffXfermode(rec, SkBlendMode::kSrcOver));
reed374772b2016-10-05 17:33:02 -0700347 xfer = srcover.get();
348
Mike Reed403d1542016-09-26 18:51:24 -0400349 }
reed374772b2016-10-05 17:33:02 -0700350 return xfer->makeFragmentProcessorForImageFilter(std::move(bgFP));
Mike Reed403d1542016-09-26 18:51:24 -0400351}
352
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000353#endif
Mike Reed667a0072016-09-23 11:43:54 -0400354///////////////////////////////////////////////////////////////////////////////////////////////////
355
Brian Salomon89cb8212017-01-09 10:48:23 -0500356sk_sp<SkFlattenable> SkXfermodeImageFilter_Base::LegacyArithmeticCreateProc(SkReadBuffer& buffer) {
Mike Reed403d1542016-09-26 18:51:24 -0400357 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 2);
Brian Salomon89cb8212017-01-09 10:48:23 -0500358 // skip the unused mode (srcover) field
reed374772b2016-10-05 17:33:02 -0700359 SkDEBUGCODE(int mode =) unflatten_blendmode(buffer, nullptr);
360 if (!buffer.isValid()) {
361 return nullptr;
362 }
363 SkASSERT(SkBlendMode::kSrcOver == (SkBlendMode)mode);
Mike Reed403d1542016-09-26 18:51:24 -0400364 float k[4];
365 for (int i = 0; i < 4; ++i) {
366 k[i] = buffer.readScalar();
367 }
368 const bool enforcePMColor = buffer.readBool();
Brian Salomon89cb8212017-01-09 10:48:23 -0500369 return SkArithmeticImageFilter::Make(k[0], k[1], k[2], k[3], enforcePMColor, common.getInput(0),
370 common.getInput(1), &common.cropRect());
Mike Reed403d1542016-09-26 18:51:24 -0400371}
372
373///////////////////////////////////////////////////////////////////////////////////////////////////
374
Mike Reed667a0072016-09-23 11:43:54 -0400375SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_START(SkXfermodeImageFilter)
376 SK_DEFINE_FLATTENABLE_REGISTRAR_ENTRY(SkXfermodeImageFilter_Base)
377 // manually register the legacy serialized name "SkXfermodeImageFilter"
378 SkFlattenable::Register("SkXfermodeImageFilter", SkXfermodeImageFilter_Base::CreateProc,
379 SkFlattenable::kSkImageFilter_Type);
Brian Salomon89cb8212017-01-09 10:48:23 -0500380 // manually register the legacy serialized name "SkArithmeticImageFilter" from when that filter
381 // was implemented as a xfermode image filter.
382 SkFlattenable::Register("SkArithmeticImageFilter",
383 SkXfermodeImageFilter_Base::LegacyArithmeticCreateProc,
384 SkFlattenable::kSkImageFilter_Type);
Mike Reed667a0072016-09-23 11:43:54 -0400385SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_END