blob: 3df59a87d2bdc425aa2a39861cf49ec6d40edb28 [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"
10#include "SkColorPriv.h"
11#include "SkFlattenableBuffers.h"
12#include "SkXfermode.h"
13#if SK_SUPPORT_GPU
14#include "GrContext.h"
15#include "effects/GrSimpleTextureEffect.h"
16#include "SkGr.h"
17#include "SkImageFilterUtils.h"
18#endif
19
20///////////////////////////////////////////////////////////////////////////////
21
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000022SkXfermodeImageFilter::SkXfermodeImageFilter(SkXfermode* mode,
23 SkImageFilter* background,
24 SkImageFilter* foreground)
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000025 : INHERITED(background, foreground), fMode(mode) {
26 SkSafeRef(fMode);
27}
28
29SkXfermodeImageFilter::~SkXfermodeImageFilter() {
30 SkSafeUnref(fMode);
31}
32
33SkXfermodeImageFilter::SkXfermodeImageFilter(SkFlattenableReadBuffer& buffer)
34 : INHERITED(buffer) {
35 fMode = buffer.readFlattenableT<SkXfermode>();
36}
37
38void SkXfermodeImageFilter::flatten(SkFlattenableWriteBuffer& buffer) const {
39 this->INHERITED::flatten(buffer);
40 buffer.writeFlattenable(fMode);
41}
42
43bool SkXfermodeImageFilter::onFilterImage(Proxy* proxy,
44 const SkBitmap& src,
45 const SkMatrix& ctm,
46 SkBitmap* dst,
47 SkIPoint* offset) {
48 SkBitmap background = src, foreground = src;
49 SkImageFilter* backgroundInput = getInput(0);
50 SkImageFilter* foregroundInput = getInput(1);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000051 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
52 if (backgroundInput &&
53 !backgroundInput->filterImage(proxy, src, ctm, &background, &backgroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000054 return false;
55 }
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000056 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
57 if (foregroundInput &&
58 !foregroundInput->filterImage(proxy, src, ctm, &foreground, &foregroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000059 return false;
60 }
61 dst->setConfig(background.config(), background.width(), background.height());
62 dst->allocPixels();
63 SkCanvas canvas(*dst);
64 SkPaint paint;
65 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
66 canvas.drawBitmap(background, 0, 0, &paint);
67 paint.setXfermode(fMode);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000068 canvas.drawBitmap(foreground,
69 SkIntToScalar(foregroundOffset.fX - backgroundOffset.fX),
70 SkIntToScalar(foregroundOffset.fY - backgroundOffset.fY),
71 &paint);
72 offset->fX += backgroundOffset.fX;
73 offset->fY += backgroundOffset.fY;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000074 return true;
75}
76
77#if SK_SUPPORT_GPU
78
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000079bool SkXfermodeImageFilter::filterImageGPU(Proxy* proxy,
80 const SkBitmap& src,
81 SkBitmap* result,
82 SkIPoint* offset) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000083 SkBitmap background;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000084 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
85 if (!SkImageFilterUtils::GetInputResultGPU(getInput(0), proxy, src, &background,
86 &backgroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000087 return false;
88 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +000089 GrTexture* backgroundTex = background.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000090 SkBitmap foreground;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000091 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
92 if (!SkImageFilterUtils::GetInputResultGPU(getInput(1), proxy, src, &foreground,
93 &foregroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000094 return false;
95 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +000096 GrTexture* foregroundTex = foreground.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000097 GrContext* context = foregroundTex->getContext();
98
99 GrEffectRef* xferEffect = NULL;
100
101 GrTextureDesc desc;
102 desc.fFlags = kRenderTarget_GrTextureFlagBit | kNoStencil_GrTextureFlagBit;
103 desc.fWidth = src.width();
104 desc.fHeight = src.height();
105 desc.fConfig = kSkia8888_GrPixelConfig;
106
107 GrAutoScratchTexture ast(context, desc);
108 SkAutoTUnref<GrTexture> dst(ast.detach());
109
110 GrContext::AutoRenderTarget art(context, dst->asRenderTarget());
111
112 SkXfermode::Coeff sm, dm;
113 if (!SkXfermode::AsNewEffectOrCoeff(fMode, context, &xferEffect, &sm, &dm, backgroundTex)) {
114 return false;
115 }
116
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000117 SkMatrix foregroundMatrix = GrEffect::MakeDivByTextureWHMatrix(foregroundTex);
118 foregroundMatrix.preTranslate(SkIntToScalar(backgroundOffset.fX-foregroundOffset.fX),
119 SkIntToScalar(backgroundOffset.fY-foregroundOffset.fY));
120
121
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000122 SkRect srcRect;
123 src.getBounds(&srcRect);
124 if (NULL != xferEffect) {
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000125 GrPaint paint;
126 paint.addColorTextureEffect(foregroundTex, foregroundMatrix);
127 paint.addColorEffect(xferEffect)->unref();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000128 context->drawRect(paint, srcRect);
129 } else {
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000130 GrPaint backgroundPaint;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000131 SkMatrix backgroundMatrix = GrEffect::MakeDivByTextureWHMatrix(backgroundTex);
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000132 backgroundPaint.addColorTextureEffect(backgroundTex, backgroundMatrix);
133 context->drawRect(backgroundPaint, srcRect);
134
135 GrPaint foregroundPaint;
136 foregroundPaint.setBlendFunc(sk_blend_to_grblend(sm), sk_blend_to_grblend(dm));
137 foregroundPaint.addColorTextureEffect(foregroundTex, foregroundMatrix);
138 context->drawRect(foregroundPaint, srcRect);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000139 }
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000140 offset->fX += backgroundOffset.fX;
141 offset->fY += backgroundOffset.fY;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000142 return SkImageFilterUtils::WrapTexture(dst, src.width(), src.height(), result);
143}
144
145#endif