blob: bca5223db4cfa24446ba99c6cbb03f4454eb407c [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"
17#include "effects/GrSimpleTextureEffect.h"
18#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],
senorblanco5e5f9482014-08-26 12:27:12 -070025 const CropRect* cropRect,
26 uint32_t uniqueID)
27 : INHERITED(2, inputs, cropRect, uniqueID), fMode(mode) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000028 SkSafeRef(fMode);
29}
30
31SkXfermodeImageFilter::~SkXfermodeImageFilter() {
32 SkSafeUnref(fMode);
33}
34
reed9fa60da2014-08-21 07:59:51 -070035#ifdef SK_SUPPORT_LEGACY_DEEPFLATTENING
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000036SkXfermodeImageFilter::SkXfermodeImageFilter(SkReadBuffer& buffer)
commit-bot@chromium.orgce33d602013-11-25 21:46:31 +000037 : INHERITED(2, buffer) {
reed@google.com35348222013-10-16 13:05:06 +000038 fMode = buffer.readXfermode();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000039}
reed9fa60da2014-08-21 07:59:51 -070040#endif
41
42SkFlattenable* SkXfermodeImageFilter::CreateProc(SkReadBuffer& buffer) {
43 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 2);
44 SkAutoTUnref<SkXfermode> mode(buffer.readXfermode());
senorblanco5e5f9482014-08-26 12:27:12 -070045 return Create(mode, common.getInput(0), common.getInput(1), &common.cropRect(), common.uniqueID());
reed9fa60da2014-08-21 07:59:51 -070046}
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000047
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000048void SkXfermodeImageFilter::flatten(SkWriteBuffer& buffer) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000049 this->INHERITED::flatten(buffer);
50 buffer.writeFlattenable(fMode);
51}
52
53bool SkXfermodeImageFilter::onFilterImage(Proxy* proxy,
54 const SkBitmap& src,
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +000055 const Context& ctx,
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000056 SkBitmap* dst,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +000057 SkIPoint* offset) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000058 SkBitmap background = src, foreground = src;
59 SkImageFilter* backgroundInput = getInput(0);
60 SkImageFilter* foregroundInput = getInput(1);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000061 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
62 if (backgroundInput &&
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +000063 !backgroundInput->filterImage(proxy, src, ctx, &background, &backgroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000064 background.reset();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000065 }
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000066 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
67 if (foregroundInput &&
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +000068 !foregroundInput->filterImage(proxy, src, ctx, &foreground, &foregroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000069 foreground.reset();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000070 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000071
senorblanco@chromium.org06820192013-12-12 21:00:58 +000072 SkIRect bounds, foregroundBounds;
senorblanco@chromium.org11825292014-03-14 17:44:41 +000073 if (!applyCropRect(ctx, foreground, foregroundOffset, &foregroundBounds)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000074 foregroundBounds.setEmpty();
75 foreground.reset();
senorblanco@chromium.org29089172014-03-14 16:48:39 +000076 }
senorblanco@chromium.org11825292014-03-14 17:44:41 +000077 if (!applyCropRect(ctx, background, backgroundOffset, &bounds)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000078 bounds.setEmpty();
79 background.reset();
senorblanco@chromium.org11825292014-03-14 17:44:41 +000080 }
81 bounds.join(foregroundBounds);
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +000082 if (bounds.isEmpty()) {
83 return false;
84 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000085
86 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
commit-bot@chromium.orgcd3b15c2013-12-04 17:06:49 +000087 if (NULL == device.get()) {
88 return false;
89 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000090 SkCanvas canvas(device);
senorblanco@chromium.org06820192013-12-12 21:00:58 +000091 canvas.translate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bounds.top()));
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000092 SkPaint paint;
93 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000094 canvas.drawBitmap(background, SkIntToScalar(backgroundOffset.fX),
95 SkIntToScalar(backgroundOffset.fY), &paint);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000096 paint.setXfermode(fMode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000097 canvas.drawBitmap(foreground, SkIntToScalar(foregroundOffset.fX),
98 SkIntToScalar(foregroundOffset.fY), &paint);
senorblanco@chromium.org06820192013-12-12 21:00:58 +000099 canvas.clipRect(SkRect::Make(foregroundBounds), SkRegion::kDifference_Op);
100 paint.setColor(SK_ColorTRANSPARENT);
101 canvas.drawPaint(paint);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000102 *dst = device->accessBitmap(false);
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000103 offset->fX = bounds.left();
104 offset->fY = bounds.top();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000105 return true;
106}
107
108#if SK_SUPPORT_GPU
109
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000110bool SkXfermodeImageFilter::canFilterImageGPU() const {
joshualittb0a8a372014-09-23 09:50:21 -0700111 return fMode && fMode->asFragmentProcessor(NULL, NULL) && !cropRectIsSet();
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000112}
113
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000114bool SkXfermodeImageFilter::filterImageGPU(Proxy* proxy,
115 const SkBitmap& src,
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +0000116 const Context& ctx,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000117 SkBitmap* result,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000118 SkIPoint* offset) const {
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000119 SkBitmap background = src;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000120 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +0000121 if (getInput(0) && !getInput(0)->getInputResultGPU(proxy, src, ctx, &background,
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000122 &backgroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000123 return onFilterImage(proxy, src, ctx, result, offset);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000124 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000125 GrTexture* backgroundTex = background.getTexture();
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000126 SkBitmap foreground = src;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000127 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
senorblanco@chromium.org4cb543d2014-03-14 15:44:01 +0000128 if (getInput(1) && !getInput(1)->getInputResultGPU(proxy, src, ctx, &foreground,
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000129 &foregroundOffset)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000130 return onFilterImage(proxy, src, ctx, result, offset);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000131 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000132 GrTexture* foregroundTex = foreground.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000133 GrContext* context = foregroundTex->getContext();
134
joshualittb0a8a372014-09-23 09:50:21 -0700135 GrFragmentProcessor* xferProcessor = NULL;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000136
137 GrTextureDesc desc;
138 desc.fFlags = kRenderTarget_GrTextureFlagBit | kNoStencil_GrTextureFlagBit;
139 desc.fWidth = src.width();
140 desc.fHeight = src.height();
141 desc.fConfig = kSkia8888_GrPixelConfig;
142
143 GrAutoScratchTexture ast(context, desc);
senorblanco673d9732014-08-15 10:48:43 -0700144 if (NULL == ast.texture()) {
145 return false;
146 }
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000147 SkAutoTUnref<GrTexture> dst(ast.detach());
148
149 GrContext::AutoRenderTarget art(context, dst->asRenderTarget());
150
joshualittb0a8a372014-09-23 09:50:21 -0700151 if (!fMode || !fMode->asFragmentProcessor(&xferProcessor, backgroundTex)) {
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000152 // canFilterImageGPU() should've taken care of this
153 SkASSERT(false);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000154 return false;
155 }
156
bsalomon6267f812014-08-29 15:05:53 -0700157 SkMatrix foregroundMatrix = GrCoordTransform::MakeDivByTextureWHMatrix(foregroundTex);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000158 foregroundMatrix.preTranslate(SkIntToScalar(backgroundOffset.fX-foregroundOffset.fX),
159 SkIntToScalar(backgroundOffset.fY-foregroundOffset.fY));
160
161
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000162 SkRect srcRect;
163 src.getBounds(&srcRect);
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000164
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000165 GrPaint paint;
joshualittb0a8a372014-09-23 09:50:21 -0700166 paint.addColorTextureProcessor(foregroundTex, foregroundMatrix);
167 paint.addColorProcessor(xferProcessor)->unref();
senorblanco@chromium.orgee845ae2014-04-01 19:15:23 +0000168 context->drawRect(paint, srcRect);
169
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000170 offset->fX = backgroundOffset.fX;
171 offset->fY = backgroundOffset.fY;
senorblanco@chromium.org6aa6fec2014-03-03 22:13:56 +0000172 WrapTexture(dst, src.width(), src.height(), result);
173 return true;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000174}
175
176#endif