blob: 866d046057437ee4da328bbb3c32f81e6f101da8 [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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -04009#include "include/core/SkBlendMode.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "include/core/SkCanvas.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040011#include "include/core/SkColor.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkColorFilter.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040013#include "include/core/SkMatrix.h"
14#include "include/core/SkPaint.h"
15#include "include/core/SkPoint.h"
16#include "include/core/SkRefCnt.h"
17#include "include/core/SkScalar.h"
18#include "include/core/SkShader.h"
19#include "include/core/SkSize.h"
20#include "include/core/SkString.h"
21#include "include/core/SkTileMode.h"
22#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "include/core/SkVertices.h"
24#include "include/effects/SkGradientShader.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040025#include "include/private/SkTDArray.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050026#include "include/utils/SkRandom.h"
27#include "src/shaders/SkLocalMatrixShader.h"
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000028
Ben Wagner7fde8e12019-05-01 17:28:53 -040029#include <initializer_list>
30#include <utility>
31
Brian Salomon3f363692017-02-02 21:05:19 -050032static constexpr SkScalar kShaderSize = 40;
Florin Malitaf614ba22017-02-16 22:12:41 -050033static sk_sp<SkShader> make_shader1(SkScalar shaderScale) {
reed@google.com85e143c2013-12-30 15:51:25 +000034 const SkColor colors[] = {
35 SK_ColorRED, SK_ColorCYAN, SK_ColorGREEN, SK_ColorWHITE,
36 SK_ColorMAGENTA, SK_ColorBLUE, SK_ColorYELLOW,
37 };
Brian Salomon3f363692017-02-02 21:05:19 -050038 const SkPoint pts[] = {{kShaderSize / 4, 0}, {3 * kShaderSize / 4, kShaderSize}};
Florin Malitaf614ba22017-02-16 22:12:41 -050039 const SkMatrix localMatrix = SkMatrix::MakeScale(shaderScale, shaderScale);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000040
Florin Malitaf614ba22017-02-16 22:12:41 -050041 sk_sp<SkShader> grad = SkGradientShader::MakeLinear(pts, colors, nullptr,
42 SK_ARRAY_COUNT(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -040043 SkTileMode::kMirror, 0,
Florin Malitaf614ba22017-02-16 22:12:41 -050044 &localMatrix);
45 // Throw in a couple of local matrix wrappers for good measure.
46 return shaderScale == 1
47 ? grad
48 : sk_make_sp<SkLocalMatrixShader>(
49 sk_make_sp<SkLocalMatrixShader>(std::move(grad), SkMatrix::MakeTrans(-10, 0)),
50 SkMatrix::MakeTrans(10, 0));
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +000051}
52
reed1a9b9642016-03-13 14:13:58 -070053static sk_sp<SkShader> make_shader2() {
Mike Reedc8bea7d2019-04-09 13:55:36 -040054 return SkShaders::Color(SK_ColorBLUE);
bsalomon093779c2016-03-08 11:53:31 -080055}
56
reedd053ce92016-03-22 10:17:23 -070057static sk_sp<SkColorFilter> make_color_filter() {
Mike Reedb286bc22019-04-08 16:23:20 -040058 return SkColorFilters::Blend(0xFFAABBCC, SkBlendMode::kDarken);
bsalomon093779c2016-03-08 11:53:31 -080059}
60
Brian Salomon3f363692017-02-02 21:05:19 -050061static constexpr SkScalar kMeshSize = 30;
62
63// start with the center of a 3x3 grid of vertices.
64static constexpr uint16_t kMeshFan[] = {
65 4,
66 0, 1, 2, 5, 8, 7, 6, 3, 0
67};
68
Brian Salomon199fb872017-02-06 09:41:10 -050069static const int kMeshIndexCnt = (int)SK_ARRAY_COUNT(kMeshFan);
Brian Salomon3f363692017-02-02 21:05:19 -050070static const int kMeshVertexCnt = 9;
71
72static void fill_mesh(SkPoint pts[kMeshVertexCnt], SkPoint texs[kMeshVertexCnt],
Florin Malitaf614ba22017-02-16 22:12:41 -050073 SkColor colors[kMeshVertexCnt], SkScalar shaderScale) {
Brian Salomon3f363692017-02-02 21:05:19 -050074 pts[0].set(0, 0);
75 pts[1].set(kMeshSize / 2, 3);
76 pts[2].set(kMeshSize, 0);
77 pts[3].set(3, kMeshSize / 2);
78 pts[4].set(kMeshSize / 2, kMeshSize / 2);
79 pts[5].set(kMeshSize - 3, kMeshSize / 2);
80 pts[6].set(0, kMeshSize);
81 pts[7].set(kMeshSize / 2, kMeshSize - 3);
82 pts[8].set(kMeshSize, kMeshSize);
83
Florin Malitaf614ba22017-02-16 22:12:41 -050084 const auto shaderSize = kShaderSize * shaderScale;
Brian Salomon3f363692017-02-02 21:05:19 -050085 texs[0].set(0, 0);
Florin Malitaf614ba22017-02-16 22:12:41 -050086 texs[1].set(shaderSize / 2, 0);
87 texs[2].set(shaderSize, 0);
88 texs[3].set(0, shaderSize / 2);
89 texs[4].set(shaderSize / 2, shaderSize / 2);
90 texs[5].set(shaderSize, shaderSize / 2);
91 texs[6].set(0, shaderSize);
92 texs[7].set(shaderSize / 2, shaderSize);
93 texs[8].set(shaderSize, shaderSize);
Brian Salomon3f363692017-02-02 21:05:19 -050094
95 SkRandom rand;
96 for (size_t i = 0; i < kMeshVertexCnt; ++i) {
97 colors[i] = rand.nextU() | 0xFF000000;
98 }
99}
100
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000101class VerticesGM : public skiagm::GM {
Brian Salomon3f363692017-02-02 21:05:19 -0500102 SkPoint fPts[kMeshVertexCnt];
103 SkPoint fTexs[kMeshVertexCnt];
104 SkColor fColors[kMeshVertexCnt];
reedd053ce92016-03-22 10:17:23 -0700105 sk_sp<SkShader> fShader1;
106 sk_sp<SkShader> fShader2;
107 sk_sp<SkColorFilter> fColorFilter;
Florin Malitaf614ba22017-02-16 22:12:41 -0500108 SkScalar fShaderScale;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000109
110public:
Mike Reed887cdf12017-04-03 11:11:09 -0400111 VerticesGM(SkScalar shaderScale) : fShaderScale(shaderScale) {}
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000112
113protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000114
mtklein36352bf2015-03-25 18:17:31 -0700115 void onOnceBeforeDraw() override {
Florin Malitaf614ba22017-02-16 22:12:41 -0500116 fill_mesh(fPts, fTexs, fColors, fShaderScale);
117 fShader1 = make_shader1(fShaderScale);
reed1a9b9642016-03-13 14:13:58 -0700118 fShader2 = make_shader2();
reedd053ce92016-03-22 10:17:23 -0700119 fColorFilter = make_color_filter();
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000120 }
121
mtklein36352bf2015-03-25 18:17:31 -0700122 SkString onShortName() override {
reed@google.com60da8f32014-05-05 20:41:21 +0000123 SkString name("vertices");
Florin Malitaf614ba22017-02-16 22:12:41 -0500124 if (fShaderScale != 1) {
125 name.append("_scaled_shader");
126 }
reed@google.com60da8f32014-05-05 20:41:21 +0000127 return name;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000128 }
129
mtklein36352bf2015-03-25 18:17:31 -0700130 SkISize onISize() override {
Brian Salomon32161802017-02-10 10:48:15 -0500131 return SkISize::Make(975, 1175);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000132 }
133
mtklein36352bf2015-03-25 18:17:31 -0700134 void onDraw(SkCanvas* canvas) override {
Mike Reed7d954ad2016-10-28 15:42:34 -0400135 const SkBlendMode modes[] = {
136 SkBlendMode::kClear,
137 SkBlendMode::kSrc,
138 SkBlendMode::kDst,
139 SkBlendMode::kSrcOver,
140 SkBlendMode::kDstOver,
141 SkBlendMode::kSrcIn,
142 SkBlendMode::kDstIn,
143 SkBlendMode::kSrcOut,
144 SkBlendMode::kDstOut,
145 SkBlendMode::kSrcATop,
146 SkBlendMode::kDstATop,
147 SkBlendMode::kXor,
148 SkBlendMode::kPlus,
149 SkBlendMode::kModulate,
150 SkBlendMode::kScreen,
151 SkBlendMode::kOverlay,
152 SkBlendMode::kDarken,
153 SkBlendMode::kLighten,
154 SkBlendMode::kColorDodge,
155 SkBlendMode::kColorBurn,
156 SkBlendMode::kHardLight,
157 SkBlendMode::kSoftLight,
158 SkBlendMode::kDifference,
159 SkBlendMode::kExclusion,
160 SkBlendMode::kMultiply,
161 SkBlendMode::kHue,
162 SkBlendMode::kSaturation,
163 SkBlendMode::kColor,
164 SkBlendMode::kLuminosity,
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000165 };
skia.committer@gmail.com4c912862013-12-30 07:01:37 +0000166
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000167 SkPaint paint;
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000168
bsalomon093779c2016-03-08 11:53:31 -0800169 canvas->translate(4, 4);
170 int x = 0;
Brian Salomon199fb872017-02-06 09:41:10 -0500171 for (auto mode : modes) {
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000172 canvas->save();
Mike Reed9407e242019-02-15 16:13:57 -0500173 for (float alpha : {1.0f, 0.5f}) {
Brian Salomon199fb872017-02-06 09:41:10 -0500174 for (const auto& cf : {sk_sp<SkColorFilter>(nullptr), fColorFilter}) {
175 for (const auto& shader : {fShader1, fShader2}) {
176 static constexpr struct {
177 bool fHasColors;
178 bool fHasTexs;
179 } kAttrs[] = {{true, false}, {false, true}, {true, true}};
180 for (auto attrs : kAttrs) {
181 paint.setShader(shader);
182 paint.setColorFilter(cf);
Mike Reed9407e242019-02-15 16:13:57 -0500183 paint.setAlphaf(alpha);
Mike Reed5fa66452017-03-16 09:06:34 -0400184
185 const SkColor* colors = attrs.fHasColors ? fColors : nullptr;
186 const SkPoint* texs = attrs.fHasTexs ? fTexs : nullptr;
Mike Reed887cdf12017-04-03 11:11:09 -0400187 auto v = SkVertices::MakeCopy(SkVertices::kTriangleFan_VertexMode,
188 kMeshVertexCnt, fPts, texs, colors,
189 kMeshIndexCnt, kMeshFan);
190 canvas->drawVertices(v, mode, paint);
Brian Salomon199fb872017-02-06 09:41:10 -0500191 canvas->translate(40, 0);
192 ++x;
193 }
194 }
195 }
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000196 }
197 canvas->restore();
bsalomon093779c2016-03-08 11:53:31 -0800198 canvas->translate(0, 40);
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000199 }
200 }
201
mike@reedtribe.org0c87ea82013-12-30 04:07:34 +0000202private:
203 typedef skiagm::GM INHERITED;
204};
205
reed@google.com60da8f32014-05-05 20:41:21 +0000206/////////////////////////////////////////////////////////////////////////////////////
207
Mike Reed887cdf12017-04-03 11:11:09 -0400208DEF_GM(return new VerticesGM(1);)
209DEF_GM(return new VerticesGM(1 / kShaderSize);)
Brian Salomon3f363692017-02-02 21:05:19 -0500210
Mike Reed887cdf12017-04-03 11:11:09 -0400211static void draw_batching(SkCanvas* canvas) {
Mike Reed97eb4fe2017-03-14 12:04:16 -0400212 // Triangle fans can't batch so we convert to regular triangles,
213 static constexpr int kNumTris = kMeshIndexCnt - 2;
Mike Reed887cdf12017-04-03 11:11:09 -0400214 SkVertices::Builder builder(SkVertices::kTriangles_VertexMode, kMeshVertexCnt, 3 * kNumTris,
Mike Reedaa9e3322017-03-16 14:38:48 -0400215 SkVertices::kHasColors_BuilderFlag |
216 SkVertices::kHasTexCoords_BuilderFlag);
Mike Reed97eb4fe2017-03-14 12:04:16 -0400217
218 SkPoint* pts = builder.positions();
219 SkPoint* texs = builder.texCoords();
220 SkColor* colors = builder.colors();
221 fill_mesh(pts, texs, colors, 1);
Brian Salomon199fb872017-02-06 09:41:10 -0500222
Brian Salomon3f363692017-02-02 21:05:19 -0500223 SkTDArray<SkMatrix> matrices;
224 matrices.push()->reset();
225 matrices.push()->setTranslate(0, 40);
226 SkMatrix* m = matrices.push();
227 m->setRotate(45, kMeshSize / 2, kMeshSize / 2);
228 m->postScale(1.2f, .8f, kMeshSize / 2, kMeshSize / 2);
229 m->postTranslate(0, 80);
230
Florin Malitaf614ba22017-02-16 22:12:41 -0500231 auto shader = make_shader1(1);
Brian Salomon3f363692017-02-02 21:05:19 -0500232
Mike Reed97eb4fe2017-03-14 12:04:16 -0400233 uint16_t* indices = builder.indices();
Brian Salomon3f363692017-02-02 21:05:19 -0500234 for (size_t i = 0; i < kNumTris; ++i) {
235 indices[3 * i] = kMeshFan[0];
236 indices[3 * i + 1] = kMeshFan[i + 1];
237 indices[3 * i + 2] = kMeshFan[i + 2];
Mike Reed97eb4fe2017-03-14 12:04:16 -0400238
Brian Salomon3f363692017-02-02 21:05:19 -0500239 }
Brian Salomon199fb872017-02-06 09:41:10 -0500240
Brian Salomon199fb872017-02-06 09:41:10 -0500241 canvas->save();
Brian Salomon3f363692017-02-02 21:05:19 -0500242 canvas->translate(10, 10);
243 for (bool useShader : {false, true}) {
244 for (bool useTex : {false, true}) {
245 for (const auto& m : matrices) {
246 canvas->save();
247 canvas->concat(m);
248 SkPaint paint;
Brian Salomon3f363692017-02-02 21:05:19 -0500249 paint.setShader(useShader ? shader : nullptr);
Mike Reed5fa66452017-03-16 09:06:34 -0400250
251 const SkPoint* t = useTex ? texs : nullptr;
Mike Reed887cdf12017-04-03 11:11:09 -0400252 auto v = SkVertices::MakeCopy(SkVertices::kTriangles_VertexMode, kMeshVertexCnt,
253 pts, t, colors, kNumTris * 3, indices);
254 canvas->drawVertices(v, SkBlendMode::kModulate, paint);
Brian Salomon3f363692017-02-02 21:05:19 -0500255 canvas->restore();
256 }
257 canvas->translate(0, 120);
258 }
259 }
Brian Salomon199fb872017-02-06 09:41:10 -0500260 canvas->restore();
261}
262
263// This test exists to exercise batching in the gpu backend.
264DEF_SIMPLE_GM(vertices_batching, canvas, 100, 500) {
Mike Reed887cdf12017-04-03 11:11:09 -0400265 draw_batching(canvas);
Brian Salomon199fb872017-02-06 09:41:10 -0500266 canvas->translate(50, 0);
Mike Reed887cdf12017-04-03 11:11:09 -0400267 draw_batching(canvas);
Brian Salomon3f363692017-02-02 21:05:19 -0500268}