blob: 76c46295a490f7d56b8aebcf90c93c52d652fa63 [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"
9#include "SkCanvas.h"
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000010#include "SkDevice.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000011#include "SkColorPriv.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000012#include "SkReadBuffer.h"
13#include "SkWriteBuffer.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000014#include "SkXfermode.h"
15#if SK_SUPPORT_GPU
16#include "GrContext.h"
senorblancod762ca22015-04-07 12:16:55 -070017#include "effects/GrTextureDomain.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000018#include "SkGr.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000019#endif
20
21///////////////////////////////////////////////////////////////////////////////
22
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000023SkXfermodeImageFilter::SkXfermodeImageFilter(SkXfermode* mode,
senorblanco9ea3d572014-07-08 09:16:22 -070024 SkImageFilter* inputs[2],
senorblanco24e06d52015-03-18 12:11:33 -070025 const CropRect* cropRect)
26 : INHERITED(2, inputs, cropRect), fMode(mode) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000027 SkSafeRef(fMode);
28}
29
30SkXfermodeImageFilter::~SkXfermodeImageFilter() {
31 SkSafeUnref(fMode);
32}
33
reed9fa60da2014-08-21 07:59:51 -070034SkFlattenable* SkXfermodeImageFilter::CreateProc(SkReadBuffer& buffer) {
35 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 2);
36 SkAutoTUnref<SkXfermode> mode(buffer.readXfermode());
senorblanco24e06d52015-03-18 12:11:33 -070037 return Create(mode, common.getInput(0), common.getInput(1), &common.cropRect());
reed9fa60da2014-08-21 07:59:51 -070038}
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000039
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000040void SkXfermodeImageFilter::flatten(SkWriteBuffer& buffer) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000041 this->INHERITED::flatten(buffer);
42 buffer.writeFlattenable(fMode);
43}
44
45bool SkXfermodeImageFilter::onFilterImage(Proxy* proxy,
46 const SkBitmap& src,
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +000047 const Context& ctx,
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000048 SkBitmap* dst,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +000049 SkIPoint* offset) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000050 SkBitmap background = src, foreground = src;
51 SkImageFilter* backgroundInput = getInput(0);
52 SkImageFilter* foregroundInput = getInput(1);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000053 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
54 if (backgroundInput &&
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +000055 !backgroundInput->filterImage(proxy, src, ctx, &background, &backgroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000056 background.reset();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000057 }
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000058 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
59 if (foregroundInput &&
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +000060 !foregroundInput->filterImage(proxy, src, ctx, &foreground, &foregroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000061 foreground.reset();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000062 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000063
senorblanco@chromium.org06820192013-12-12 21:00:58 +000064 SkIRect bounds, foregroundBounds;
senorblanco@chromium.org11825292014-03-14 17:44:41 +000065 if (!applyCropRect(ctx, foreground, foregroundOffset, &foregroundBounds)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000066 foregroundBounds.setEmpty();
67 foreground.reset();
senorblanco@chromium.org29089172014-03-14 16:48:39 +000068 }
senorblanco@chromium.org11825292014-03-14 17:44:41 +000069 if (!applyCropRect(ctx, background, backgroundOffset, &bounds)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000070 bounds.setEmpty();
71 background.reset();
senorblanco@chromium.org11825292014-03-14 17:44:41 +000072 }
73 bounds.join(foregroundBounds);
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000074 if (bounds.isEmpty()) {
75 return false;
76 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000077
78 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
commit-bot@chromium.orgcd3b15c2013-12-04 17:06:49 +000079 if (NULL == device.get()) {
80 return false;
81 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000082 SkCanvas canvas(device);
senorblanco@chromium.org06820192013-12-12 21:00:58 +000083 canvas.translate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bounds.top()));
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000084 SkPaint paint;
85 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000086 canvas.drawBitmap(background, SkIntToScalar(backgroundOffset.fX),
87 SkIntToScalar(backgroundOffset.fY), &paint);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000088 paint.setXfermode(fMode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000089 canvas.drawBitmap(foreground, SkIntToScalar(foregroundOffset.fX),
90 SkIntToScalar(foregroundOffset.fY), &paint);
senorblanco@chromium.org06820192013-12-12 21:00:58 +000091 canvas.clipRect(SkRect::Make(foregroundBounds), SkRegion::kDifference_Op);
92 paint.setColor(SK_ColorTRANSPARENT);
93 canvas.drawPaint(paint);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000094 *dst = device->accessBitmap(false);
senorblanco@chromium.org6776b822014-01-03 21:48:22 +000095 offset->fX = bounds.left();
96 offset->fY = bounds.top();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000097 return true;
98}
99
robertphillipsf3f5bad2014-12-19 13:49:15 -0800100#ifndef SK_IGNORE_TO_STRING
101void SkXfermodeImageFilter::toString(SkString* str) const {
102 str->appendf("SkXfermodeImageFilter: (");
103 str->appendf("xfermode: (");
104 if (fMode) {
105 fMode->toString(str);
106 }
107 str->append("))");
108}
109#endif
110
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000111#if SK_SUPPORT_GPU
112
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000113bool SkXfermodeImageFilter::canFilterImageGPU() const {
joshualittb0a8a372014-09-23 09:50:21 -0700114 return fMode && fMode->asFragmentProcessor(NULL, NULL) && !cropRectIsSet();
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000115}
116
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000117bool SkXfermodeImageFilter::filterImageGPU(Proxy* proxy,
118 const SkBitmap& src,
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +0000119 const Context& ctx,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000120 SkBitmap* result,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000121 SkIPoint* offset) const {
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000122 SkBitmap background = src;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000123 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +0000124 if (getInput(0) && !getInput(0)->getInputResultGPU(proxy, src, ctx, &background,
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000125 &backgroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000126 return onFilterImage(proxy, src, ctx, result, offset);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000127 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000128 GrTexture* backgroundTex = background.getTexture();
egdaniel38cd0552015-01-14 08:05:11 -0800129
130 if (NULL == backgroundTex) {
131 SkASSERT(false);
132 return false;
133 }
134
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000135 SkBitmap foreground = src;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000136 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +0000137 if (getInput(1) && !getInput(1)->getInputResultGPU(proxy, src, ctx, &foreground,
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000138 &foregroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000139 return onFilterImage(proxy, src, ctx, result, offset);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000140 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000141 GrTexture* foregroundTex = foreground.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000142 GrContext* context = foregroundTex->getContext();
143
joshualittb0a8a372014-09-23 09:50:21 -0700144 GrFragmentProcessor* xferProcessor = NULL;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000145
bsalomonf2703d82014-10-28 14:33:06 -0700146 GrSurfaceDesc desc;
bsalomon6bc1b5f2015-02-23 09:06:38 -0800147 desc.fFlags = kRenderTarget_GrSurfaceFlag;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000148 desc.fWidth = src.width();
149 desc.fHeight = src.height();
150 desc.fConfig = kSkia8888_GrPixelConfig;
bsalomond309e7a2015-04-30 14:18:54 -0700151 SkAutoTUnref<GrTexture> dst(context->textureProvider()->refScratchTexture(
152 desc, GrTextureProvider::kApprox_ScratchTexMatch));
bsalomone3059732014-10-14 11:47:22 -0700153 if (!dst) {
senorblanco673d9732014-08-15 10:48:43 -0700154 return false;
155 }
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000156
joshualittb0a8a372014-09-23 09:50:21 -0700157 if (!fMode || !fMode->asFragmentProcessor(&xferProcessor, backgroundTex)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000158 // canFilterImageGPU() should've taken care of this
159 SkASSERT(false);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000160 return false;
161 }
162
bsalomon6267f812014-08-29 15:05:53 -0700163 SkMatrix foregroundMatrix = GrCoordTransform::MakeDivByTextureWHMatrix(foregroundTex);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000164 foregroundMatrix.preTranslate(SkIntToScalar(backgroundOffset.fX-foregroundOffset.fX),
165 SkIntToScalar(backgroundOffset.fY-foregroundOffset.fY));
166
167
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000168 SkRect srcRect;
169 src.getBounds(&srcRect);
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000170
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000171 GrPaint paint;
senorblancod762ca22015-04-07 12:16:55 -0700172 SkAutoTUnref<GrFragmentProcessor> foregroundDomain(GrTextureDomainEffect::Create(
173 foregroundTex, foregroundMatrix,
174 GrTextureDomain::MakeTexelDomain(foregroundTex, foreground.bounds()),
175 GrTextureDomain::kDecal_Mode,
176 GrTextureParams::kNone_FilterMode)
177 );
178
179 paint.addColorProcessor(foregroundDomain.get());
joshualittb0a8a372014-09-23 09:50:21 -0700180 paint.addColorProcessor(xferProcessor)->unref();
joshualitt570d2f82015-02-25 13:19:48 -0800181 context->drawRect(dst->asRenderTarget(), GrClip::WideOpen(), paint, SkMatrix::I(), srcRect);
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000182
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000183 offset->fX = backgroundOffset.fX;
184 offset->fY = backgroundOffset.fY;
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000185 WrapTexture(dst, src.width(), src.height(), result);
186 return true;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000187}
188
189#endif
robertphillipsf3f5bad2014-12-19 13:49:15 -0800190