xref: /aosp_15_r20/external/skia/tools/viewer/AtlasSlide.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2015 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkCanvas.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkDrawable.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkFont.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPath.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRSXform.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkSurface.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/utils/SkTextUtils.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkRandom.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkPaintPriv.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "tools/fonts/FontToolUtils.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "tools/viewer/Slide.h"
19*c8dee2aaSAndroid Build Coastguard Worker 
20*c8dee2aaSAndroid Build Coastguard Worker typedef void (*DrawAtlasProc)(SkCanvas*, SkImage*, const SkRSXform[], const SkRect[],
21*c8dee2aaSAndroid Build Coastguard Worker                               const SkColor[], int, const SkRect*, const SkSamplingOptions&,
22*c8dee2aaSAndroid Build Coastguard Worker                               const SkPaint*);
23*c8dee2aaSAndroid Build Coastguard Worker 
draw_atlas(SkCanvas * canvas,SkImage * atlas,const SkRSXform xform[],const SkRect tex[],const SkColor colors[],int count,const SkRect * cull,const SkSamplingOptions & sampling,const SkPaint * paint)24*c8dee2aaSAndroid Build Coastguard Worker static void draw_atlas(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[],
25*c8dee2aaSAndroid Build Coastguard Worker                        const SkRect tex[], const SkColor colors[], int count, const SkRect* cull,
26*c8dee2aaSAndroid Build Coastguard Worker                        const SkSamplingOptions& sampling, const SkPaint* paint) {
27*c8dee2aaSAndroid Build Coastguard Worker     canvas->drawAtlas(atlas, xform, tex, colors, count, SkBlendMode::kModulate,
28*c8dee2aaSAndroid Build Coastguard Worker                       sampling, cull, paint);
29*c8dee2aaSAndroid Build Coastguard Worker }
30*c8dee2aaSAndroid Build Coastguard Worker 
draw_atlas_sim(SkCanvas * canvas,SkImage * atlas,const SkRSXform xform[],const SkRect tex[],const SkColor colors[],int count,const SkRect * cull,const SkSamplingOptions & sampling,const SkPaint * paint)31*c8dee2aaSAndroid Build Coastguard Worker static void draw_atlas_sim(SkCanvas* canvas, SkImage* atlas, const SkRSXform xform[],
32*c8dee2aaSAndroid Build Coastguard Worker                            const SkRect tex[], const SkColor colors[], int count, const SkRect* cull,
33*c8dee2aaSAndroid Build Coastguard Worker                            const SkSamplingOptions& sampling, const SkPaint* paint) {
34*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < count; ++i) {
35*c8dee2aaSAndroid Build Coastguard Worker         SkMatrix matrix;
36*c8dee2aaSAndroid Build Coastguard Worker         matrix.setRSXform(xform[i]);
37*c8dee2aaSAndroid Build Coastguard Worker 
38*c8dee2aaSAndroid Build Coastguard Worker         canvas->save();
39*c8dee2aaSAndroid Build Coastguard Worker         canvas->concat(matrix);
40*c8dee2aaSAndroid Build Coastguard Worker         canvas->drawImageRect(atlas, tex[i], tex[i].makeOffset(-tex[i].x(), -tex[i].y()),
41*c8dee2aaSAndroid Build Coastguard Worker                               sampling, paint, SkCanvas::kFast_SrcRectConstraint);
42*c8dee2aaSAndroid Build Coastguard Worker         canvas->restore();
43*c8dee2aaSAndroid Build Coastguard Worker     }
44*c8dee2aaSAndroid Build Coastguard Worker }
45*c8dee2aaSAndroid Build Coastguard Worker 
make_atlas(int atlasSize,int cellSize)46*c8dee2aaSAndroid Build Coastguard Worker static sk_sp<SkImage> make_atlas(int atlasSize, int cellSize) {
47*c8dee2aaSAndroid Build Coastguard Worker     SkImageInfo info = SkImageInfo::MakeN32Premul(atlasSize, atlasSize);
48*c8dee2aaSAndroid Build Coastguard Worker     auto surface(SkSurfaces::Raster(info));
49*c8dee2aaSAndroid Build Coastguard Worker     SkCanvas* canvas = surface->getCanvas();
50*c8dee2aaSAndroid Build Coastguard Worker 
51*c8dee2aaSAndroid Build Coastguard Worker     SkPaint paint;
52*c8dee2aaSAndroid Build Coastguard Worker     SkRandom rand;
53*c8dee2aaSAndroid Build Coastguard Worker 
54*c8dee2aaSAndroid Build Coastguard Worker     const SkScalar half = cellSize * SK_ScalarHalf;
55*c8dee2aaSAndroid Build Coastguard Worker     const char* s = "01234567890!@#$%^&*=+<>?abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
56*c8dee2aaSAndroid Build Coastguard Worker     SkFont font(ToolUtils::DefaultTypeface(), 28);
57*c8dee2aaSAndroid Build Coastguard Worker 
58*c8dee2aaSAndroid Build Coastguard Worker     int i = 0;
59*c8dee2aaSAndroid Build Coastguard Worker     for (int y = 0; y < atlasSize; y += cellSize) {
60*c8dee2aaSAndroid Build Coastguard Worker         for (int x = 0; x < atlasSize; x += cellSize) {
61*c8dee2aaSAndroid Build Coastguard Worker             paint.setColor(rand.nextU());
62*c8dee2aaSAndroid Build Coastguard Worker             paint.setAlpha(0xFF);
63*c8dee2aaSAndroid Build Coastguard Worker             int index = i % strlen(s);
64*c8dee2aaSAndroid Build Coastguard Worker             SkTextUtils::Draw(canvas, &s[index], 1, SkTextEncoding::kUTF8,
65*c8dee2aaSAndroid Build Coastguard Worker                               x + half, y + half + half/2, font, paint,
66*c8dee2aaSAndroid Build Coastguard Worker                               SkTextUtils::kCenter_Align);
67*c8dee2aaSAndroid Build Coastguard Worker             i += 1;
68*c8dee2aaSAndroid Build Coastguard Worker         }
69*c8dee2aaSAndroid Build Coastguard Worker     }
70*c8dee2aaSAndroid Build Coastguard Worker     return surface->makeImageSnapshot();
71*c8dee2aaSAndroid Build Coastguard Worker }
72*c8dee2aaSAndroid Build Coastguard Worker 
73*c8dee2aaSAndroid Build Coastguard Worker class DrawAtlasDrawable : public SkDrawable {
74*c8dee2aaSAndroid Build Coastguard Worker     static constexpr int kMaxScale = 2;
75*c8dee2aaSAndroid Build Coastguard Worker     static constexpr int kCellSize = 32;
76*c8dee2aaSAndroid Build Coastguard Worker     static constexpr int kAtlasSize = 512;
77*c8dee2aaSAndroid Build Coastguard Worker 
78*c8dee2aaSAndroid Build Coastguard Worker     struct Rec {
79*c8dee2aaSAndroid Build Coastguard Worker         SkPoint     fCenter;
80*c8dee2aaSAndroid Build Coastguard Worker         SkVector    fVelocity;
81*c8dee2aaSAndroid Build Coastguard Worker         SkScalar    fScale;
82*c8dee2aaSAndroid Build Coastguard Worker         SkScalar    fDScale;
83*c8dee2aaSAndroid Build Coastguard Worker         SkScalar    fRadian;
84*c8dee2aaSAndroid Build Coastguard Worker         SkScalar    fDRadian;
85*c8dee2aaSAndroid Build Coastguard Worker         SkScalar    fAlpha;
86*c8dee2aaSAndroid Build Coastguard Worker         SkScalar    fDAlpha;
87*c8dee2aaSAndroid Build Coastguard Worker 
advanceDrawAtlasDrawable::Rec88*c8dee2aaSAndroid Build Coastguard Worker         void advance(const SkRect& bounds) {
89*c8dee2aaSAndroid Build Coastguard Worker             fCenter += fVelocity;
90*c8dee2aaSAndroid Build Coastguard Worker             if (fCenter.fX > bounds.right()) {
91*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(fVelocity.fX > 0);
92*c8dee2aaSAndroid Build Coastguard Worker                 fVelocity.fX = -fVelocity.fX;
93*c8dee2aaSAndroid Build Coastguard Worker             } else if (fCenter.fX < bounds.left()) {
94*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(fVelocity.fX < 0);
95*c8dee2aaSAndroid Build Coastguard Worker                 fVelocity.fX = -fVelocity.fX;
96*c8dee2aaSAndroid Build Coastguard Worker             }
97*c8dee2aaSAndroid Build Coastguard Worker             if (fCenter.fY > bounds.bottom()) {
98*c8dee2aaSAndroid Build Coastguard Worker                 if (fVelocity.fY > 0) {
99*c8dee2aaSAndroid Build Coastguard Worker                     fVelocity.fY = -fVelocity.fY;
100*c8dee2aaSAndroid Build Coastguard Worker                 }
101*c8dee2aaSAndroid Build Coastguard Worker             } else if (fCenter.fY < bounds.top()) {
102*c8dee2aaSAndroid Build Coastguard Worker                 if (fVelocity.fY < 0) {
103*c8dee2aaSAndroid Build Coastguard Worker                     fVelocity.fY = -fVelocity.fY;
104*c8dee2aaSAndroid Build Coastguard Worker                 }
105*c8dee2aaSAndroid Build Coastguard Worker             }
106*c8dee2aaSAndroid Build Coastguard Worker 
107*c8dee2aaSAndroid Build Coastguard Worker             fScale += fDScale;
108*c8dee2aaSAndroid Build Coastguard Worker             if (fScale > 2 || fScale < SK_Scalar1/2) {
109*c8dee2aaSAndroid Build Coastguard Worker                 fDScale = -fDScale;
110*c8dee2aaSAndroid Build Coastguard Worker             }
111*c8dee2aaSAndroid Build Coastguard Worker 
112*c8dee2aaSAndroid Build Coastguard Worker             fRadian += fDRadian;
113*c8dee2aaSAndroid Build Coastguard Worker             fRadian = SkScalarMod(fRadian, 2 * SK_ScalarPI);
114*c8dee2aaSAndroid Build Coastguard Worker 
115*c8dee2aaSAndroid Build Coastguard Worker             fAlpha += fDAlpha;
116*c8dee2aaSAndroid Build Coastguard Worker             if (fAlpha > 1) {
117*c8dee2aaSAndroid Build Coastguard Worker                 fAlpha = 1;
118*c8dee2aaSAndroid Build Coastguard Worker                 fDAlpha = -fDAlpha;
119*c8dee2aaSAndroid Build Coastguard Worker             } else if (fAlpha < 0) {
120*c8dee2aaSAndroid Build Coastguard Worker                 fAlpha = 0;
121*c8dee2aaSAndroid Build Coastguard Worker                 fDAlpha = -fDAlpha;
122*c8dee2aaSAndroid Build Coastguard Worker             }
123*c8dee2aaSAndroid Build Coastguard Worker         }
124*c8dee2aaSAndroid Build Coastguard Worker 
asRSXformDrawAtlasDrawable::Rec125*c8dee2aaSAndroid Build Coastguard Worker         SkRSXform asRSXform() const {
126*c8dee2aaSAndroid Build Coastguard Worker             return SkRSXform::MakeFromRadians(fScale, fRadian, fCenter.x(), fCenter.y(),
127*c8dee2aaSAndroid Build Coastguard Worker                                               SkScalarHalf(kCellSize), SkScalarHalf(kCellSize));
128*c8dee2aaSAndroid Build Coastguard Worker         }
129*c8dee2aaSAndroid Build Coastguard Worker     };
130*c8dee2aaSAndroid Build Coastguard Worker 
131*c8dee2aaSAndroid Build Coastguard Worker     DrawAtlasProc fProc;
132*c8dee2aaSAndroid Build Coastguard Worker 
133*c8dee2aaSAndroid Build Coastguard Worker     enum {
134*c8dee2aaSAndroid Build Coastguard Worker         N = 256,
135*c8dee2aaSAndroid Build Coastguard Worker     };
136*c8dee2aaSAndroid Build Coastguard Worker 
137*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkImage> fAtlas;
138*c8dee2aaSAndroid Build Coastguard Worker     Rec         fRec[N];
139*c8dee2aaSAndroid Build Coastguard Worker     SkRect      fTex[N];
140*c8dee2aaSAndroid Build Coastguard Worker     SkRect      fBounds;
141*c8dee2aaSAndroid Build Coastguard Worker     bool        fUseColors;
142*c8dee2aaSAndroid Build Coastguard Worker 
143*c8dee2aaSAndroid Build Coastguard Worker public:
DrawAtlasDrawable(DrawAtlasProc proc,const SkRect & r)144*c8dee2aaSAndroid Build Coastguard Worker     DrawAtlasDrawable(DrawAtlasProc proc, const SkRect& r)
145*c8dee2aaSAndroid Build Coastguard Worker         : fProc(proc), fBounds(r), fUseColors(false)
146*c8dee2aaSAndroid Build Coastguard Worker     {
147*c8dee2aaSAndroid Build Coastguard Worker         SkRandom rand;
148*c8dee2aaSAndroid Build Coastguard Worker         fAtlas = make_atlas(kAtlasSize, kCellSize);
149*c8dee2aaSAndroid Build Coastguard Worker         const SkScalar kMaxSpeed = 5;
150*c8dee2aaSAndroid Build Coastguard Worker         const SkScalar cell = SkIntToScalar(kCellSize);
151*c8dee2aaSAndroid Build Coastguard Worker         int i = 0;
152*c8dee2aaSAndroid Build Coastguard Worker         for (int y = 0; y < kAtlasSize; y += kCellSize) {
153*c8dee2aaSAndroid Build Coastguard Worker             for (int x = 0; x < kAtlasSize; x += kCellSize) {
154*c8dee2aaSAndroid Build Coastguard Worker                 const SkScalar sx = SkIntToScalar(x);
155*c8dee2aaSAndroid Build Coastguard Worker                 const SkScalar sy = SkIntToScalar(y);
156*c8dee2aaSAndroid Build Coastguard Worker                 fTex[i].setXYWH(sx, sy, cell, cell);
157*c8dee2aaSAndroid Build Coastguard Worker 
158*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fCenter.set(sx + cell/2, sy + 3*cell/4);
159*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fVelocity.fX = rand.nextSScalar1() * kMaxSpeed;
160*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fVelocity.fY = rand.nextSScalar1() * kMaxSpeed;
161*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fScale = 1;
162*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fDScale = rand.nextSScalar1() / 16;
163*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fRadian = 0;
164*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fDRadian = rand.nextSScalar1() / 8;
165*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fAlpha = rand.nextUScalar1();
166*c8dee2aaSAndroid Build Coastguard Worker                 fRec[i].fDAlpha = rand.nextSScalar1() / 10;
167*c8dee2aaSAndroid Build Coastguard Worker                 i += 1;
168*c8dee2aaSAndroid Build Coastguard Worker             }
169*c8dee2aaSAndroid Build Coastguard Worker         }
170*c8dee2aaSAndroid Build Coastguard Worker     }
171*c8dee2aaSAndroid Build Coastguard Worker 
toggleUseColors()172*c8dee2aaSAndroid Build Coastguard Worker     void toggleUseColors() {
173*c8dee2aaSAndroid Build Coastguard Worker         fUseColors = !fUseColors;
174*c8dee2aaSAndroid Build Coastguard Worker     }
175*c8dee2aaSAndroid Build Coastguard Worker 
176*c8dee2aaSAndroid Build Coastguard Worker protected:
onDraw(SkCanvas * canvas)177*c8dee2aaSAndroid Build Coastguard Worker     void onDraw(SkCanvas* canvas) override {
178*c8dee2aaSAndroid Build Coastguard Worker         SkRSXform xform[N];
179*c8dee2aaSAndroid Build Coastguard Worker         SkColor colors[N];
180*c8dee2aaSAndroid Build Coastguard Worker 
181*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < N; ++i) {
182*c8dee2aaSAndroid Build Coastguard Worker             fRec[i].advance(fBounds);
183*c8dee2aaSAndroid Build Coastguard Worker             xform[i] = fRec[i].asRSXform();
184*c8dee2aaSAndroid Build Coastguard Worker             if (fUseColors) {
185*c8dee2aaSAndroid Build Coastguard Worker                 colors[i] = SkColorSetARGB((int)(fRec[i].fAlpha * 0xFF), 0xFF, 0xFF, 0xFF);
186*c8dee2aaSAndroid Build Coastguard Worker             }
187*c8dee2aaSAndroid Build Coastguard Worker         }
188*c8dee2aaSAndroid Build Coastguard Worker         SkPaint paint;
189*c8dee2aaSAndroid Build Coastguard Worker         SkSamplingOptions sampling(SkFilterMode::kLinear);
190*c8dee2aaSAndroid Build Coastguard Worker 
191*c8dee2aaSAndroid Build Coastguard Worker         const SkRect cull = this->getBounds();
192*c8dee2aaSAndroid Build Coastguard Worker         const SkColor* colorsPtr = fUseColors ? colors : nullptr;
193*c8dee2aaSAndroid Build Coastguard Worker         fProc(canvas, fAtlas.get(), xform, fTex, colorsPtr, N, &cull, sampling, &paint);
194*c8dee2aaSAndroid Build Coastguard Worker     }
195*c8dee2aaSAndroid Build Coastguard Worker 
onGetBounds()196*c8dee2aaSAndroid Build Coastguard Worker     SkRect onGetBounds() override {
197*c8dee2aaSAndroid Build Coastguard Worker         const SkScalar border = kMaxScale * kCellSize;
198*c8dee2aaSAndroid Build Coastguard Worker         SkRect r = fBounds;
199*c8dee2aaSAndroid Build Coastguard Worker         r.outset(border, border);
200*c8dee2aaSAndroid Build Coastguard Worker         return r;
201*c8dee2aaSAndroid Build Coastguard Worker     }
202*c8dee2aaSAndroid Build Coastguard Worker };
203*c8dee2aaSAndroid Build Coastguard Worker 
204*c8dee2aaSAndroid Build Coastguard Worker class DrawAtlasSlide : public Slide {
205*c8dee2aaSAndroid Build Coastguard Worker     DrawAtlasProc fProc;
206*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<DrawAtlasDrawable> fDrawable;
207*c8dee2aaSAndroid Build Coastguard Worker 
208*c8dee2aaSAndroid Build Coastguard Worker public:
DrawAtlasSlide(const char name[],DrawAtlasProc proc)209*c8dee2aaSAndroid Build Coastguard Worker     DrawAtlasSlide(const char name[], DrawAtlasProc proc) : fProc(proc) { fName = name; }
210*c8dee2aaSAndroid Build Coastguard Worker 
onChar(SkUnichar uni)211*c8dee2aaSAndroid Build Coastguard Worker     bool onChar(SkUnichar uni) override {
212*c8dee2aaSAndroid Build Coastguard Worker             switch (uni) {
213*c8dee2aaSAndroid Build Coastguard Worker                 case 'C': fDrawable->toggleUseColors(); return true;
214*c8dee2aaSAndroid Build Coastguard Worker                 default: break;
215*c8dee2aaSAndroid Build Coastguard Worker             }
216*c8dee2aaSAndroid Build Coastguard Worker             return false;
217*c8dee2aaSAndroid Build Coastguard Worker     }
218*c8dee2aaSAndroid Build Coastguard Worker 
draw(SkCanvas * canvas)219*c8dee2aaSAndroid Build Coastguard Worker     void draw(SkCanvas* canvas) override {
220*c8dee2aaSAndroid Build Coastguard Worker         canvas->drawDrawable(fDrawable.get());
221*c8dee2aaSAndroid Build Coastguard Worker     }
222*c8dee2aaSAndroid Build Coastguard Worker 
animate(double)223*c8dee2aaSAndroid Build Coastguard Worker     bool animate(double /*nanos*/) override { return true; }
224*c8dee2aaSAndroid Build Coastguard Worker #if 0
225*c8dee2aaSAndroid Build Coastguard Worker     // TODO: switch over to use this for our animation
226*c8dee2aaSAndroid Build Coastguard Worker     bool animate(double nanos) override {
227*c8dee2aaSAndroid Build Coastguard Worker         SkScalar angle = SkDoubleToScalar(fmod(1e-9 * nanos * 360 / 24, 360));
228*c8dee2aaSAndroid Build Coastguard Worker         fAnimatingDrawable->setSweep(angle);
229*c8dee2aaSAndroid Build Coastguard Worker         return true;
230*c8dee2aaSAndroid Build Coastguard Worker     }
231*c8dee2aaSAndroid Build Coastguard Worker #endif
232*c8dee2aaSAndroid Build Coastguard Worker 
load(SkScalar winWidth,SkScalar winHeight)233*c8dee2aaSAndroid Build Coastguard Worker     void load(SkScalar winWidth, SkScalar winHeight) override {
234*c8dee2aaSAndroid Build Coastguard Worker         fDrawable = sk_make_sp<DrawAtlasDrawable>(fProc, SkRect::Make(this->getDimensions()));
235*c8dee2aaSAndroid Build Coastguard Worker     }
236*c8dee2aaSAndroid Build Coastguard Worker 
getDimensions() const237*c8dee2aaSAndroid Build Coastguard Worker     SkISize getDimensions() const override { return {640, 480}; }
238*c8dee2aaSAndroid Build Coastguard Worker };
239*c8dee2aaSAndroid Build Coastguard Worker 
240*c8dee2aaSAndroid Build Coastguard Worker //////////////////////////////////////////////////////////////////////////////
241*c8dee2aaSAndroid Build Coastguard Worker 
242*c8dee2aaSAndroid Build Coastguard Worker DEF_SLIDE( return new DrawAtlasSlide("DrawAtlas", draw_atlas); )
243*c8dee2aaSAndroid Build Coastguard Worker DEF_SLIDE( return new DrawAtlasSlide("DrawAtlasSim", draw_atlas_sim); )
244