2 * Copyright 2013 Google Inc.
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
9 #include "sk_tool_utils.h"
11 #include "SkGradientShader.h"
12 #include "SkSurface.h"
13 #include "SkBlendModePriv.h"
14 #include "SkColorPriv.h"
17 #include "GrContext.h"
23 * This tests drawing device-covering rects with solid colors and bitmap shaders over a
24 * checkerboard background using different xfermodes.
26 class Xfermodes3GM : public GM {
31 SkString onShortName() override {
32 return SkString("xfermodes3");
35 SkISize onISize() override {
36 return SkISize::Make(630, 1215);
39 void onDrawBackground(SkCanvas* canvas) override {
41 bgPaint.setColor(sk_tool_utils::color_to_565(0xFF70D0E0));
42 canvas->drawPaint(bgPaint);
45 void onDraw(SkCanvas* canvas) override {
46 canvas->translate(SkIntToScalar(10), SkIntToScalar(20));
49 labelP.setAntiAlias(true);
50 sk_tool_utils::set_portable_typeface(&labelP);
52 constexpr SkColor kSolidColors[] = {
58 constexpr SkColor kBmpAlphas[] = {
63 auto tempSurface(this->possiblyCreateTempSurface(canvas, kSize, kSize));
67 constexpr struct { SkPaint::Style fStyle; SkScalar fWidth; } kStrokes[] = {
68 {SkPaint::kFill_Style, 0},
69 {SkPaint::kStroke_Style, SkIntToScalar(kSize) / 2},
71 for (size_t s = 0; s < SK_ARRAY_COUNT(kStrokes); ++s) {
72 for (size_t m = 0; m <= (size_t)SkBlendMode::kLastMode; ++m) {
73 SkBlendMode mode = static_cast<SkBlendMode>(m);
74 canvas->drawString(SkBlendMode_Name(mode),
76 SkIntToScalar(y + kSize + 3) + labelP.getTextSize(),
78 for (size_t c = 0; c < SK_ARRAY_COUNT(kSolidColors); ++c) {
80 modePaint.setBlendMode(mode);
81 modePaint.setColor(kSolidColors[c]);
82 modePaint.setStyle(kStrokes[s].fStyle);
83 modePaint.setStrokeWidth(kStrokes[s].fWidth);
85 this->drawMode(canvas, x, y, kSize, kSize, modePaint, tempSurface.get());
89 if (!(test % kTestsPerRow)) {
94 for (size_t a = 0; a < SK_ARRAY_COUNT(kBmpAlphas); ++a) {
96 modePaint.setBlendMode(mode);
97 modePaint.setAlpha(kBmpAlphas[a]);
98 modePaint.setShader(fBmpShader);
99 modePaint.setStyle(kStrokes[s].fStyle);
100 modePaint.setStrokeWidth(kStrokes[s].fWidth);
102 this->drawMode(canvas, x, y, kSize, kSize, modePaint, tempSurface.get());
106 if (!(test % kTestsPerRow)) {
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.
124 sk_sp<SkSurface> possiblyCreateTempSurface(SkCanvas* baseCanvas, int w, int h) {
126 GrContext* context = baseCanvas->getGrContext();
127 SkImageInfo baseInfo = baseCanvas->imageInfo();
128 SkImageInfo info = SkImageInfo::Make(w, h, baseInfo.colorType(), baseInfo.alphaType(),
129 baseInfo.refColorSpace());
130 SkSurfaceProps canvasProps(SkSurfaceProps::kLegacyFontHost_InitType);
131 baseCanvas->getProps(&canvasProps);
132 return SkSurface::MakeRenderTarget(context, SkBudgeted::kNo, info, 0, &canvasProps);
138 void drawMode(SkCanvas* canvas,
139 int x, int y, int w, int h,
140 const SkPaint& modePaint, SkSurface* surface) {
142 canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
144 SkRect r = SkRect::MakeWH(SkIntToScalar(w), SkIntToScalar(h));
146 SkCanvas* modeCanvas;
147 if (nullptr == surface) {
148 canvas->saveLayer(&r, nullptr);
151 modeCanvas = surface->getCanvas();
155 bgPaint.setAntiAlias(false);
156 bgPaint.setShader(fBGShader);
157 modeCanvas->drawRect(r, bgPaint);
158 modeCanvas->drawRect(r, modePaint);
159 modeCanvas = nullptr;
161 if (nullptr == surface) {
164 surface->draw(canvas, 0, 0, nullptr);
167 r.inset(-SK_ScalarHalf, -SK_ScalarHalf);
169 borderPaint.setStyle(SkPaint::kStroke_Style);
170 canvas->drawRect(r, borderPaint);
175 void onOnceBeforeDraw() override {
176 const uint32_t kCheckData[] = {
177 SkPackARGB32(0xFF, 0x42, 0x41, 0x42),
178 SkPackARGB32(0xFF, 0xD6, 0xD3, 0xD6),
179 SkPackARGB32(0xFF, 0xD6, 0xD3, 0xD6),
180 SkPackARGB32(0xFF, 0x42, 0x41, 0x42)
183 bg.allocN32Pixels(2, 2, true);
184 memcpy(bg.getPixels(), kCheckData, sizeof(kCheckData));
187 lm.setScale(SkIntToScalar(kCheckSize), SkIntToScalar(kCheckSize));
188 fBGShader = SkShader::MakeBitmapShader(bg, SkShader::kRepeat_TileMode,
189 SkShader::kRepeat_TileMode, &lm);
192 const SkPoint kCenter = { SkIntToScalar(kSize) / 2, SkIntToScalar(kSize) / 2 };
193 const SkColor kColors[] = {
194 SK_ColorTRANSPARENT, 0x80800000, 0xF020F060, SK_ColorWHITE
196 bmpPaint.setShader(SkGradientShader::MakeRadial(kCenter, 3 * SkIntToScalar(kSize) / 4,
197 kColors, nullptr, SK_ARRAY_COUNT(kColors),
198 SkShader::kRepeat_TileMode));
201 bmp.allocN32Pixels(kSize, kSize);
202 SkCanvas bmpCanvas(bmp);
204 bmpCanvas.clear(SK_ColorTRANSPARENT);
205 SkRect rect = { SkIntToScalar(kSize) / 8, SkIntToScalar(kSize) / 8,
206 7 * SkIntToScalar(kSize) / 8, 7 * SkIntToScalar(kSize) / 8};
207 bmpCanvas.drawRect(rect, bmpPaint);
209 fBmpShader = SkShader::MakeBitmapShader(bmp, SkShader::kClamp_TileMode,
210 SkShader::kClamp_TileMode);
219 sk_sp<SkShader> fBGShader;
220 sk_sp<SkShader> fBmpShader;
222 typedef GM INHERITED;
225 //////////////////////////////////////////////////////////////////////////////
227 DEF_GM(return new Xfermodes3GM;)