blob: 8f7f116dc76d65283b493a44994d660a24a54cf2 [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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "include/core/SkCanvas.h"
9#include "include/core/SkDrawable.h"
10#include "include/core/SkPath.h"
11#include "include/core/SkRSXform.h"
12#include "include/core/SkSurface.h"
13#include "include/utils/SkRandom.h"
14#include "include/utils/SkTextUtils.h"
15#include "samplecode/Sample.h"
Mike Reede78f8ce2020-12-19 15:14:06 -050016#include "src/core/SkPaintPriv.h"
reed71c3c762015-06-24 10:29:17 -070017
reedfd3d87c2015-08-06 05:14:11 -070018typedef void (*DrawAtlasProc)(SkCanvas*, SkImage*, const SkRSXform[], const SkRect[],
Mike Reed34c56a52021-01-22 15:26:41 -050019 const SkColor[], int, const SkRect*, const SkSamplingOptions&,
20 const SkPaint*);
reedfd3d87c2015-08-06 05:14:11 -070021
22static void draw_atlas(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[],
23 const SkRect tex[], const SkColor colors[], int count, const SkRect* cull,
Mike Reed34c56a52021-01-22 15:26:41 -050024 const SkSamplingOptions& sampling, const SkPaint* paint) {
25 canvas->drawAtlas(atlas, xform, tex, colors, count, SkBlendMode::kModulate,
26 sampling, cull, paint);
reedfd3d87c2015-08-06 05:14:11 -070027}
28
29static void draw_atlas_sim(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[],
30 const SkRect tex[], const SkColor colors[], int count, const SkRect* cull,
Mike Reed34c56a52021-01-22 15:26:41 -050031 const SkSamplingOptions& sampling, const SkPaint* paint) {
reedfd3d87c2015-08-06 05:14:11 -070032 for (int i = 0; i < count; ++i) {
33 SkMatrix matrix;
34 matrix.setRSXform(xform[i]);
halcanary9d524f22016-03-29 09:03:52 -070035
reedfd3d87c2015-08-06 05:14:11 -070036 canvas->save();
37 canvas->concat(matrix);
Mike Reed34c56a52021-01-22 15:26:41 -050038 canvas->drawImageRect(atlas, tex[i], tex[i].makeOffset(-tex[i].x(), -tex[i].y()),
39 sampling, paint, SkCanvas::kFast_SrcRectConstraint);
reedfd3d87c2015-08-06 05:14:11 -070040 canvas->restore();
41 }
42}
43
reed9ce9d672016-03-17 10:51:11 -070044static sk_sp<SkImage> make_atlas(int atlasSize, int cellSize) {
reed71c3c762015-06-24 10:29:17 -070045 SkImageInfo info = SkImageInfo::MakeN32Premul(atlasSize, atlasSize);
reede8f30622016-03-23 18:59:25 -070046 auto surface(SkSurface::MakeRaster(info));
reed71c3c762015-06-24 10:29:17 -070047 SkCanvas* canvas = surface->getCanvas();
48
49 SkPaint paint;
reed71c3c762015-06-24 10:29:17 -070050 SkRandom rand;
51
52 const SkScalar half = cellSize * SK_ScalarHalf;
53 const char* s = "01234567890!@#$%^&*=+<>?abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
Hal Canary4484b8f2019-01-08 14:00:08 -050054 SkFont font(nullptr, 28);
55
reed71c3c762015-06-24 10:29:17 -070056 int i = 0;
57 for (int y = 0; y < atlasSize; y += cellSize) {
58 for (int x = 0; x < atlasSize; x += cellSize) {
59 paint.setColor(rand.nextU());
60 paint.setAlpha(0xFF);
61 int index = i % strlen(s);
Ben Wagner51e15a62019-05-07 15:38:46 -040062 SkTextUtils::Draw(canvas, &s[index], 1, SkTextEncoding::kUTF8,
Hal Canary4484b8f2019-01-08 14:00:08 -050063 x + half, y + half + half/2, font, paint,
Mike Reedb579f072019-01-03 15:45:53 -050064 SkTextUtils::kCenter_Align);
reed71c3c762015-06-24 10:29:17 -070065 i += 1;
66 }
67 }
reed9ce9d672016-03-17 10:51:11 -070068 return surface->makeImageSnapshot();
reed71c3c762015-06-24 10:29:17 -070069}
70
71class DrawAtlasDrawable : public SkDrawable {
72 enum {
73 kMaxScale = 2,
74 kCellSize = 32,
75 kAtlasSize = 512,
76 };
77
78 struct Rec {
79 SkPoint fCenter;
80 SkVector fVelocity;
81 SkScalar fScale;
82 SkScalar fDScale;
83 SkScalar fRadian;
84 SkScalar fDRadian;
85 SkScalar fAlpha;
86 SkScalar fDAlpha;
87
88 void advance(const SkRect& bounds) {
89 fCenter += fVelocity;
90 if (fCenter.fX > bounds.right()) {
91 SkASSERT(fVelocity.fX > 0);
92 fVelocity.fX = -fVelocity.fX;
93 } else if (fCenter.fX < bounds.left()) {
94 SkASSERT(fVelocity.fX < 0);
95 fVelocity.fX = -fVelocity.fX;
96 }
97 if (fCenter.fY > bounds.bottom()) {
reedca109532015-06-25 16:25:25 -070098 if (fVelocity.fY > 0) {
99 fVelocity.fY = -fVelocity.fY;
100 }
reed71c3c762015-06-24 10:29:17 -0700101 } else if (fCenter.fY < bounds.top()) {
reedca109532015-06-25 16:25:25 -0700102 if (fVelocity.fY < 0) {
103 fVelocity.fY = -fVelocity.fY;
104 }
reed71c3c762015-06-24 10:29:17 -0700105 }
106
107 fScale += fDScale;
108 if (fScale > 2 || fScale < SK_Scalar1/2) {
109 fDScale = -fDScale;
110 }
111
112 fRadian += fDRadian;
113 fRadian = SkScalarMod(fRadian, 2 * SK_ScalarPI);
114
115 fAlpha += fDAlpha;
116 if (fAlpha > 1) {
117 fAlpha = 1;
118 fDAlpha = -fDAlpha;
119 } else if (fAlpha < 0) {
120 fAlpha = 0;
121 fDAlpha = -fDAlpha;
122 }
123 }
halcanary9d524f22016-03-29 09:03:52 -0700124
reed71c3c762015-06-24 10:29:17 -0700125 SkRSXform asRSXform() const {
reed6b38eab2015-07-30 05:46:05 -0700126 return SkRSXform::MakeFromRadians(fScale, fRadian, fCenter.x(), fCenter.y(),
127 SkScalarHalf(kCellSize), SkScalarHalf(kCellSize));
reed71c3c762015-06-24 10:29:17 -0700128 }
129 };
130
reedfd3d87c2015-08-06 05:14:11 -0700131 DrawAtlasProc fProc;
132
reed71c3c762015-06-24 10:29:17 -0700133 enum {
134 N = 256,
135 };
136
reed9ce9d672016-03-17 10:51:11 -0700137 sk_sp<SkImage> fAtlas;
reed71c3c762015-06-24 10:29:17 -0700138 Rec fRec[N];
139 SkRect fTex[N];
140 SkRect fBounds;
141 bool fUseColors;
142
143public:
reedfd3d87c2015-08-06 05:14:11 -0700144 DrawAtlasDrawable(DrawAtlasProc proc, const SkRect& r)
145 : fProc(proc), fBounds(r), fUseColors(false)
146 {
reed71c3c762015-06-24 10:29:17 -0700147 SkRandom rand;
reed9ce9d672016-03-17 10:51:11 -0700148 fAtlas = make_atlas(kAtlasSize, kCellSize);
reed71c3c762015-06-24 10:29:17 -0700149 const SkScalar kMaxSpeed = 5;
150 const SkScalar cell = SkIntToScalar(kCellSize);
151 int i = 0;
152 for (int y = 0; y < kAtlasSize; y += kCellSize) {
153 for (int x = 0; x < kAtlasSize; x += kCellSize) {
154 const SkScalar sx = SkIntToScalar(x);
155 const SkScalar sy = SkIntToScalar(y);
156 fTex[i].setXYWH(sx, sy, cell, cell);
halcanary9d524f22016-03-29 09:03:52 -0700157
reed71c3c762015-06-24 10:29:17 -0700158 fRec[i].fCenter.set(sx + cell/2, sy + 3*cell/4);
159 fRec[i].fVelocity.fX = rand.nextSScalar1() * kMaxSpeed;
160 fRec[i].fVelocity.fY = rand.nextSScalar1() * kMaxSpeed;
161 fRec[i].fScale = 1;
reed6b38eab2015-07-30 05:46:05 -0700162 fRec[i].fDScale = rand.nextSScalar1() / 16;
reed71c3c762015-06-24 10:29:17 -0700163 fRec[i].fRadian = 0;
164 fRec[i].fDRadian = rand.nextSScalar1() / 8;
165 fRec[i].fAlpha = rand.nextUScalar1();
166 fRec[i].fDAlpha = rand.nextSScalar1() / 10;
167 i += 1;
168 }
169 }
170 }
171
172 void toggleUseColors() {
173 fUseColors = !fUseColors;
174 }
175
176protected:
177 void onDraw(SkCanvas* canvas) override {
178 SkRSXform xform[N];
179 SkColor colors[N];
180
181 for (int i = 0; i < N; ++i) {
182 fRec[i].advance(fBounds);
183 xform[i] = fRec[i].asRSXform();
184 if (fUseColors) {
185 colors[i] = SkColorSetARGB((int)(fRec[i].fAlpha * 0xFF), 0xFF, 0xFF, 0xFF);
186 }
187 }
188 SkPaint paint;
Mike Reed34c56a52021-01-22 15:26:41 -0500189 SkSamplingOptions sampling(SkFilterMode::kLinear);
reed71c3c762015-06-24 10:29:17 -0700190
191 const SkRect cull = this->getBounds();
halcanary96fcdcc2015-08-27 07:41:13 -0700192 const SkColor* colorsPtr = fUseColors ? colors : nullptr;
Mike Reed34c56a52021-01-22 15:26:41 -0500193 fProc(canvas, fAtlas.get(), xform, fTex, colorsPtr, N, &cull, sampling, &paint);
reed71c3c762015-06-24 10:29:17 -0700194 }
halcanary9d524f22016-03-29 09:03:52 -0700195
reed71c3c762015-06-24 10:29:17 -0700196 SkRect onGetBounds() override {
197 const SkScalar border = kMaxScale * kCellSize;
198 SkRect r = fBounds;
199 r.outset(border, border);
200 return r;
201 }
202
203private:
John Stiles7571f9e2020-09-02 22:42:33 -0400204 using INHERITED = SkDrawable;
reed71c3c762015-06-24 10:29:17 -0700205};
206
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400207class DrawAtlasView : public Sample {
Ben Wagner4d516112018-06-07 13:11:37 -0400208 const char* fName;
209 DrawAtlasProc fProc;
210 sk_sp<DrawAtlasDrawable> fDrawable;
reed71c3c762015-06-24 10:29:17 -0700211
212public:
Ben Wagner4d516112018-06-07 13:11:37 -0400213 DrawAtlasView(const char name[], DrawAtlasProc proc) : fName(name), fProc(proc) { }
reed71c3c762015-06-24 10:29:17 -0700214
215protected:
Hal Canary8a027312019-07-03 10:55:44 -0400216 SkString name() override { return SkString(fName); }
217
Hal Canary6cc65e12019-07-03 15:53:04 -0400218 bool onChar(SkUnichar uni) override {
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 }
Hal Canary6cc65e12019-07-03 15:53:04 -0400223 return false;
reed71c3c762015-06-24 10:29:17 -0700224 }
225
Ben Wagner4d516112018-06-07 13:11:37 -0400226 void onOnceBeforeDraw() override {
227 fDrawable = sk_make_sp<DrawAtlasDrawable>(fProc, SkRect::MakeWH(640, 480));
228 }
229
reed71c3c762015-06-24 10:29:17 -0700230 void onDrawContent(SkCanvas* canvas) override {
Ben Wagner4d516112018-06-07 13:11:37 -0400231 canvas->drawDrawable(fDrawable.get());
reed71c3c762015-06-24 10:29:17 -0700232 }
233
Hal Canary41248072019-07-11 16:32:53 -0400234 bool onAnimate(double /*nanos*/) override { return true; }
reed71c3c762015-06-24 10:29:17 -0700235#if 0
236 // TODO: switch over to use this for our animation
Hal Canary41248072019-07-11 16:32:53 -0400237 bool onAnimate(double nanos) override {
238 SkScalar angle = SkDoubleToScalar(fmod(1e-9 * nanos * 360 / 24, 360));
reed71c3c762015-06-24 10:29:17 -0700239 fAnimatingDrawable->setSweep(angle);
240 return true;
241 }
242#endif
243
244private:
John Stiles7571f9e2020-09-02 22:42:33 -0400245 using INHERITED = Sample;
reed71c3c762015-06-24 10:29:17 -0700246};
247
248//////////////////////////////////////////////////////////////////////////////
249
reedfd3d87c2015-08-06 05:14:11 -0700250DEF_SAMPLE( return new DrawAtlasView("DrawAtlas", draw_atlas); )
251DEF_SAMPLE( return new DrawAtlasView("DrawAtlasSim", draw_atlas_sim); )