2018-01-25 21:26:25 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2018 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2019-03-20 16:55:08 +00:00
|
|
|
#include "AnimTimer.h"
|
2018-08-08 15:36:17 +00:00
|
|
|
#include "Sample.h"
|
2018-01-25 21:26:25 +00:00
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkColorFilter.h"
|
2019-03-20 16:55:08 +00:00
|
|
|
#include "SkColorPriv.h"
|
|
|
|
#include "SkFont.h"
|
2018-01-25 21:26:25 +00:00
|
|
|
#include "SkImage.h"
|
|
|
|
#include "SkRandom.h"
|
|
|
|
#include "SkTime.h"
|
|
|
|
#include "SkTypeface.h"
|
|
|
|
#include "Timer.h"
|
|
|
|
|
|
|
|
#if SK_SUPPORT_GPU
|
|
|
|
#include "GrContext.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Create an animation of a bunch of letters that rotate in place. This is intended to stress
|
|
|
|
// the glyph atlas and test that we don't see corruption or bad slowdowns.
|
2018-08-08 15:36:17 +00:00
|
|
|
class FlutterAnimateView : public Sample {
|
2018-01-25 21:26:25 +00:00
|
|
|
public:
|
|
|
|
FlutterAnimateView() : fCurrTime(0), fResetTime(0) {}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
void onOnceBeforeDraw() override {
|
|
|
|
fTypeface = SkTypeface::MakeFromFile("/skimages/samplefont.ttf");
|
|
|
|
initChars();
|
|
|
|
}
|
|
|
|
|
2018-08-08 15:36:17 +00:00
|
|
|
bool onQuery(Sample::Event* evt) override {
|
|
|
|
if (Sample::TitleQ(*evt)) {
|
|
|
|
Sample::TitleR(evt, "FlutterAnimate");
|
2018-01-25 21:26:25 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this->INHERITED::onQuery(evt);
|
|
|
|
}
|
|
|
|
|
|
|
|
void onDrawContent(SkCanvas* canvas) override {
|
2019-01-08 19:00:08 +00:00
|
|
|
SkFont font(fTypeface, 50);
|
2018-01-25 21:26:25 +00:00
|
|
|
SkPaint paint;
|
|
|
|
paint.setFilterQuality(kMedium_SkFilterQuality);
|
|
|
|
|
|
|
|
// rough center of each glyph
|
|
|
|
static constexpr auto kMidX = 35;
|
|
|
|
static constexpr auto kMidY = 50;
|
|
|
|
|
|
|
|
canvas->clear(SK_ColorWHITE);
|
|
|
|
for (int i = 0; i < kNumChars; ++i) {
|
|
|
|
canvas->save();
|
|
|
|
double rot = SkScalarInterp(fChars[i].fStartRotation, fChars[i].fEndRotation,
|
|
|
|
fCurrTime/kDuration);
|
|
|
|
canvas->translate(fChars[i].fPosition.fX + kMidX, fChars[i].fPosition.fY - kMidY);
|
|
|
|
canvas->rotate(SkRadiansToDegrees(rot));
|
|
|
|
canvas->translate(-35,+50);
|
2019-01-08 19:00:08 +00:00
|
|
|
canvas->drawString(fChars[i].fChar, 0, 0, font, paint);
|
2018-01-25 21:26:25 +00:00
|
|
|
canvas->restore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-20 16:55:08 +00:00
|
|
|
bool onAnimate(const AnimTimer& timer) override {
|
2018-01-25 21:26:25 +00:00
|
|
|
fCurrTime = timer.secs() - fResetTime;
|
|
|
|
if (fCurrTime > kDuration) {
|
|
|
|
this->initChars();
|
|
|
|
fResetTime = timer.secs();
|
|
|
|
fCurrTime = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void initChars() {
|
|
|
|
for (int i = 0; i < kNumChars; ++i) {
|
|
|
|
char c = fRand.nextULessThan(26) + 65;
|
|
|
|
fChars[i].fChar[0] = c;
|
|
|
|
fChars[i].fChar[1] = '\0';
|
|
|
|
fChars[i].fPosition = SkPoint::Make(fRand.nextF()*748 + 10, fRand.nextF()*1004 + 10);
|
|
|
|
fChars[i].fStartRotation = fRand.nextF();
|
|
|
|
fChars[i].fEndRotation = fRand.nextF() * 20 - 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr double kDuration = 5.0;
|
|
|
|
double fCurrTime;
|
|
|
|
double fResetTime;
|
|
|
|
SkRandom fRand;
|
|
|
|
|
|
|
|
struct AnimatedChar {
|
|
|
|
char fChar[2];
|
|
|
|
SkPoint fPosition;
|
|
|
|
SkScalar fStartRotation;
|
|
|
|
SkScalar fEndRotation;
|
|
|
|
};
|
|
|
|
sk_sp<SkTypeface> fTypeface;
|
|
|
|
static constexpr int kNumChars = 40;
|
|
|
|
AnimatedChar fChars[kNumChars];
|
|
|
|
|
2018-08-08 15:36:17 +00:00
|
|
|
typedef Sample INHERITED;
|
2018-01-25 21:26:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2018-08-08 15:36:17 +00:00
|
|
|
DEF_SAMPLE( return new FlutterAnimateView(); )
|