blob: a6ab820fe753bb41692f5c731df96a62c03c642f [file] [log] [blame]
reed71c3c762015-06-24 10:29:17 -07001/*
2 * Copyright 2015 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
Ben Wagnerb2c4ea62018-08-08 11:36:17 -04008#include "Sample.h"
reed71c3c762015-06-24 10:29:17 -07009#include "SkAnimTimer.h"
reed71c3c762015-06-24 10:29:17 -070010#include "SkCanvas.h"
11#include "SkDrawable.h"
12#include "SkPath.h"
13#include "SkRandom.h"
14#include "SkRSXform.h"
15#include "SkSurface.h"
16
reedfd3d87c2015-08-06 05:14:11 -070017typedef void (*DrawAtlasProc)(SkCanvas*, SkImage*, const SkRSXform[], const SkRect[],
18 const SkColor[], int, const SkRect*, const SkPaint*);
19
20static void draw_atlas(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[],
21 const SkRect tex[], const SkColor colors[], int count, const SkRect* cull,
22 const SkPaint* paint) {
Mike Reed7d954ad2016-10-28 15:42:34 -040023 canvas->drawAtlas(atlas, xform, tex, colors, count, SkBlendMode::kModulate, cull, paint);
reedfd3d87c2015-08-06 05:14:11 -070024}
25
26static void draw_atlas_sim(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[],
27 const SkRect tex[], const SkColor colors[], int count, const SkRect* cull,
28 const SkPaint* paint) {
29 for (int i = 0; i < count; ++i) {
30 SkMatrix matrix;
31 matrix.setRSXform(xform[i]);
halcanary9d524f22016-03-29 09:03:52 -070032
reedfd3d87c2015-08-06 05:14:11 -070033 canvas->save();
34 canvas->concat(matrix);
reede47829b2015-08-06 10:02:53 -070035 canvas->drawImageRect(atlas, tex[i], tex[i].makeOffset(-tex[i].x(), -tex[i].y()), paint,
reed2dcc7592015-08-06 06:51:52 -070036 SkCanvas::kFast_SrcRectConstraint);
reedfd3d87c2015-08-06 05:14:11 -070037 canvas->restore();
38 }
39}
40
reed9ce9d672016-03-17 10:51:11 -070041static sk_sp<SkImage> make_atlas(int atlasSize, int cellSize) {
reed71c3c762015-06-24 10:29:17 -070042 SkImageInfo info = SkImageInfo::MakeN32Premul(atlasSize, atlasSize);
reede8f30622016-03-23 18:59:25 -070043 auto surface(SkSurface::MakeRaster(info));
reed71c3c762015-06-24 10:29:17 -070044 SkCanvas* canvas = surface->getCanvas();
45
46 SkPaint paint;
47 paint.setAntiAlias(true);
48 SkRandom rand;
49
50 const SkScalar half = cellSize * SK_ScalarHalf;
51 const char* s = "01234567890!@#$%^&*=+<>?abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
52 paint.setTextSize(28);
53 paint.setTextAlign(SkPaint::kCenter_Align);
54 int i = 0;
55 for (int y = 0; y < atlasSize; y += cellSize) {
56 for (int x = 0; x < atlasSize; x += cellSize) {
57 paint.setColor(rand.nextU());
58 paint.setAlpha(0xFF);
59 int index = i % strlen(s);
60 canvas->drawText(&s[index], 1, x + half, y + half + half/2, paint);
61 i += 1;
62 }
63 }
reed9ce9d672016-03-17 10:51:11 -070064 return surface->makeImageSnapshot();
reed71c3c762015-06-24 10:29:17 -070065}
66
67class DrawAtlasDrawable : public SkDrawable {
68 enum {
69 kMaxScale = 2,
70 kCellSize = 32,
71 kAtlasSize = 512,
72 };
73
74 struct Rec {
75 SkPoint fCenter;
76 SkVector fVelocity;
77 SkScalar fScale;
78 SkScalar fDScale;
79 SkScalar fRadian;
80 SkScalar fDRadian;
81 SkScalar fAlpha;
82 SkScalar fDAlpha;
83
84 void advance(const SkRect& bounds) {
85 fCenter += fVelocity;
86 if (fCenter.fX > bounds.right()) {
87 SkASSERT(fVelocity.fX > 0);
88 fVelocity.fX = -fVelocity.fX;
89 } else if (fCenter.fX < bounds.left()) {
90 SkASSERT(fVelocity.fX < 0);
91 fVelocity.fX = -fVelocity.fX;
92 }
93 if (fCenter.fY > bounds.bottom()) {
reedca109532015-06-25 16:25:25 -070094 if (fVelocity.fY > 0) {
95 fVelocity.fY = -fVelocity.fY;
96 }
reed71c3c762015-06-24 10:29:17 -070097 } else if (fCenter.fY < bounds.top()) {
reedca109532015-06-25 16:25:25 -070098 if (fVelocity.fY < 0) {
99 fVelocity.fY = -fVelocity.fY;
100 }
reed71c3c762015-06-24 10:29:17 -0700101 }
102
103 fScale += fDScale;
104 if (fScale > 2 || fScale < SK_Scalar1/2) {
105 fDScale = -fDScale;
106 }
107
108 fRadian += fDRadian;
109 fRadian = SkScalarMod(fRadian, 2 * SK_ScalarPI);
110
111 fAlpha += fDAlpha;
112 if (fAlpha > 1) {
113 fAlpha = 1;
114 fDAlpha = -fDAlpha;
115 } else if (fAlpha < 0) {
116 fAlpha = 0;
117 fDAlpha = -fDAlpha;
118 }
119 }
halcanary9d524f22016-03-29 09:03:52 -0700120
reed71c3c762015-06-24 10:29:17 -0700121 SkRSXform asRSXform() const {
reed6b38eab2015-07-30 05:46:05 -0700122 return SkRSXform::MakeFromRadians(fScale, fRadian, fCenter.x(), fCenter.y(),
123 SkScalarHalf(kCellSize), SkScalarHalf(kCellSize));
reed71c3c762015-06-24 10:29:17 -0700124 }
125 };
126
reedfd3d87c2015-08-06 05:14:11 -0700127 DrawAtlasProc fProc;
128
reed71c3c762015-06-24 10:29:17 -0700129 enum {
130 N = 256,
131 };
132
reed9ce9d672016-03-17 10:51:11 -0700133 sk_sp<SkImage> fAtlas;
reed71c3c762015-06-24 10:29:17 -0700134 Rec fRec[N];
135 SkRect fTex[N];
136 SkRect fBounds;
137 bool fUseColors;
138
139public:
reedfd3d87c2015-08-06 05:14:11 -0700140 DrawAtlasDrawable(DrawAtlasProc proc, const SkRect& r)
141 : fProc(proc), fBounds(r), fUseColors(false)
142 {
reed71c3c762015-06-24 10:29:17 -0700143 SkRandom rand;
reed9ce9d672016-03-17 10:51:11 -0700144 fAtlas = make_atlas(kAtlasSize, kCellSize);
reed71c3c762015-06-24 10:29:17 -0700145 const SkScalar kMaxSpeed = 5;
146 const SkScalar cell = SkIntToScalar(kCellSize);
147 int i = 0;
148 for (int y = 0; y < kAtlasSize; y += kCellSize) {
149 for (int x = 0; x < kAtlasSize; x += kCellSize) {
150 const SkScalar sx = SkIntToScalar(x);
151 const SkScalar sy = SkIntToScalar(y);
152 fTex[i].setXYWH(sx, sy, cell, cell);
halcanary9d524f22016-03-29 09:03:52 -0700153
reed71c3c762015-06-24 10:29:17 -0700154 fRec[i].fCenter.set(sx + cell/2, sy + 3*cell/4);
155 fRec[i].fVelocity.fX = rand.nextSScalar1() * kMaxSpeed;
156 fRec[i].fVelocity.fY = rand.nextSScalar1() * kMaxSpeed;
157 fRec[i].fScale = 1;
reed6b38eab2015-07-30 05:46:05 -0700158 fRec[i].fDScale = rand.nextSScalar1() / 16;
reed71c3c762015-06-24 10:29:17 -0700159 fRec[i].fRadian = 0;
160 fRec[i].fDRadian = rand.nextSScalar1() / 8;
161 fRec[i].fAlpha = rand.nextUScalar1();
162 fRec[i].fDAlpha = rand.nextSScalar1() / 10;
163 i += 1;
164 }
165 }
166 }
167
168 void toggleUseColors() {
169 fUseColors = !fUseColors;
170 }
171
172protected:
173 void onDraw(SkCanvas* canvas) override {
174 SkRSXform xform[N];
175 SkColor colors[N];
176
177 for (int i = 0; i < N; ++i) {
178 fRec[i].advance(fBounds);
179 xform[i] = fRec[i].asRSXform();
180 if (fUseColors) {
181 colors[i] = SkColorSetARGB((int)(fRec[i].fAlpha * 0xFF), 0xFF, 0xFF, 0xFF);
182 }
183 }
184 SkPaint paint;
185 paint.setFilterQuality(kLow_SkFilterQuality);
186
187 const SkRect cull = this->getBounds();
halcanary96fcdcc2015-08-27 07:41:13 -0700188 const SkColor* colorsPtr = fUseColors ? colors : nullptr;
reed9ce9d672016-03-17 10:51:11 -0700189 fProc(canvas, fAtlas.get(), xform, fTex, colorsPtr, N, &cull, &paint);
reed71c3c762015-06-24 10:29:17 -0700190 }
halcanary9d524f22016-03-29 09:03:52 -0700191
reed71c3c762015-06-24 10:29:17 -0700192 SkRect onGetBounds() override {
193 const SkScalar border = kMaxScale * kCellSize;
194 SkRect r = fBounds;
195 r.outset(border, border);
196 return r;
197 }
198
199private:
200 typedef SkDrawable INHERITED;
201};
202
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400203class DrawAtlasView : public Sample {
Ben Wagner4d516112018-06-07 13:11:37 -0400204 const char* fName;
205 DrawAtlasProc fProc;
206 sk_sp<DrawAtlasDrawable> fDrawable;
reed71c3c762015-06-24 10:29:17 -0700207
208public:
Ben Wagner4d516112018-06-07 13:11:37 -0400209 DrawAtlasView(const char name[], DrawAtlasProc proc) : fName(name), fProc(proc) { }
reed71c3c762015-06-24 10:29:17 -0700210
211protected:
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400212 bool onQuery(Sample::Event* evt) override {
213 if (Sample::TitleQ(*evt)) {
214 Sample::TitleR(evt, fName);
reed71c3c762015-06-24 10:29:17 -0700215 return true;
216 }
217 SkUnichar uni;
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400218 if (Sample::CharQ(*evt, &uni)) {
reed71c3c762015-06-24 10:29:17 -0700219 switch (uni) {
Brian Osmanede860e2017-11-22 16:36:07 -0500220 case 'C': fDrawable->toggleUseColors(); return true;
reed71c3c762015-06-24 10:29:17 -0700221 default: break;
222 }
223 }
224 return this->INHERITED::onQuery(evt);
225 }
226
Ben Wagner4d516112018-06-07 13:11:37 -0400227 void onOnceBeforeDraw() override {
228 fDrawable = sk_make_sp<DrawAtlasDrawable>(fProc, SkRect::MakeWH(640, 480));
229 }
230
reed71c3c762015-06-24 10:29:17 -0700231 void onDrawContent(SkCanvas* canvas) override {
Ben Wagner4d516112018-06-07 13:11:37 -0400232 canvas->drawDrawable(fDrawable.get());
reed71c3c762015-06-24 10:29:17 -0700233 }
234
jvanverthc7027ab2016-06-16 09:52:35 -0700235 bool onAnimate(const SkAnimTimer&) override {
236 return true;
237 }
reed71c3c762015-06-24 10:29:17 -0700238#if 0
239 // TODO: switch over to use this for our animation
240 bool onAnimate(const SkAnimTimer& timer) override {
241 SkScalar angle = SkDoubleToScalar(fmod(timer.secs() * 360 / 24, 360));
242 fAnimatingDrawable->setSweep(angle);
243 return true;
244 }
245#endif
246
247private:
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400248 typedef Sample INHERITED;
reed71c3c762015-06-24 10:29:17 -0700249};
250
251//////////////////////////////////////////////////////////////////////////////
252
reedfd3d87c2015-08-06 05:14:11 -0700253DEF_SAMPLE( return new DrawAtlasView("DrawAtlas", draw_atlas); )
254DEF_SAMPLE( return new DrawAtlasView("DrawAtlasSim", draw_atlas_sim); )