blob: a8f74bface68b2098019d0b1feeda23c69306037 [file] [log] [blame]
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +00001/*
2 * Copyright 2013 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"
9#include "SkCanvas.h"
bsalomon093779c2016-03-08 11:53:31 -080010#include "SkColorFilter.h"
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000011#include "SkGradientShader.h"
Florin Malitaf614ba22017-02-16 22:12:41 -050012#include "SkLocalMatrixShader.h"
mike@reedtribe.org2c326b72013-12-30 04:20:38 +000013#include "SkRandom.h"
Brian Salomon199fb872017-02-06 09:41:10 -050014#include "SkVertices.h"
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000015
Brian Salomon3f363692017-02-02 21:05:19 -050016static constexpr SkScalar kShaderSize = 40;
Florin Malitaf614ba22017-02-16 22:12:41 -050017static sk_sp<SkShader> make_shader1(SkScalar shaderScale) {
reed@google.com85e143c2013-12-30 15:51:25 +000018 const SkColor colors[] = {
19 SK_ColorRED, SK_ColorCYAN, SK_ColorGREEN, SK_ColorWHITE,
20 SK_ColorMAGENTA, SK_ColorBLUE, SK_ColorYELLOW,
21 };
Brian Salomon3f363692017-02-02 21:05:19 -050022 const SkPoint pts[] = {{kShaderSize / 4, 0}, {3 * kShaderSize / 4, kShaderSize}};
Florin Malitaf614ba22017-02-16 22:12:41 -050023 const SkMatrix localMatrix = SkMatrix::MakeScale(shaderScale, shaderScale);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000024
Florin Malitaf614ba22017-02-16 22:12:41 -050025 sk_sp<SkShader> grad = SkGradientShader::MakeLinear(pts, colors, nullptr,
26 SK_ARRAY_COUNT(colors),
27 SkShader::kMirror_TileMode, 0,
28 &localMatrix);
29 // Throw in a couple of local matrix wrappers for good measure.
30 return shaderScale == 1
31 ? grad
32 : sk_make_sp<SkLocalMatrixShader>(
33 sk_make_sp<SkLocalMatrixShader>(std::move(grad), SkMatrix::MakeTrans(-10, 0)),
34 SkMatrix::MakeTrans(10, 0));
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000035}
36
reed1a9b9642016-03-13 14:13:58 -070037static sk_sp<SkShader> make_shader2() {
38 return SkShader::MakeColorShader(SK_ColorBLUE);
bsalomon093779c2016-03-08 11:53:31 -080039}
40
reedd053ce92016-03-22 10:17:23 -070041static sk_sp<SkColorFilter> make_color_filter() {
Mike Reed7d954ad2016-10-28 15:42:34 -040042 return SkColorFilter::MakeModeFilter(0xFFAABBCC, SkBlendMode::kDarken);
bsalomon093779c2016-03-08 11:53:31 -080043}
44
Brian Salomon3f363692017-02-02 21:05:19 -050045static constexpr SkScalar kMeshSize = 30;
46
47// start with the center of a 3x3 grid of vertices.
48static constexpr uint16_t kMeshFan[] = {
49 4,
50 0, 1, 2, 5, 8, 7, 6, 3, 0
51};
52
Brian Salomon199fb872017-02-06 09:41:10 -050053static const int kMeshIndexCnt = (int)SK_ARRAY_COUNT(kMeshFan);
Brian Salomon3f363692017-02-02 21:05:19 -050054static const int kMeshVertexCnt = 9;
55
56static void fill_mesh(SkPoint pts[kMeshVertexCnt], SkPoint texs[kMeshVertexCnt],
Florin Malitaf614ba22017-02-16 22:12:41 -050057 SkColor colors[kMeshVertexCnt], SkScalar shaderScale) {
Brian Salomon3f363692017-02-02 21:05:19 -050058 pts[0].set(0, 0);
59 pts[1].set(kMeshSize / 2, 3);
60 pts[2].set(kMeshSize, 0);
61 pts[3].set(3, kMeshSize / 2);
62 pts[4].set(kMeshSize / 2, kMeshSize / 2);
63 pts[5].set(kMeshSize - 3, kMeshSize / 2);
64 pts[6].set(0, kMeshSize);
65 pts[7].set(kMeshSize / 2, kMeshSize - 3);
66 pts[8].set(kMeshSize, kMeshSize);
67
Florin Malitaf614ba22017-02-16 22:12:41 -050068 const auto shaderSize = kShaderSize * shaderScale;
Brian Salomon3f363692017-02-02 21:05:19 -050069 texs[0].set(0, 0);
Florin Malitaf614ba22017-02-16 22:12:41 -050070 texs[1].set(shaderSize / 2, 0);
71 texs[2].set(shaderSize, 0);
72 texs[3].set(0, shaderSize / 2);
73 texs[4].set(shaderSize / 2, shaderSize / 2);
74 texs[5].set(shaderSize, shaderSize / 2);
75 texs[6].set(0, shaderSize);
76 texs[7].set(shaderSize / 2, shaderSize);
77 texs[8].set(shaderSize, shaderSize);
Brian Salomon3f363692017-02-02 21:05:19 -050078
79 SkRandom rand;
80 for (size_t i = 0; i < kMeshVertexCnt; ++i) {
81 colors[i] = rand.nextU() | 0xFF000000;
82 }
83}
84
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000085class VerticesGM : public skiagm::GM {
Brian Salomon3f363692017-02-02 21:05:19 -050086 SkPoint fPts[kMeshVertexCnt];
87 SkPoint fTexs[kMeshVertexCnt];
88 SkColor fColors[kMeshVertexCnt];
reedd053ce92016-03-22 10:17:23 -070089 sk_sp<SkShader> fShader1;
90 sk_sp<SkShader> fShader2;
91 sk_sp<SkColorFilter> fColorFilter;
Brian Salomon199fb872017-02-06 09:41:10 -050092 sk_sp<SkVertices> fVertices;
93 bool fUseObject;
Florin Malitaf614ba22017-02-16 22:12:41 -050094 SkScalar fShaderScale;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000095
96public:
Florin Malitaf614ba22017-02-16 22:12:41 -050097 VerticesGM(bool useObject, SkScalar shaderScale = 1)
98 : fUseObject(useObject), fShaderScale(shaderScale) {}
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000099
100protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000101
mtklein36352bf2015-03-25 18:17:31 -0700102 void onOnceBeforeDraw() override {
Florin Malitaf614ba22017-02-16 22:12:41 -0500103 fill_mesh(fPts, fTexs, fColors, fShaderScale);
104 fShader1 = make_shader1(fShaderScale);
reed1a9b9642016-03-13 14:13:58 -0700105 fShader2 = make_shader2();
reedd053ce92016-03-22 10:17:23 -0700106 fColorFilter = make_color_filter();
Brian Salomon199fb872017-02-06 09:41:10 -0500107 if (fUseObject) {
108 std::unique_ptr<SkPoint[]> points(new SkPoint[kMeshVertexCnt]);
109 std::unique_ptr<SkPoint[]> texs(new SkPoint[kMeshVertexCnt]);
110 std::unique_ptr<SkColor[]> colors(new SkColor[kMeshVertexCnt]);
111 std::unique_ptr<uint16_t[]> indices(new uint16_t[kMeshIndexCnt]);
112 memcpy(points.get(), fPts, sizeof(SkPoint) * kMeshVertexCnt);
113 memcpy(colors.get(), fColors, sizeof(SkColor) * kMeshVertexCnt);
114 memcpy(texs.get(), fTexs, sizeof(SkPoint) * kMeshVertexCnt);
115 memcpy(indices.get(), kMeshFan, sizeof(uint16_t) * kMeshIndexCnt);
Brian Salomon604c9892017-02-06 14:11:51 -0500116 // Older libstdc++ does not allow moving a std::unique_ptr<T[]> into a
117 // std::unique_ptr<const T[]>. Hence the release() calls below.
118 fVertices = SkVertices::MakeIndexed(
119 SkCanvas::kTriangleFan_VertexMode,
Brian Salomon3a970542017-02-06 15:30:06 -0500120 std::unique_ptr<const SkPoint[]>((const SkPoint*)points.release()),
121 std::unique_ptr<const SkColor[]>((const SkColor*)colors.release()),
122 std::unique_ptr<const SkPoint[]>((const SkPoint*)texs.release()),
123 kMeshVertexCnt,
124 std::unique_ptr<const uint16_t[]>((const uint16_t*)indices.release()),
125 kMeshIndexCnt);
Brian Salomon199fb872017-02-06 09:41:10 -0500126 }
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000127 }
128
mtklein36352bf2015-03-25 18:17:31 -0700129 SkString onShortName() override {
reed@google.com60da8f32014-05-05 20:41:21 +0000130 SkString name("vertices");
Brian Salomon199fb872017-02-06 09:41:10 -0500131 if (fUseObject) {
132 name.append("_object");
133 }
Florin Malitaf614ba22017-02-16 22:12:41 -0500134 if (fShaderScale != 1) {
135 name.append("_scaled_shader");
136 }
reed@google.com60da8f32014-05-05 20:41:21 +0000137 return name;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000138 }
139
mtklein36352bf2015-03-25 18:17:31 -0700140 SkISize onISize() override {
Brian Salomon32161802017-02-10 10:48:15 -0500141 return SkISize::Make(975, 1175);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000142 }
143
mtklein36352bf2015-03-25 18:17:31 -0700144 void onDraw(SkCanvas* canvas) override {
Mike Reed7d954ad2016-10-28 15:42:34 -0400145 const SkBlendMode modes[] = {
146 SkBlendMode::kClear,
147 SkBlendMode::kSrc,
148 SkBlendMode::kDst,
149 SkBlendMode::kSrcOver,
150 SkBlendMode::kDstOver,
151 SkBlendMode::kSrcIn,
152 SkBlendMode::kDstIn,
153 SkBlendMode::kSrcOut,
154 SkBlendMode::kDstOut,
155 SkBlendMode::kSrcATop,
156 SkBlendMode::kDstATop,
157 SkBlendMode::kXor,
158 SkBlendMode::kPlus,
159 SkBlendMode::kModulate,
160 SkBlendMode::kScreen,
161 SkBlendMode::kOverlay,
162 SkBlendMode::kDarken,
163 SkBlendMode::kLighten,
164 SkBlendMode::kColorDodge,
165 SkBlendMode::kColorBurn,
166 SkBlendMode::kHardLight,
167 SkBlendMode::kSoftLight,
168 SkBlendMode::kDifference,
169 SkBlendMode::kExclusion,
170 SkBlendMode::kMultiply,
171 SkBlendMode::kHue,
172 SkBlendMode::kSaturation,
173 SkBlendMode::kColor,
174 SkBlendMode::kLuminosity,
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000175 };
skia.committer@gmail.com4c912862013-12-30 07:01:37 +0000176
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000177 SkPaint paint;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000178
bsalomon093779c2016-03-08 11:53:31 -0800179 canvas->translate(4, 4);
180 int x = 0;
Brian Salomon199fb872017-02-06 09:41:10 -0500181 for (auto mode : modes) {
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000182 canvas->save();
Brian Salomon199fb872017-02-06 09:41:10 -0500183 for (uint8_t alpha : {0xFF, 0x80}) {
184 for (const auto& cf : {sk_sp<SkColorFilter>(nullptr), fColorFilter}) {
185 for (const auto& shader : {fShader1, fShader2}) {
186 static constexpr struct {
187 bool fHasColors;
188 bool fHasTexs;
189 } kAttrs[] = {{true, false}, {false, true}, {true, true}};
190 for (auto attrs : kAttrs) {
191 paint.setShader(shader);
192 paint.setColorFilter(cf);
193 paint.setAlpha(alpha);
194 if (fUseObject) {
195 uint32_t flags = 0;
196 flags |=
197 attrs.fHasColors ? 0 : SkCanvas::kIgnoreColors_VerticesFlag;
198 flags |= attrs.fHasTexs ? 0
199 : SkCanvas::kIgnoreTexCoords_VerticesFlag;
200 canvas->drawVertices(fVertices, mode, paint, flags);
201 } else {
202 const SkColor* colors = attrs.fHasColors ? fColors : nullptr;
203 const SkPoint* texs = attrs.fHasTexs ? fTexs : nullptr;
204 canvas->drawVertices(SkCanvas::kTriangleFan_VertexMode,
205 kMeshVertexCnt, fPts, texs, colors, mode,
206 kMeshFan, kMeshIndexCnt, paint);
207 }
208 canvas->translate(40, 0);
209 ++x;
210 }
211 }
212 }
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000213 }
214 canvas->restore();
bsalomon093779c2016-03-08 11:53:31 -0800215 canvas->translate(0, 40);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000216 }
217 }
218
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000219private:
220 typedef skiagm::GM INHERITED;
221};
222
reed@google.com60da8f32014-05-05 20:41:21 +0000223/////////////////////////////////////////////////////////////////////////////////////
224
Brian Salomon199fb872017-02-06 09:41:10 -0500225DEF_GM(return new VerticesGM(true);)
226DEF_GM(return new VerticesGM(false);)
Florin Malitaf614ba22017-02-16 22:12:41 -0500227DEF_GM(return new VerticesGM(false, 1 / kShaderSize);)
Brian Salomon3f363692017-02-02 21:05:19 -0500228
Brian Salomon199fb872017-02-06 09:41:10 -0500229static void draw_batching(SkCanvas* canvas, bool useObject) {
230 std::unique_ptr<SkPoint[]> pts(new SkPoint[kMeshVertexCnt]);
231 std::unique_ptr<SkPoint[]> texs(new SkPoint[kMeshVertexCnt]);
232 std::unique_ptr<SkColor[]> colors(new SkColor[kMeshVertexCnt]);
Florin Malitaf614ba22017-02-16 22:12:41 -0500233 fill_mesh(pts.get(), texs.get(), colors.get(), 1);
Brian Salomon199fb872017-02-06 09:41:10 -0500234
Brian Salomon3f363692017-02-02 21:05:19 -0500235 SkTDArray<SkMatrix> matrices;
236 matrices.push()->reset();
237 matrices.push()->setTranslate(0, 40);
238 SkMatrix* m = matrices.push();
239 m->setRotate(45, kMeshSize / 2, kMeshSize / 2);
240 m->postScale(1.2f, .8f, kMeshSize / 2, kMeshSize / 2);
241 m->postTranslate(0, 80);
242
Florin Malitaf614ba22017-02-16 22:12:41 -0500243 auto shader = make_shader1(1);
Brian Salomon3f363692017-02-02 21:05:19 -0500244
245 // Triangle fans can't batch so we convert to regular triangles,
Brian Salomon199fb872017-02-06 09:41:10 -0500246 static constexpr int kNumTris = kMeshIndexCnt - 2;
247 std::unique_ptr<uint16_t[]> indices(new uint16_t[3 * kNumTris]);
Brian Salomon3f363692017-02-02 21:05:19 -0500248 for (size_t i = 0; i < kNumTris; ++i) {
249 indices[3 * i] = kMeshFan[0];
250 indices[3 * i + 1] = kMeshFan[i + 1];
251 indices[3 * i + 2] = kMeshFan[i + 2];
252 }
Brian Salomon199fb872017-02-06 09:41:10 -0500253
254 sk_sp<SkVertices> vertices;
255 if (useObject) {
Brian Salomon604c9892017-02-06 14:11:51 -0500256 // Older libstdc++ does not allow moving a std::unique_ptr<T[]> into a
257 // std::unique_ptr<const T[]>. Hence the release() calls below.
258 vertices = SkVertices::MakeIndexed(
Brian Salomon3a970542017-02-06 15:30:06 -0500259 SkCanvas::kTriangles_VertexMode,
260 std::unique_ptr<const SkPoint[]>((const SkPoint*)pts.release()),
261 std::unique_ptr<const SkColor[]>((const SkColor*)colors.release()),
262 std::unique_ptr<const SkPoint[]>((const SkPoint*)texs.release()), kMeshVertexCnt,
263 std::unique_ptr<const uint16_t[]>((const uint16_t*)indices.release()),
264 3 * kNumTris);
Brian Salomon199fb872017-02-06 09:41:10 -0500265 }
266 canvas->save();
Brian Salomon3f363692017-02-02 21:05:19 -0500267 canvas->translate(10, 10);
268 for (bool useShader : {false, true}) {
269 for (bool useTex : {false, true}) {
270 for (const auto& m : matrices) {
271 canvas->save();
272 canvas->concat(m);
273 SkPaint paint;
Brian Salomon3f363692017-02-02 21:05:19 -0500274 paint.setShader(useShader ? shader : nullptr);
Brian Salomon199fb872017-02-06 09:41:10 -0500275 if (useObject) {
276 uint32_t flags = useTex ? 0 : SkCanvas::kIgnoreTexCoords_VerticesFlag;
277 canvas->drawVertices(vertices, SkBlendMode::kModulate, paint, flags);
278 } else {
279 const SkPoint* t = useTex ? texs.get() : nullptr;
280 canvas->drawVertices(SkCanvas::kTriangles_VertexMode, kMeshVertexCnt, pts.get(),
281 t, colors.get(), indices.get(), kNumTris * 3, paint);
282 }
Brian Salomon3f363692017-02-02 21:05:19 -0500283 canvas->restore();
284 }
285 canvas->translate(0, 120);
286 }
287 }
Brian Salomon199fb872017-02-06 09:41:10 -0500288 canvas->restore();
289}
290
291// This test exists to exercise batching in the gpu backend.
292DEF_SIMPLE_GM(vertices_batching, canvas, 100, 500) {
293 draw_batching(canvas, false);
294 canvas->translate(50, 0);
295 draw_batching(canvas, true);
Brian Salomon3f363692017-02-02 21:05:19 -0500296}