blob: 5fdce92a8eaa8223422252f0290a7062c49f1b72 [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"
mike@reedtribe.org2c326b72013-12-30 04:20:38 +000012#include "SkRandom.h"
Brian Salomon199fb872017-02-06 09:41:10 -050013#include "SkVertices.h"
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000014
Brian Salomon3f363692017-02-02 21:05:19 -050015static constexpr SkScalar kShaderSize = 40;
16static sk_sp<SkShader> make_shader1() {
reed@google.com85e143c2013-12-30 15:51:25 +000017 const SkColor colors[] = {
18 SK_ColorRED, SK_ColorCYAN, SK_ColorGREEN, SK_ColorWHITE,
19 SK_ColorMAGENTA, SK_ColorBLUE, SK_ColorYELLOW,
20 };
Brian Salomon3f363692017-02-02 21:05:19 -050021 const SkPoint pts[] = {{kShaderSize / 4, 0}, {3 * kShaderSize / 4, kShaderSize}};
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000022
reed1a9b9642016-03-13 14:13:58 -070023 return SkGradientShader::MakeLinear(pts, colors, nullptr, SK_ARRAY_COUNT(colors),
24 SkShader::kMirror_TileMode);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000025}
26
reed1a9b9642016-03-13 14:13:58 -070027static sk_sp<SkShader> make_shader2() {
28 return SkShader::MakeColorShader(SK_ColorBLUE);
bsalomon093779c2016-03-08 11:53:31 -080029}
30
reedd053ce92016-03-22 10:17:23 -070031static sk_sp<SkColorFilter> make_color_filter() {
Mike Reed7d954ad2016-10-28 15:42:34 -040032 return SkColorFilter::MakeModeFilter(0xFFAABBCC, SkBlendMode::kDarken);
bsalomon093779c2016-03-08 11:53:31 -080033}
34
Brian Salomon3f363692017-02-02 21:05:19 -050035static constexpr SkScalar kMeshSize = 30;
36
37// start with the center of a 3x3 grid of vertices.
38static constexpr uint16_t kMeshFan[] = {
39 4,
40 0, 1, 2, 5, 8, 7, 6, 3, 0
41};
42
Brian Salomon199fb872017-02-06 09:41:10 -050043static const int kMeshIndexCnt = (int)SK_ARRAY_COUNT(kMeshFan);
Brian Salomon3f363692017-02-02 21:05:19 -050044static const int kMeshVertexCnt = 9;
45
46static void fill_mesh(SkPoint pts[kMeshVertexCnt], SkPoint texs[kMeshVertexCnt],
47 SkColor colors[kMeshVertexCnt]) {
48 pts[0].set(0, 0);
49 pts[1].set(kMeshSize / 2, 3);
50 pts[2].set(kMeshSize, 0);
51 pts[3].set(3, kMeshSize / 2);
52 pts[4].set(kMeshSize / 2, kMeshSize / 2);
53 pts[5].set(kMeshSize - 3, kMeshSize / 2);
54 pts[6].set(0, kMeshSize);
55 pts[7].set(kMeshSize / 2, kMeshSize - 3);
56 pts[8].set(kMeshSize, kMeshSize);
57
58 texs[0].set(0, 0);
59 texs[1].set(kShaderSize / 2, 0);
60 texs[2].set(kShaderSize, 0);
61 texs[3].set(0, kShaderSize / 2);
62 texs[4].set(kShaderSize / 2, kShaderSize / 2);
63 texs[5].set(kShaderSize, kShaderSize / 2);
64 texs[6].set(0, kShaderSize);
65 texs[7].set(kShaderSize / 2, kShaderSize);
66 texs[8].set(kShaderSize, kShaderSize);
67
68 SkRandom rand;
69 for (size_t i = 0; i < kMeshVertexCnt; ++i) {
70 colors[i] = rand.nextU() | 0xFF000000;
71 }
72}
73
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000074class VerticesGM : public skiagm::GM {
Brian Salomon3f363692017-02-02 21:05:19 -050075 SkPoint fPts[kMeshVertexCnt];
76 SkPoint fTexs[kMeshVertexCnt];
77 SkColor fColors[kMeshVertexCnt];
reedd053ce92016-03-22 10:17:23 -070078 sk_sp<SkShader> fShader1;
79 sk_sp<SkShader> fShader2;
80 sk_sp<SkColorFilter> fColorFilter;
Brian Salomon199fb872017-02-06 09:41:10 -050081 sk_sp<SkVertices> fVertices;
82 bool fUseObject;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000083
84public:
Brian Salomon199fb872017-02-06 09:41:10 -050085 VerticesGM(bool useObject) : fUseObject(useObject) {}
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000086
87protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +000088
mtklein36352bf2015-03-25 18:17:31 -070089 void onOnceBeforeDraw() override {
Brian Salomon3f363692017-02-02 21:05:19 -050090 fill_mesh(fPts, fTexs, fColors);
91 fShader1 = make_shader1();
reed1a9b9642016-03-13 14:13:58 -070092 fShader2 = make_shader2();
reedd053ce92016-03-22 10:17:23 -070093 fColorFilter = make_color_filter();
Brian Salomon199fb872017-02-06 09:41:10 -050094 if (fUseObject) {
95 std::unique_ptr<SkPoint[]> points(new SkPoint[kMeshVertexCnt]);
96 std::unique_ptr<SkPoint[]> texs(new SkPoint[kMeshVertexCnt]);
97 std::unique_ptr<SkColor[]> colors(new SkColor[kMeshVertexCnt]);
98 std::unique_ptr<uint16_t[]> indices(new uint16_t[kMeshIndexCnt]);
99 memcpy(points.get(), fPts, sizeof(SkPoint) * kMeshVertexCnt);
100 memcpy(colors.get(), fColors, sizeof(SkColor) * kMeshVertexCnt);
101 memcpy(texs.get(), fTexs, sizeof(SkPoint) * kMeshVertexCnt);
102 memcpy(indices.get(), kMeshFan, sizeof(uint16_t) * kMeshIndexCnt);
103 fVertices = SkVertices::MakeIndexed(SkCanvas::kTriangleFan_VertexMode,
104 std::move(points),
105 std::move(colors), std::move(texs), kMeshVertexCnt,
106 std::move(indices), kMeshIndexCnt);
107 }
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000108 }
109
mtklein36352bf2015-03-25 18:17:31 -0700110 SkString onShortName() override {
reed@google.com60da8f32014-05-05 20:41:21 +0000111 SkString name("vertices");
Brian Salomon199fb872017-02-06 09:41:10 -0500112 if (fUseObject) {
113 name.append("_object");
114 }
reed@google.com60da8f32014-05-05 20:41:21 +0000115 return name;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000116 }
117
mtklein36352bf2015-03-25 18:17:31 -0700118 SkISize onISize() override {
reed@google.com85e143c2013-12-30 15:51:25 +0000119 return SkISize::Make(600, 600);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000120 }
121
mtklein36352bf2015-03-25 18:17:31 -0700122 void onDraw(SkCanvas* canvas) override {
Mike Reed7d954ad2016-10-28 15:42:34 -0400123 const SkBlendMode modes[] = {
124 SkBlendMode::kClear,
125 SkBlendMode::kSrc,
126 SkBlendMode::kDst,
127 SkBlendMode::kSrcOver,
128 SkBlendMode::kDstOver,
129 SkBlendMode::kSrcIn,
130 SkBlendMode::kDstIn,
131 SkBlendMode::kSrcOut,
132 SkBlendMode::kDstOut,
133 SkBlendMode::kSrcATop,
134 SkBlendMode::kDstATop,
135 SkBlendMode::kXor,
136 SkBlendMode::kPlus,
137 SkBlendMode::kModulate,
138 SkBlendMode::kScreen,
139 SkBlendMode::kOverlay,
140 SkBlendMode::kDarken,
141 SkBlendMode::kLighten,
142 SkBlendMode::kColorDodge,
143 SkBlendMode::kColorBurn,
144 SkBlendMode::kHardLight,
145 SkBlendMode::kSoftLight,
146 SkBlendMode::kDifference,
147 SkBlendMode::kExclusion,
148 SkBlendMode::kMultiply,
149 SkBlendMode::kHue,
150 SkBlendMode::kSaturation,
151 SkBlendMode::kColor,
152 SkBlendMode::kLuminosity,
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000153 };
skia.committer@gmail.com4c912862013-12-30 07:01:37 +0000154
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000155 SkPaint paint;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000156
bsalomon093779c2016-03-08 11:53:31 -0800157 canvas->translate(4, 4);
158 int x = 0;
Brian Salomon199fb872017-02-06 09:41:10 -0500159 for (auto mode : modes) {
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000160 canvas->save();
Brian Salomon199fb872017-02-06 09:41:10 -0500161 for (uint8_t alpha : {0xFF, 0x80}) {
162 for (const auto& cf : {sk_sp<SkColorFilter>(nullptr), fColorFilter}) {
163 for (const auto& shader : {fShader1, fShader2}) {
164 static constexpr struct {
165 bool fHasColors;
166 bool fHasTexs;
167 } kAttrs[] = {{true, false}, {false, true}, {true, true}};
168 for (auto attrs : kAttrs) {
169 paint.setShader(shader);
170 paint.setColorFilter(cf);
171 paint.setAlpha(alpha);
172 if (fUseObject) {
173 uint32_t flags = 0;
174 flags |=
175 attrs.fHasColors ? 0 : SkCanvas::kIgnoreColors_VerticesFlag;
176 flags |= attrs.fHasTexs ? 0
177 : SkCanvas::kIgnoreTexCoords_VerticesFlag;
178 canvas->drawVertices(fVertices, mode, paint, flags);
179 } else {
180 const SkColor* colors = attrs.fHasColors ? fColors : nullptr;
181 const SkPoint* texs = attrs.fHasTexs ? fTexs : nullptr;
182 canvas->drawVertices(SkCanvas::kTriangleFan_VertexMode,
183 kMeshVertexCnt, fPts, texs, colors, mode,
184 kMeshFan, kMeshIndexCnt, paint);
185 }
186 canvas->translate(40, 0);
187 ++x;
188 }
189 }
190 }
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000191 }
192 canvas->restore();
bsalomon093779c2016-03-08 11:53:31 -0800193 canvas->translate(0, 40);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000194 }
195 }
196
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000197private:
198 typedef skiagm::GM INHERITED;
199};
200
reed@google.com60da8f32014-05-05 20:41:21 +0000201/////////////////////////////////////////////////////////////////////////////////////
202
Brian Salomon199fb872017-02-06 09:41:10 -0500203DEF_GM(return new VerticesGM(true);)
204DEF_GM(return new VerticesGM(false);)
Brian Salomon3f363692017-02-02 21:05:19 -0500205
Brian Salomon199fb872017-02-06 09:41:10 -0500206static void draw_batching(SkCanvas* canvas, bool useObject) {
207 std::unique_ptr<SkPoint[]> pts(new SkPoint[kMeshVertexCnt]);
208 std::unique_ptr<SkPoint[]> texs(new SkPoint[kMeshVertexCnt]);
209 std::unique_ptr<SkColor[]> colors(new SkColor[kMeshVertexCnt]);
210 fill_mesh(pts.get(), texs.get(), colors.get());
211
Brian Salomon3f363692017-02-02 21:05:19 -0500212 SkTDArray<SkMatrix> matrices;
213 matrices.push()->reset();
214 matrices.push()->setTranslate(0, 40);
215 SkMatrix* m = matrices.push();
216 m->setRotate(45, kMeshSize / 2, kMeshSize / 2);
217 m->postScale(1.2f, .8f, kMeshSize / 2, kMeshSize / 2);
218 m->postTranslate(0, 80);
219
220 auto shader = make_shader1();
221
222 // Triangle fans can't batch so we convert to regular triangles,
Brian Salomon199fb872017-02-06 09:41:10 -0500223 static constexpr int kNumTris = kMeshIndexCnt - 2;
224 std::unique_ptr<uint16_t[]> indices(new uint16_t[3 * kNumTris]);
Brian Salomon3f363692017-02-02 21:05:19 -0500225 for (size_t i = 0; i < kNumTris; ++i) {
226 indices[3 * i] = kMeshFan[0];
227 indices[3 * i + 1] = kMeshFan[i + 1];
228 indices[3 * i + 2] = kMeshFan[i + 2];
229 }
Brian Salomon199fb872017-02-06 09:41:10 -0500230
231 sk_sp<SkVertices> vertices;
232 if (useObject) {
233 vertices =
234 SkVertices::MakeIndexed(SkCanvas::kTriangles_VertexMode, std::move(pts),
235 std::move(colors),
236 std::move(texs), kMeshVertexCnt, std::move(indices),
237 3 * kNumTris);
238 }
239 canvas->save();
Brian Salomon3f363692017-02-02 21:05:19 -0500240 canvas->translate(10, 10);
241 for (bool useShader : {false, true}) {
242 for (bool useTex : {false, true}) {
243 for (const auto& m : matrices) {
244 canvas->save();
245 canvas->concat(m);
246 SkPaint paint;
Brian Salomon3f363692017-02-02 21:05:19 -0500247 paint.setShader(useShader ? shader : nullptr);
Brian Salomon199fb872017-02-06 09:41:10 -0500248 if (useObject) {
249 uint32_t flags = useTex ? 0 : SkCanvas::kIgnoreTexCoords_VerticesFlag;
250 canvas->drawVertices(vertices, SkBlendMode::kModulate, paint, flags);
251 } else {
252 const SkPoint* t = useTex ? texs.get() : nullptr;
253 canvas->drawVertices(SkCanvas::kTriangles_VertexMode, kMeshVertexCnt, pts.get(),
254 t, colors.get(), indices.get(), kNumTris * 3, paint);
255 }
Brian Salomon3f363692017-02-02 21:05:19 -0500256 canvas->restore();
257 }
258 canvas->translate(0, 120);
259 }
260 }
Brian Salomon199fb872017-02-06 09:41:10 -0500261 canvas->restore();
262}
263
264// This test exists to exercise batching in the gpu backend.
265DEF_SIMPLE_GM(vertices_batching, canvas, 100, 500) {
266 draw_batching(canvas, false);
267 canvas->translate(50, 0);
268 draw_batching(canvas, true);
Brian Salomon3f363692017-02-02 21:05:19 -0500269}