blob: 74ca643b23a5bb71a80b857720db61c59a80e1ec [file] [log] [blame]
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +00001
2/*
3 * Copyright 2013 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8#include "gm.h"
9#include "SkBitmap.h"
10#include "SkGradientShader.h"
11#include "SkXfermode.h"
12#include "SkColorPriv.h"
13
14#if SK_SUPPORT_GPU
15#include "GrContext.h"
16#include "SkGpuDevice.h"
17#endif
18
19namespace skiagm {
20
21/**
22 * This tests drawing device-covering rects with solid colors and bitmap shaders over a
23 * checkerboard background using different xfermodes.
24 */
25class Xfermodes3GM : public GM {
26public:
27 Xfermodes3GM() {}
28
29protected:
30 virtual SkString onShortName() SK_OVERRIDE {
31 return SkString("xfermodes3");
32 }
33
34 virtual SkISize onISize() SK_OVERRIDE {
tfarinaf5393182014-06-09 23:59:03 -070035 return SkISize::Make(630, 1215);
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +000036 }
37
38 virtual void onDrawBackground(SkCanvas* canvas) SK_OVERRIDE {
39 SkPaint bgPaint;
40 bgPaint.setColor(0xFF70D0E0);
41 canvas->drawPaint(bgPaint);
42 }
43
44 virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE {
45 canvas->translate(SkIntToScalar(10), SkIntToScalar(20));
46
47 SkPaint labelP;
48 labelP.setAntiAlias(true);
Cary Clark992c7b02014-07-31 08:58:44 -040049 sk_tool_utils::set_portable_typeface(&labelP);
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +000050
51 static const SkColor kSolidColors[] = {
52 SK_ColorTRANSPARENT,
53 SK_ColorBLUE,
54 0x80808000
55 };
56
57 static const SkColor kBmpAlphas[] = {
58 0xff,
59 0x80,
60 };
61
62 SkAutoTUnref<SkCanvas> tempCanvas(this->possiblyCreateTempCanvas(canvas, kSize, kSize));
63
64 int test = 0;
65 int x = 0, y = 0;
commit-bot@chromium.orgde1559c2013-08-14 19:20:45 +000066 static const struct { SkPaint::Style fStyle; SkScalar fWidth; } kStrokes[] = {
67 {SkPaint::kFill_Style, 0},
68 {SkPaint::kStroke_Style, SkIntToScalar(kSize) / 2},
69 };
70 for (size_t s = 0; s < SK_ARRAY_COUNT(kStrokes); ++s) {
71 for (size_t m = 0; m <= SkXfermode::kLastMode; ++m) {
72 SkXfermode::Mode mode = static_cast<SkXfermode::Mode>(m);
73 canvas->drawText(SkXfermode::ModeName(mode),
74 strlen(SkXfermode::ModeName(mode)),
75 SkIntToScalar(x),
76 SkIntToScalar(y + kSize + 3) + labelP.getTextSize(),
77 labelP);
78 for (size_t c = 0; c < SK_ARRAY_COUNT(kSolidColors); ++c) {
79 SkPaint modePaint;
80 modePaint.setXfermodeMode(mode);
81 modePaint.setColor(kSolidColors[c]);
82 modePaint.setStyle(kStrokes[s].fStyle);
83 modePaint.setStrokeWidth(kStrokes[s].fWidth);
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +000084
commit-bot@chromium.orgde1559c2013-08-14 19:20:45 +000085 this->drawMode(canvas, x, y, kSize, kSize, modePaint, tempCanvas.get());
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +000086
commit-bot@chromium.orgde1559c2013-08-14 19:20:45 +000087 ++test;
88 x += kSize + 10;
89 if (!(test % kTestsPerRow)) {
90 x = 0;
91 y += kSize + 30;
92 }
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +000093 }
commit-bot@chromium.orgde1559c2013-08-14 19:20:45 +000094 for (size_t a = 0; a < SK_ARRAY_COUNT(kBmpAlphas); ++a) {
95 SkPaint modePaint;
96 modePaint.setXfermodeMode(mode);
97 modePaint.setAlpha(kBmpAlphas[a]);
98 modePaint.setShader(fBmpShader);
99 modePaint.setStyle(kStrokes[s].fStyle);
100 modePaint.setStrokeWidth(kStrokes[s].fWidth);
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000101
commit-bot@chromium.orgde1559c2013-08-14 19:20:45 +0000102 this->drawMode(canvas, x, y, kSize, kSize, modePaint, tempCanvas.get());
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000103
commit-bot@chromium.orgde1559c2013-08-14 19:20:45 +0000104 ++test;
105 x += kSize + 10;
106 if (!(test % kTestsPerRow)) {
107 x = 0;
108 y += kSize + 30;
109 }
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000110 }
111 }
112 }
113 }
114
115private:
116 /**
117 * GrContext has optimizations around full rendertarget draws that can be replaced with clears.
118 * We are trying to test those. We could use saveLayer() to create small SkGpuDevices but
119 * saveLayer() uses the texture cache. This means that the actual render target may be larger
120 * than the layer. Because the clip will contain the layer's bounds, no draws will be full-RT.
121 * So when running on a GPU canvas we explicitly create a temporary canvas using a texture with
122 * dimensions exactly matching the layer size.
123 */
124 SkCanvas* possiblyCreateTempCanvas(SkCanvas* baseCanvas, int w, int h) {
125 SkCanvas* tempCanvas = NULL;
126#if SK_SUPPORT_GPU
reed52d9ac62014-06-30 09:05:34 -0700127 GrContext* context = baseCanvas->getGrContext();
bsalomon49f085d2014-09-05 13:34:00 -0700128 if (context) {
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000129 GrTextureDesc desc;
130 desc.fWidth = w;
131 desc.fHeight = h;
reed52d9ac62014-06-30 09:05:34 -0700132 desc.fConfig = SkImageInfo2GrPixelConfig(baseCanvas->imageInfo());
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000133 desc.fFlags = kRenderTarget_GrTextureFlagBit;
134 SkAutoTUnref<GrSurface> surface(context->createUncachedTexture(desc, NULL, 0));
robertphillips@google.com1f2f3382013-08-29 11:54:56 +0000135 SkAutoTUnref<SkBaseDevice> device(SkGpuDevice::Create(surface.get()));
bsalomon49f085d2014-09-05 13:34:00 -0700136 if (device.get()) {
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000137 tempCanvas = SkNEW_ARGS(SkCanvas, (device.get()));
138 }
139 }
140#endif
141 return tempCanvas;
142 }
143
144 void drawMode(SkCanvas* canvas,
145 int x, int y, int w, int h,
146 const SkPaint& modePaint, SkCanvas* layerCanvas) {
147 canvas->save();
148
149 canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
150
151 SkRect r = SkRect::MakeWH(SkIntToScalar(w), SkIntToScalar(h));
152
153 SkCanvas* modeCanvas;
154 if (NULL == layerCanvas) {
commit-bot@chromium.org091a5942014-04-18 14:19:31 +0000155 canvas->saveLayer(&r, NULL);
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000156 modeCanvas = canvas;
157 } else {
158 modeCanvas = layerCanvas;
159 }
160
161 SkPaint bgPaint;
162 bgPaint.setAntiAlias(false);
163 bgPaint.setShader(fBGShader);
164 modeCanvas->drawRect(r, bgPaint);
165 modeCanvas->drawRect(r, modePaint);
166 modeCanvas = NULL;
167
168 if (NULL == layerCanvas) {
169 canvas->restore();
170 } else {
commit-bot@chromium.orgc3bd8af2014-02-13 17:14:46 +0000171 SkAutoROCanvasPixels ropixels(layerCanvas);
172 SkBitmap bitmap;
173 if (ropixels.asROBitmap(&bitmap)) {
174 canvas->drawBitmap(bitmap, 0, 0);
175 }
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000176 }
177
178 r.inset(-SK_ScalarHalf, -SK_ScalarHalf);
179 SkPaint borderPaint;
180 borderPaint.setStyle(SkPaint::kStroke_Style);
181 canvas->drawRect(r, borderPaint);
182
183 canvas->restore();
184 }
185
186 virtual void onOnceBeforeDraw() SK_OVERRIDE {
187 static const uint32_t kCheckData[] = {
188 SkPackARGB32(0xFF, 0x40, 0x40, 0x40),
189 SkPackARGB32(0xFF, 0xD0, 0xD0, 0xD0),
190 SkPackARGB32(0xFF, 0xD0, 0xD0, 0xD0),
191 SkPackARGB32(0xFF, 0x40, 0x40, 0x40)
192 };
193 SkBitmap bg;
reed@google.comeb9a46c2014-01-25 16:46:20 +0000194 bg.allocN32Pixels(2, 2, true);
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000195 SkAutoLockPixels bgAlp(bg);
196 memcpy(bg.getPixels(), kCheckData, sizeof(kCheckData));
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000197
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000198 SkMatrix lm;
199 lm.setScale(SkIntToScalar(kCheckSize), SkIntToScalar(kCheckSize));
commit-bot@chromium.org9c9005a2014-04-28 14:55:39 +0000200 fBGShader.reset(SkShader::CreateBitmapShader(bg,
201 SkShader::kRepeat_TileMode,
202 SkShader::kRepeat_TileMode,
203 &lm));
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000204
205 SkPaint bmpPaint;
206 static const SkPoint kCenter = { SkIntToScalar(kSize) / 2, SkIntToScalar(kSize) / 2 };
207 static const SkColor kColors[] = { SK_ColorTRANSPARENT, 0x80800000,
208 0xF020F060, SK_ColorWHITE };
209 bmpPaint.setShader(SkGradientShader::CreateRadial(kCenter,
210 3 * SkIntToScalar(kSize) / 4,
211 kColors,
212 NULL,
213 SK_ARRAY_COUNT(kColors),
214 SkShader::kRepeat_TileMode))->unref();
215
216 SkBitmap bmp;
reed@google.comeb9a46c2014-01-25 16:46:20 +0000217 bmp.allocN32Pixels(kSize, kSize);
commit-bot@chromium.org103f2d02013-08-08 21:13:38 +0000218 SkCanvas bmpCanvas(bmp);
219
220 bmpCanvas.clear(SK_ColorTRANSPARENT);
221 SkRect rect = { SkIntToScalar(kSize) / 8, SkIntToScalar(kSize) / 8,
222 7 * SkIntToScalar(kSize) / 8, 7 * SkIntToScalar(kSize) / 8};
223 bmpCanvas.drawRect(rect, bmpPaint);
224
225 fBmpShader.reset(SkShader::CreateBitmapShader(bmp,
226 SkShader::kClamp_TileMode,
227 SkShader::kClamp_TileMode));
228 }
229
230 enum {
231 kCheckSize = 8,
232 kSize = 30,
233 kTestsPerRow = 15,
234 };
235
236 SkAutoTUnref<SkShader> fBGShader;
237 SkAutoTUnref<SkShader> fBmpShader;
238
239 typedef GM INHERITED;
240};
241
242//////////////////////////////////////////////////////////////////////////////
243
244DEF_GM(return new Xfermodes3GM;)
245
246}