2 * Copyright 2013 The Android Open Source Project
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
8 #include "SkXfermodeImageFilter.h"
11 #include "SkColorPriv.h"
12 #include "SkReadBuffer.h"
13 #include "SkWriteBuffer.h"
14 #include "SkXfermode.h"
16 #include "GrContext.h"
17 #include "effects/GrSimpleTextureEffect.h"
21 ///////////////////////////////////////////////////////////////////////////////
23 SkXfermodeImageFilter::SkXfermodeImageFilter(SkXfermode* mode,
24 SkImageFilter* inputs[2],
25 const CropRect* cropRect,
27 : INHERITED(2, inputs, cropRect, uniqueID), fMode(mode) {
31 SkXfermodeImageFilter::~SkXfermodeImageFilter() {
35 #ifdef SK_SUPPORT_LEGACY_DEEPFLATTENING
36 SkXfermodeImageFilter::SkXfermodeImageFilter(SkReadBuffer& buffer)
37 : INHERITED(2, buffer) {
38 fMode = buffer.readXfermode();
42 SkFlattenable* SkXfermodeImageFilter::CreateProc(SkReadBuffer& buffer) {
43 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 2);
44 SkAutoTUnref<SkXfermode> mode(buffer.readXfermode());
45 return Create(mode, common.getInput(0), common.getInput(1), &common.cropRect(), common.uniqueID());
48 void SkXfermodeImageFilter::flatten(SkWriteBuffer& buffer) const {
49 this->INHERITED::flatten(buffer);
50 buffer.writeFlattenable(fMode);
53 bool SkXfermodeImageFilter::onFilterImage(Proxy* proxy,
57 SkIPoint* offset) const {
58 SkBitmap background = src, foreground = src;
59 SkImageFilter* backgroundInput = getInput(0);
60 SkImageFilter* foregroundInput = getInput(1);
61 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
62 if (backgroundInput &&
63 !backgroundInput->filterImage(proxy, src, ctx, &background, &backgroundOffset)) {
66 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
67 if (foregroundInput &&
68 !foregroundInput->filterImage(proxy, src, ctx, &foreground, &foregroundOffset)) {
72 SkIRect bounds, foregroundBounds;
73 if (!applyCropRect(ctx, foreground, foregroundOffset, &foregroundBounds)) {
74 foregroundBounds.setEmpty();
77 if (!applyCropRect(ctx, background, backgroundOffset, &bounds)) {
81 bounds.join(foregroundBounds);
82 if (bounds.isEmpty()) {
86 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
87 if (NULL == device.get()) {
90 SkCanvas canvas(device);
91 canvas.translate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bounds.top()));
93 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
94 canvas.drawBitmap(background, SkIntToScalar(backgroundOffset.fX),
95 SkIntToScalar(backgroundOffset.fY), &paint);
96 paint.setXfermode(fMode);
97 canvas.drawBitmap(foreground, SkIntToScalar(foregroundOffset.fX),
98 SkIntToScalar(foregroundOffset.fY), &paint);
99 canvas.clipRect(SkRect::Make(foregroundBounds), SkRegion::kDifference_Op);
100 paint.setColor(SK_ColorTRANSPARENT);
101 canvas.drawPaint(paint);
102 *dst = device->accessBitmap(false);
103 offset->fX = bounds.left();
104 offset->fY = bounds.top();
110 bool SkXfermodeImageFilter::canFilterImageGPU() const {
111 return fMode && fMode->asFragmentProcessor(NULL, NULL) && !cropRectIsSet();
114 bool SkXfermodeImageFilter::filterImageGPU(Proxy* proxy,
118 SkIPoint* offset) const {
119 SkBitmap background = src;
120 SkIPoint backgroundOffset = SkIPoint::Make(0, 0);
121 if (getInput(0) && !getInput(0)->getInputResultGPU(proxy, src, ctx, &background,
122 &backgroundOffset)) {
123 return onFilterImage(proxy, src, ctx, result, offset);
125 GrTexture* backgroundTex = background.getTexture();
126 SkBitmap foreground = src;
127 SkIPoint foregroundOffset = SkIPoint::Make(0, 0);
128 if (getInput(1) && !getInput(1)->getInputResultGPU(proxy, src, ctx, &foreground,
129 &foregroundOffset)) {
130 return onFilterImage(proxy, src, ctx, result, offset);
132 GrTexture* foregroundTex = foreground.getTexture();
133 GrContext* context = foregroundTex->getContext();
135 GrFragmentProcessor* xferProcessor = NULL;
138 desc.fFlags = kRenderTarget_GrSurfaceFlag | kNoStencil_GrSurfaceFlag;
139 desc.fWidth = src.width();
140 desc.fHeight = src.height();
141 desc.fConfig = kSkia8888_GrPixelConfig;
142 SkAutoTUnref<GrTexture> dst(
143 context->refScratchTexture(desc, GrContext::kApprox_ScratchTexMatch));
147 GrContext::AutoRenderTarget art(context, dst->asRenderTarget());
149 if (!fMode || !fMode->asFragmentProcessor(&xferProcessor, backgroundTex)) {
150 // canFilterImageGPU() should've taken care of this
155 SkMatrix foregroundMatrix = GrCoordTransform::MakeDivByTextureWHMatrix(foregroundTex);
156 foregroundMatrix.preTranslate(SkIntToScalar(backgroundOffset.fX-foregroundOffset.fX),
157 SkIntToScalar(backgroundOffset.fY-foregroundOffset.fY));
161 src.getBounds(&srcRect);
164 paint.addColorTextureProcessor(foregroundTex, foregroundMatrix);
165 paint.addColorProcessor(xferProcessor)->unref();
166 context->drawRect(paint, srcRect);
168 offset->fX = backgroundOffset.fX;
169 offset->fY = backgroundOffset.fY;
170 WrapTexture(dst, src.width(), src.height(), result);