2018-06-01 17:46:46 +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.
|
|
|
|
|
*/
|
2018-08-08 15:36:17 +00:00
|
|
|
|
#include "Sample.h"
|
2018-06-01 17:46:46 +00:00
|
|
|
|
#include "sk_tool_utils.h"
|
|
|
|
|
|
|
|
|
|
#include "SkAnimTimer.h"
|
|
|
|
|
#include "SkCanvas.h"
|
|
|
|
|
#include "SkPath.h"
|
|
|
|
|
#include "SkRandom.h"
|
|
|
|
|
#include "SkRRect.h"
|
|
|
|
|
#include "SkTypeface.h"
|
|
|
|
|
|
2018-08-15 22:43:02 +00:00
|
|
|
|
#include <cmath>
|
|
|
|
|
|
2018-06-01 17:46:46 +00:00
|
|
|
|
// Implementation in C++ of Animated Emoji
|
|
|
|
|
// See https://t.d3fc.io/status/705212795936247808
|
2018-08-15 22:43:02 +00:00
|
|
|
|
// See https://crbug.com/848616
|
2018-06-01 17:46:46 +00:00
|
|
|
|
|
2018-08-08 15:36:17 +00:00
|
|
|
|
class GlyphTransformView : public Sample {
|
2018-06-01 17:46:46 +00:00
|
|
|
|
public:
|
|
|
|
|
GlyphTransformView() {}
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
void onOnceBeforeDraw() override {
|
|
|
|
|
fEmojiFont.fTypeface = sk_tool_utils::emoji_typeface();
|
|
|
|
|
fEmojiFont.fText = sk_tool_utils::emoji_sample_text();
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-08 15:36:17 +00:00
|
|
|
|
bool onQuery(Sample::Event* evt) override {
|
|
|
|
|
if (Sample::TitleQ(*evt)) {
|
|
|
|
|
Sample::TitleR(evt, "Glyph Transform");
|
2018-06-01 17:46:46 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return this->INHERITED::onQuery(evt);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void onDrawContent(SkCanvas* canvas) override {
|
|
|
|
|
SkPaint paint;
|
2019-01-02 17:21:01 +00:00
|
|
|
|
|
|
|
|
|
SkFont font(fEmojiFont.fTypeface);
|
2018-06-01 17:46:46 +00:00
|
|
|
|
const char* text = fEmojiFont.fText;
|
|
|
|
|
|
2018-08-15 22:43:02 +00:00
|
|
|
|
double baseline = this->height() / 2;
|
|
|
|
|
canvas->drawLine(0, baseline, this->width(), baseline, paint);
|
2018-06-01 17:46:46 +00:00
|
|
|
|
|
|
|
|
|
SkMatrix ctm;
|
2018-08-15 22:43:02 +00:00
|
|
|
|
ctm.setRotate(fRotate); // d3 rotate takes degrees
|
|
|
|
|
ctm.postScale(fScale * 4, fScale * 4);
|
|
|
|
|
ctm.postTranslate(fTranslate.fX + this->width() * 0.8, fTranslate.fY + baseline);
|
2018-06-01 17:46:46 +00:00
|
|
|
|
canvas->concat(ctm);
|
2018-08-15 22:43:02 +00:00
|
|
|
|
|
|
|
|
|
// d3 by default anchors text around the middle
|
|
|
|
|
SkRect bounds;
|
2019-01-02 17:21:01 +00:00
|
|
|
|
font.measureText(text, strlen(text), kUTF8_SkTextEncoding, &bounds);
|
|
|
|
|
canvas->drawSimpleText(text, strlen(text), kUTF8_SkTextEncoding, -bounds.centerX(), -bounds.centerY(),
|
|
|
|
|
font, paint);
|
2018-06-01 17:46:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool onAnimate(const SkAnimTimer& timer) override {
|
2018-08-15 22:43:02 +00:00
|
|
|
|
constexpr SkScalar maxt = 100000;
|
|
|
|
|
double t = timer.pingPong(20, 0, 0, maxt); // d3 t is in milliseconds
|
2018-06-01 17:46:46 +00:00
|
|
|
|
|
2018-08-15 22:43:02 +00:00
|
|
|
|
fTranslate.set(sin(t / 3000) - t * this->width() * 0.7 / maxt, sin(t / 999) / t);
|
|
|
|
|
fScale = 4.5 - std::sqrt(t) / 99;
|
2018-06-01 17:46:46 +00:00
|
|
|
|
fRotate = sin(t / 734);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
struct EmojiFont {
|
|
|
|
|
sk_sp<SkTypeface> fTypeface;
|
|
|
|
|
const char* fText;
|
|
|
|
|
} fEmojiFont;
|
|
|
|
|
|
|
|
|
|
SkVector fTranslate;
|
|
|
|
|
SkScalar fScale;
|
|
|
|
|
SkScalar fRotate;
|
|
|
|
|
|
2018-08-08 15:36:17 +00:00
|
|
|
|
typedef Sample INHERITED;
|
2018-06-01 17:46:46 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2018-08-08 15:36:17 +00:00
|
|
|
|
DEF_SAMPLE( return new GlyphTransformView(); )
|