blob: 5175fd4890784ea7aa82cded860d0e0cf6818178 [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"
12#include "SkFlattenableBuffers.h"
13#include "SkXfermode.h"
14#if SK_SUPPORT_GPU
15#include "GrContext.h"
16#include "effects/GrSimpleTextureEffect.h"
17#include "SkGr.h"
18#include "SkImageFilterUtils.h"
19#endif
20
21///////////////////////////////////////////////////////////////////////////////
22
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000023SkXfermodeImageFilter::SkXfermodeImageFilter(SkXfermode* mode,
24 SkImageFilter* background,
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000025 SkImageFilter* foreground,
senorblanco@chromium.orgb295fb62013-10-10 13:51:19 +000026 const CropRect* cropRect)
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000027 : INHERITED(background, foreground, cropRect), fMode(mode) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000028 SkSafeRef(fMode);
29}
30
31SkXfermodeImageFilter::~SkXfermodeImageFilter() {
32 SkSafeUnref(fMode);
33}
34
35SkXfermodeImageFilter::SkXfermodeImageFilter(SkFlattenableReadBuffer& buffer)
36 : INHERITED(buffer) {
37 fMode = buffer.readFlattenableT<SkXfermode>();
38}
39
40void SkXfermodeImageFilter::flatten(SkFlattenableWriteBuffer& buffer) const {
41 this->INHERITED::flatten(buffer);
42 buffer.writeFlattenable(fMode);
43}
44
45bool SkXfermodeImageFilter::onFilterImage(Proxy* proxy,
46 const SkBitmap& src,
47 const SkMatrix& ctm,
48 SkBitmap* dst,
49 SkIPoint* offset) {
50 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 &&
55 !backgroundInput->filterImage(proxy, src, ctm, &background, &backgroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000056 return false;
57 }
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000058 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
59 if (foregroundInput &&
60 !foregroundInput->filterImage(proxy, src, ctm, &foreground, &foregroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000061 return false;
62 }
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000063
64 SkIRect bounds;
65 background.getBounds(&bounds);
66 if (!applyCropRect(&bounds, ctm)) {
67 return false;
68 }
69 backgroundOffset.fX -= bounds.left();
70 backgroundOffset.fY -= bounds.top();
71 foregroundOffset.fX -= bounds.left();
72 foregroundOffset.fY -= bounds.top();
73
74 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
75 SkCanvas canvas(device);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000076 SkPaint paint;
77 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000078 canvas.drawBitmap(background, SkIntToScalar(backgroundOffset.fX),
79 SkIntToScalar(backgroundOffset.fY), &paint);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000080 paint.setXfermode(fMode);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +000081 canvas.drawBitmap(foreground, SkIntToScalar(foregroundOffset.fX),
82 SkIntToScalar(foregroundOffset.fY), &paint);
83 *dst = device->accessBitmap(false);
84 offset->fX += bounds.left();
85 offset->fY += bounds.top();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000086 return true;
87}
88
89#if SK_SUPPORT_GPU
90
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000091bool SkXfermodeImageFilter::filterImageGPU(Proxy* proxy,
92 const SkBitmap& src,
commit-bot@chromium.org1aa54bf2013-08-05 16:53:50 +000093 const SkMatrix& ctm,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000094 SkBitmap* result,
95 SkIPoint* offset) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000096 SkBitmap background;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000097 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
commit-bot@chromium.org1aa54bf2013-08-05 16:53:50 +000098 if (!SkImageFilterUtils::GetInputResultGPU(getInput(0), proxy, src, ctm, &background,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000099 &backgroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000100 return false;
101 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000102 GrTexture* backgroundTex = background.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000103 SkBitmap foreground;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000104 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
commit-bot@chromium.org1aa54bf2013-08-05 16:53:50 +0000105 if (!SkImageFilterUtils::GetInputResultGPU(getInput(1), proxy, src, ctm, &foreground,
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000106 &foregroundOffset)) {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000107 return false;
108 }
commit-bot@chromium.orgb8d00db2013-06-26 19:18:23 +0000109 GrTexture* foregroundTex = foreground.getTexture();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000110 GrContext* context = foregroundTex->getContext();
111
112 GrEffectRef* xferEffect = NULL;
113
114 GrTextureDesc desc;
115 desc.fFlags = kRenderTarget_GrTextureFlagBit | kNoStencil_GrTextureFlagBit;
116 desc.fWidth = src.width();
117 desc.fHeight = src.height();
118 desc.fConfig = kSkia8888_GrPixelConfig;
119
120 GrAutoScratchTexture ast(context, desc);
121 SkAutoTUnref<GrTexture> dst(ast.detach());
122
123 GrContext::AutoRenderTarget art(context, dst->asRenderTarget());
124
125 SkXfermode::Coeff sm, dm;
126 if (!SkXfermode::AsNewEffectOrCoeff(fMode, context, &xferEffect, &sm, &dm, backgroundTex)) {
127 return false;
128 }
129
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000130 SkMatrix foregroundMatrix = GrEffect::MakeDivByTextureWHMatrix(foregroundTex);
131 foregroundMatrix.preTranslate(SkIntToScalar(backgroundOffset.fX-foregroundOffset.fX),
132 SkIntToScalar(backgroundOffset.fY-foregroundOffset.fY));
133
134
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000135 SkRect srcRect;
136 src.getBounds(&srcRect);
137 if (NULL != xferEffect) {
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000138 GrPaint paint;
139 paint.addColorTextureEffect(foregroundTex, foregroundMatrix);
140 paint.addColorEffect(xferEffect)->unref();
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000141 context->drawRect(paint, srcRect);
142 } else {
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000143 GrPaint backgroundPaint;
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000144 SkMatrix backgroundMatrix = GrEffect::MakeDivByTextureWHMatrix(backgroundTex);
commit-bot@chromium.org42dacab2013-07-13 17:24:24 +0000145 backgroundPaint.addColorTextureEffect(backgroundTex, backgroundMatrix);
146 context->drawRect(backgroundPaint, srcRect);
147
148 GrPaint foregroundPaint;
149 foregroundPaint.setBlendFunc(sk_blend_to_grblend(sm), sk_blend_to_grblend(dm));
150 foregroundPaint.addColorTextureEffect(foregroundTex, foregroundMatrix);
151 context->drawRect(foregroundPaint, srcRect);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000152 }
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000153 offset->fX += backgroundOffset.fX;
154 offset->fY += backgroundOffset.fY;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000155 return SkImageFilterUtils::WrapTexture(dst, src.width(), src.height(), result);
156}
157
158#endif