blob: f8424ba0f80ecb357c0b6f6d21ec9c4d2d0d8aa7 [file] [log] [blame]
reed4a8126e2014-09-22 07:29:03 -07001/*
2 * Copyright 2014 Google Inc.
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 "gm.h"
Mike Klein33d20552017-03-22 13:47:51 -04009#include "sk_tool_utils.h"
reed4a8126e2014-09-22 07:29:03 -070010#include "SkGradientShader.h"
11#include "SkSurface.h"
12#include "SkSurfaceProps.h"
Mike Reeda697df92018-10-26 07:28:30 -040013#include "SkTextUtils.h"
reed4a8126e2014-09-22 07:29:03 -070014
15#define W 200
16#define H 100
17
reed1a9b9642016-03-13 14:13:58 -070018static sk_sp<SkShader> make_shader() {
reed4a8126e2014-09-22 07:29:03 -070019 int a = 0x99;
20 int b = 0xBB;
21 SkPoint pts[] = { { 0, 0 }, { W, H } };
22 SkColor colors[] = { SkColorSetRGB(a, a, a), SkColorSetRGB(b, b, b) };
reed1a9b9642016-03-13 14:13:58 -070023 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkShader::kClamp_TileMode);
reed4a8126e2014-09-22 07:29:03 -070024}
25
reed7c123542016-08-18 09:30:44 -070026static sk_sp<SkSurface> make_surface(GrContext* ctx, const SkImageInfo& info, SkPixelGeometry geo) {
27 SkSurfaceProps props(0, geo);
reed4a8126e2014-09-22 07:29:03 -070028 if (ctx) {
reede8f30622016-03-23 18:59:25 -070029 return SkSurface::MakeRenderTarget(ctx, SkBudgeted::kNo, info, 0, &props);
reed4a8126e2014-09-22 07:29:03 -070030 } else {
reede8f30622016-03-23 18:59:25 -070031 return SkSurface::MakeRaster(info, &props);
reed4a8126e2014-09-22 07:29:03 -070032 }
33}
34
35static void test_draw(SkCanvas* canvas, const char label[]) {
36 SkPaint paint;
37
38 paint.setAntiAlias(true);
reed4a8126e2014-09-22 07:29:03 -070039 paint.setDither(true);
40
reed1a9b9642016-03-13 14:13:58 -070041 paint.setShader(make_shader());
reed4a8126e2014-09-22 07:29:03 -070042 canvas->drawRect(SkRect::MakeWH(W, H), paint);
halcanary96fcdcc2015-08-27 07:41:13 -070043 paint.setShader(nullptr);
reed4a8126e2014-09-22 07:29:03 -070044
45 paint.setColor(SK_ColorWHITE);
Mike Reedb579f072019-01-03 15:45:53 -050046 SkFont font(sk_tool_utils::create_portable_typeface(), 32);
47 font.setEdging(SkFont::Edging::kSubpixelAntiAlias);
48 SkTextUtils::DrawString(canvas, label, W / 2, H * 3 / 4, font, paint,
49 SkTextUtils::kCenter_Align);
reed4a8126e2014-09-22 07:29:03 -070050}
51
52class SurfacePropsGM : public skiagm::GM {
53public:
54 SurfacePropsGM() {}
55
56protected:
mtklein36352bf2015-03-25 18:17:31 -070057 SkString onShortName() override {
reed4a8126e2014-09-22 07:29:03 -070058 return SkString("surfaceprops");
59 }
60
mtklein36352bf2015-03-25 18:17:31 -070061 SkISize onISize() override {
reed7c123542016-08-18 09:30:44 -070062 return SkISize::Make(W, H * 5);
reed4a8126e2014-09-22 07:29:03 -070063 }
64
mtklein36352bf2015-03-25 18:17:31 -070065 void onDraw(SkCanvas* canvas) override {
reed4a8126e2014-09-22 07:29:03 -070066 GrContext* ctx = canvas->getGrContext();
67
68 // must be opaque to have a hope of testing LCD text
brianosman0e22eb82016-08-30 07:07:59 -070069 const SkImageInfo info = SkImageInfo::MakeN32(W, H, kOpaque_SkAlphaType);
reed4a8126e2014-09-22 07:29:03 -070070
71 const struct {
72 SkPixelGeometry fGeo;
73 const char* fLabel;
scroggoe6cad6b2016-05-09 13:20:58 -070074 } recs[] = {
reed4a8126e2014-09-22 07:29:03 -070075 { kUnknown_SkPixelGeometry, "Unknown" },
76 { kRGB_H_SkPixelGeometry, "RGB_H" },
77 { kBGR_H_SkPixelGeometry, "BGR_H" },
78 { kRGB_V_SkPixelGeometry, "RGB_V" },
79 { kBGR_V_SkPixelGeometry, "BGR_V" },
80 };
halcanary9d524f22016-03-29 09:03:52 -070081
reed4a8126e2014-09-22 07:29:03 -070082 SkScalar x = 0;
reed7c123542016-08-18 09:30:44 -070083 SkScalar y = 0;
84 for (const auto& rec : recs) {
85 auto surface(make_surface(ctx, info, rec.fGeo));
86 if (!surface) {
87 SkDebugf("failed to create surface! label: %s", rec.fLabel);
88 continue;
reed4a8126e2014-09-22 07:29:03 -070089 }
reed7c123542016-08-18 09:30:44 -070090 test_draw(surface->getCanvas(), rec.fLabel);
91 surface->draw(canvas, x, y, nullptr);
92 y += H;
reed4a8126e2014-09-22 07:29:03 -070093 }
94 }
95
96private:
97 typedef GM INHERITED;
98};
reed4a8126e2014-09-22 07:29:03 -070099DEF_GM( return new SurfacePropsGM )
reed4af267b2014-11-21 08:46:37 -0800100
101#ifdef SK_DEBUG
102static bool equal(const SkSurfaceProps& a, const SkSurfaceProps& b) {
103 return a.flags() == b.flags() && a.pixelGeometry() == b.pixelGeometry();
104}
105#endif
106
107class NewSurfaceGM : public skiagm::GM {
108public:
109 NewSurfaceGM() {}
110
111protected:
mtklein36352bf2015-03-25 18:17:31 -0700112 SkString onShortName() override {
reed4af267b2014-11-21 08:46:37 -0800113 return SkString("surfacenew");
114 }
115
mtklein36352bf2015-03-25 18:17:31 -0700116 SkISize onISize() override {
reed4af267b2014-11-21 08:46:37 -0800117 return SkISize::Make(300, 140);
118 }
119
120 static void drawInto(SkCanvas* canvas) {
121 canvas->drawColor(SK_ColorRED);
122 }
123
mtklein36352bf2015-03-25 18:17:31 -0700124 void onDraw(SkCanvas* canvas) override {
reed4af267b2014-11-21 08:46:37 -0800125 SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100);
126
Cary Clarka24712e2018-09-05 18:41:40 +0000127 auto surf(sk_tool_utils::makeSurface(canvas, info, nullptr));
reed4af267b2014-11-21 08:46:37 -0800128 drawInto(surf->getCanvas());
129
reed9ce9d672016-03-17 10:51:11 -0700130 sk_sp<SkImage> image(surf->makeImageSnapshot());
halcanary96fcdcc2015-08-27 07:41:13 -0700131 canvas->drawImage(image, 10, 10, nullptr);
reed4af267b2014-11-21 08:46:37 -0800132
reede8f30622016-03-23 18:59:25 -0700133 auto surf2(surf->makeSurface(info));
reed4af267b2014-11-21 08:46:37 -0800134 drawInto(surf2->getCanvas());
135
136 // Assert that the props were communicated transitively through the first image
137 SkASSERT(equal(surf->props(), surf2->props()));
138
reed9ce9d672016-03-17 10:51:11 -0700139 sk_sp<SkImage> image2(surf2->makeImageSnapshot());
140 canvas->drawImage(image2.get(), 10 + SkIntToScalar(image->width()) + 10, 10, nullptr);
reed4af267b2014-11-21 08:46:37 -0800141 }
142
143private:
144 typedef GM INHERITED;
145};
146DEF_GM( return new NewSurfaceGM )
Mike Reedd94abc52017-03-06 16:37:07 -0500147
148///////////////////////////////////////////////////////////////////////////////////////////////////
149
150DEF_SIMPLE_GM(copy_on_write_retain, canvas, 256, 256) {
151 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
Cary Clarka24712e2018-09-05 18:41:40 +0000152 sk_sp<SkSurface> surf = sk_tool_utils::makeSurface(canvas, info);
Mike Reedd94abc52017-03-06 16:37:07 -0500153
154 surf->getCanvas()->clear(SK_ColorRED);
155 // its important that image survives longer than the next draw, so the surface will see
156 // an outstanding image, and have to decide if it should retain or discard those pixels
157 sk_sp<SkImage> image = surf->makeImageSnapshot();
158
159 // normally a clear+opaque should trigger the discard optimization, but since we have a clip
160 // it should not (we need the previous red pixels).
161 surf->getCanvas()->clipRect(SkRect::MakeWH(128, 256));
162 surf->getCanvas()->clear(SK_ColorBLUE);
163
164 // expect to see two rects: blue | red
165 canvas->drawImage(surf->makeImageSnapshot(), 0, 0, nullptr);
166}
Mike Reeda1361362017-03-07 09:37:29 -0500167
168DEF_SIMPLE_GM(copy_on_write_savelayer, canvas, 256, 256) {
169 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
Cary Clarka24712e2018-09-05 18:41:40 +0000170 sk_sp<SkSurface> surf = sk_tool_utils::makeSurface(canvas, info);
Mike Reeda1361362017-03-07 09:37:29 -0500171 surf->getCanvas()->clear(SK_ColorRED);
172 // its important that image survives longer than the next draw, so the surface will see
173 // an outstanding image, and have to decide if it should retain or discard those pixels
174 sk_sp<SkImage> image = surf->makeImageSnapshot();
175
176 // now draw into a full-screen layer. This should (a) trigger a copy-on-write, but it should
177 // not trigger discard, even tho its alpha (SK_ColorBLUE) is opaque, since it is in a layer
178 // with a non-opaque paint.
179 SkPaint paint;
180 paint.setAlpha(0x40);
181 surf->getCanvas()->saveLayer({0, 0, 256, 256}, &paint);
182 surf->getCanvas()->clear(SK_ColorBLUE);
183 surf->getCanvas()->restore();
184
185 // expect to see two rects: blue blended on red
186 canvas->drawImage(surf->makeImageSnapshot(), 0, 0, nullptr);
187}
Mike Reed114bde82018-11-21 09:12:09 -0500188
189DEF_SIMPLE_GM(surface_underdraw, canvas, 256, 256) {
190 SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256, nullptr);
191 auto surf = sk_tool_utils::makeSurface(canvas, info);
192
193 const SkIRect subset = SkIRect::MakeLTRB(180, 0, 256, 256);
194
195 // noisy background
196 {
197 SkPoint pts[] = {{0, 0}, {40, 50}};
198 SkColor colors[] = {SK_ColorRED, SK_ColorBLUE};
199 auto sh = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkShader::kRepeat_TileMode);
200 SkPaint paint;
201 paint.setShader(sh);
202 surf->getCanvas()->drawPaint(paint);
203 }
204
205 // save away the right-hand strip, then clear it
206 sk_sp<SkImage> saveImg = surf->makeImageSnapshot(subset);
207 {
208 SkPaint paint;
209 paint.setBlendMode(SkBlendMode::kClear);
210 surf->getCanvas()->drawRect(SkRect::Make(subset), paint);
211 }
212
213 // draw the "foreground"
214 {
215 SkPaint paint;
216 paint.setColor(SK_ColorGREEN);
217 SkRect r = { 0, 10, 256, 35 };
218 while (r.fBottom < 256) {
219 surf->getCanvas()->drawRect(r, paint);
220 r.offset(0, r.height() * 2);
221 }
222 }
223
224 // apply the "fade"
225 {
226 SkPoint pts[] = {{SkIntToScalar(subset.left()), 0}, {SkIntToScalar(subset.right()), 0}};
227 SkColor colors[] = {0xFF000000, 0};
228 auto sh = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkShader::kClamp_TileMode);
229 SkPaint paint;
230 paint.setShader(sh);
231 paint.setBlendMode(SkBlendMode::kDstIn);
232 surf->getCanvas()->drawRect(SkRect::Make(subset), paint);
233 }
234
235 // restore the original strip, drawing it "under" the current foreground
236 {
237 SkPaint paint;
238 paint.setBlendMode(SkBlendMode::kDstOver);
239 surf->getCanvas()->drawImage(saveImg,
240 SkIntToScalar(subset.left()), SkIntToScalar(subset.top()),
241 &paint);
242 }
243
244 // show it on screen
245 surf->draw(canvas, 0, 0, nullptr);
246}