skia2/samplecode/SampleGlyphTransform.cpp
Hal Canary 41248071ac tools: separate TimeUtils from AnimTimer
gm, slides, and samples no longer need to know about the implementation
details of AnimTimer.

This
    virtual bool onAnimate(const AnimTimer&);
becomes this:
    virtual bool onAnimate(double /*nanoseconds*/);
which is much easier to reason about.

AnimTimer itself is now part of viewer.

Change-Id: Ib70bf7a0798b1991f25204ae84f70463cdbeb358
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/226838
Reviewed-by: Ben Wagner <bungeman@google.com>
Commit-Queue: Hal Canary <halcanary@google.com>
2019-07-12 15:05:01 +00:00

84 lines
2.4 KiB
C++

/*
* Copyright 2018 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "samplecode/Sample.h"
#include "tools/ToolUtils.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkPath.h"
#include "include/core/SkRRect.h"
#include "include/core/SkTypeface.h"
#include "include/utils/SkRandom.h"
#include "tools/timer/TimeUtils.h"
#include <cmath>
// Implementation in C++ of Animated Emoji
// See https://t.d3fc.io/status/705212795936247808
// See https://crbug.com/848616
class GlyphTransformView : public Sample {
public:
GlyphTransformView() {}
protected:
void onOnceBeforeDraw() override {
fEmojiFont.fTypeface = ToolUtils::emoji_typeface();
fEmojiFont.fText = ToolUtils::emoji_sample_text();
}
SkString name() override { return SkString("Glyph Transform"); }
void onDrawContent(SkCanvas* canvas) override {
SkPaint paint;
SkFont font(fEmojiFont.fTypeface);
const char* text = fEmojiFont.fText;
double baseline = this->height() / 2;
canvas->drawLine(0, baseline, this->width(), baseline, paint);
SkMatrix ctm;
ctm.setRotate(fRotate); // d3 rotate takes degrees
ctm.postScale(fScale * 4, fScale * 4);
ctm.postTranslate(fTranslate.fX + this->width() * 0.8, fTranslate.fY + baseline);
canvas->concat(ctm);
// d3 by default anchors text around the middle
SkRect bounds;
font.measureText(text, strlen(text), SkTextEncoding::kUTF8, &bounds);
canvas->drawSimpleText(text, strlen(text), SkTextEncoding::kUTF8, -bounds.centerX(), -bounds.centerY(),
font, paint);
}
bool onAnimate(double nanos) override {
constexpr SkScalar maxt = 100000;
double t = TimeUtils::PingPong(1e-9 * nanos, 20, 0, 0, maxt); // d3 t is in milliseconds
fTranslate.set(sin(t / 3000) - t * this->width() * 0.7 / maxt, sin(t / 999) / t);
fScale = 4.5 - std::sqrt(t) / 99;
fRotate = sin(t / 734);
return true;
}
private:
struct EmojiFont {
sk_sp<SkTypeface> fTypeface;
const char* fText;
} fEmojiFont;
SkVector fTranslate;
SkScalar fScale;
SkScalar fRotate;
typedef Sample INHERITED;
};
//////////////////////////////////////////////////////////////////////////////
DEF_SAMPLE( return new GlyphTransformView(); )