blob: d43699db6435eb938a91f2058d62b80f3ff793a3 [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"
19#include "SkImageFilterUtils.h"
20#endif
21
22///////////////////////////////////////////////////////////////////////////////
23
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000024SkXfermodeImageFilter::SkXfermodeImageFilter(SkXfermode* mode,
25 SkImageFilter* background,
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000026 SkImageFilter* foreground,
senorblanco@chromium.orgb295fb62013-10-10 13:51:19 +000027 const CropRect* cropRect)
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000028 : INHERITED(background, foreground, cropRect), fMode(mode) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000029 SkSafeRef(fMode);
30}
31
32SkXfermodeImageFilter::~SkXfermodeImageFilter() {
33 SkSafeUnref(fMode);
34}
35
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}
40
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000041void SkXfermodeImageFilter::flatten(SkWriteBuffer& buffer) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000042 this->INHERITED::flatten(buffer);
43 buffer.writeFlattenable(fMode);
44}
45
46bool SkXfermodeImageFilter::onFilterImage(Proxy* proxy,
47 const SkBitmap& src,
48 const SkMatrix& ctm,
49 SkBitmap* dst,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +000050 SkIPoint* offset) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000051 SkBitmap background = src, foreground = src;
52 SkImageFilter* backgroundInput = getInput(0);
53 SkImageFilter* foregroundInput = getInput(1);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000054 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
55 if (backgroundInput &&
56 !backgroundInput->filterImage(proxy, src, ctm, &background, &backgroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000057 return false;
58 }
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000059 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
60 if (foregroundInput &&
61 !foregroundInput->filterImage(proxy, src, ctm, &foreground, &foregroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000062 return false;
63 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000064
senorblanco@chromium.org06820192013-12-12 21:00:58 +000065 SkIRect bounds, foregroundBounds;
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000066 background.getBounds(&bounds);
senorblanco@chromium.org06820192013-12-12 21:00:58 +000067 bounds.offset(backgroundOffset);
68 foreground.getBounds(&foregroundBounds);
69 foregroundBounds.offset(foregroundOffset);
70 bounds.join(foregroundBounds);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000071 if (!applyCropRect(&bounds, ctm)) {
72 return false;
73 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000074
75 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
commit-bot@chromium.orgcd3b15c2013-12-04 17:06:49 +000076 if (NULL == device.get()) {
77 return false;
78 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000079 SkCanvas canvas(device);
senorblanco@chromium.org06820192013-12-12 21:00:58 +000080 canvas.translate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bounds.top()));
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000081 SkPaint paint;
82 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000083 canvas.drawBitmap(background, SkIntToScalar(backgroundOffset.fX),
84 SkIntToScalar(backgroundOffset.fY), &paint);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000085 paint.setXfermode(fMode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000086 canvas.drawBitmap(foreground, SkIntToScalar(foregroundOffset.fX),
87 SkIntToScalar(foregroundOffset.fY), &paint);
senorblanco@chromium.org06820192013-12-12 21:00:58 +000088 canvas.clipRect(SkRect::Make(foregroundBounds), SkRegion::kDifference_Op);
89 paint.setColor(SK_ColorTRANSPARENT);
90 canvas.drawPaint(paint);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000091 *dst = device->accessBitmap(false);
senorblanco@chromium.org6776b822014-01-03 21:48:22 +000092 offset->fX = bounds.left();
93 offset->fY = bounds.top();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000094 return true;
95}
96
97#if SK_SUPPORT_GPU
98
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000099bool SkXfermodeImageFilter::filterImageGPU(Proxy* proxy,
100 const SkBitmap& src,
commit-bot@chromium.org1aa54bf2013-08-05 16:53:50 +0000101 const SkMatrix& ctm,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000102 SkBitmap* result,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000103 SkIPoint* offset) const {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000104 SkBitmap background;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000105 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
commit-bot@chromium.org1aa54bf2013-08-05 16:53:50 +0000106 if (!SkImageFilterUtils::GetInputResultGPU(getInput(0), proxy, src, ctm, &background,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000107 &backgroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000108 return false;
109 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000110 GrTexture* backgroundTex = background.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000111 SkBitmap foreground;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000112 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
commit-bot@chromium.org1aa54bf2013-08-05 16:53:50 +0000113 if (!SkImageFilterUtils::GetInputResultGPU(getInput(1), proxy, src, ctm, &foreground,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000114 &foregroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000115 return false;
116 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000117 GrTexture* foregroundTex = foreground.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000118 GrContext* context = foregroundTex->getContext();
119
120 GrEffectRef* xferEffect = NULL;
121
122 GrTextureDesc desc;
123 desc.fFlags = kRenderTarget_GrTextureFlagBit | kNoStencil_GrTextureFlagBit;
124 desc.fWidth = src.width();
125 desc.fHeight = src.height();
126 desc.fConfig = kSkia8888_GrPixelConfig;
127
128 GrAutoScratchTexture ast(context, desc);
129 SkAutoTUnref<GrTexture> dst(ast.detach());
130
131 GrContext::AutoRenderTarget art(context, dst->asRenderTarget());
132
133 SkXfermode::Coeff sm, dm;
senorblanco@chromium.orgfbcd4152013-10-23 14:03:22 +0000134 if (!SkXfermode::AsNewEffectOrCoeff(fMode, &xferEffect, &sm, &dm, backgroundTex)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000135 return false;
136 }
137
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000138 SkMatrix foregroundMatrix = GrEffect::MakeDivByTextureWHMatrix(foregroundTex);
139 foregroundMatrix.preTranslate(SkIntToScalar(backgroundOffset.fX-foregroundOffset.fX),
140 SkIntToScalar(backgroundOffset.fY-foregroundOffset.fY));
141
142
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000143 SkRect srcRect;
144 src.getBounds(&srcRect);
145 if (NULL != xferEffect) {
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000146 GrPaint paint;
147 paint.addColorTextureEffect(foregroundTex, foregroundMatrix);
148 paint.addColorEffect(xferEffect)->unref();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000149 context->drawRect(paint, srcRect);
150 } else {
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000151 GrPaint backgroundPaint;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000152 SkMatrix backgroundMatrix = GrEffect::MakeDivByTextureWHMatrix(backgroundTex);
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000153 backgroundPaint.addColorTextureEffect(backgroundTex, backgroundMatrix);
154 context->drawRect(backgroundPaint, srcRect);
155
156 GrPaint foregroundPaint;
157 foregroundPaint.setBlendFunc(sk_blend_to_grblend(sm), sk_blend_to_grblend(dm));
158 foregroundPaint.addColorTextureEffect(foregroundTex, foregroundMatrix);
159 context->drawRect(foregroundPaint, srcRect);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000160 }
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000161 offset->fX = backgroundOffset.fX;
162 offset->fY = backgroundOffset.fY;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000163 return SkImageFilterUtils::WrapTexture(dst, src.width(), src.height(), result);
164}
165
166#endif