blob: b89986995c6f5804061e0a4eea7c7765cbc5acff [file] [log] [blame]
/*
* Copyright 2017 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/core/SkCanvas.h"
#include "include/core/SkFontMetrics.h"
#include "include/core/SkFontMgr.h"
#include "include/core/SkTextBlob.h"
#include "include/core/SkTypeface.h"
#include "src/base/SkRandom.h"
#include "src/core/SkTHash.h"
#include "tools/Resources.h"
#include "tools/ToolUtils.h"
#include "tools/fonts/FontToolUtils.h"
#include "tools/timer/TimeUtils.h"
#include "tools/viewer/Slide.h"
#if defined(SK_GANESH)
#include "include/gpu/ganesh/GrDirectContext.h"
#include "src/gpu/AtlasTypes.h"
#include "src/gpu/ganesh/GrDirectContextPriv.h"
using MaskFormat = skgpu::MaskFormat;
#endif
static sk_sp<SkTypeface> chinese_typeface() {
#ifdef SK_BUILD_FOR_ANDROID
return ToolUtils::CreateTypefaceFromResource("fonts/NotoSansCJK-Regular.ttc");
#elif defined(SK_BUILD_FOR_WIN)
return ToolUtils::CreateTestTypeface("SimSun", SkFontStyle());
#elif defined(SK_BUILD_FOR_MAC)
return ToolUtils::CreateTestTypeface("Hiragino Sans GB W3", SkFontStyle());
#elif defined(SK_BUILD_FOR_IOS)
return ToolUtils::CreateTestTypeface("Hiragino Sans GB W3", SkFontStyle());
#elif defined(SK_BUILD_FOR_UNIX)
return ToolUtils::CreateTestTypeface("Noto Sans CJK SC", SkFontStyle());
#else
return nullptr;
#endif
}
class ChineseFlingSlide : public Slide {
inline static constexpr int kNumBlobs = 200;
inline static constexpr int kWordLength = 16;
sk_sp<SkTypeface> fTypeface;
SkFontMetrics fMetrics;
sk_sp<SkTextBlob> fBlobs[kNumBlobs];
SkRandom fRand;
int fIndex = 0;
public:
ChineseFlingSlide() { fName = "chinese-fling"; }
void draw(SkCanvas* canvas) override {
canvas->clear(0xFFDDDDDD);
SkPaint paint;
paint.setColor(0xDE000000);
// draw a consistent run of the 'words' - one word per line
int index = fIndex;
for (SkScalar y = 0.0f; y < 1024.0f; ) {
y += -fMetrics.fAscent;
canvas->drawTextBlob(fBlobs[index], 0, y, paint);
y += fMetrics.fDescent + fMetrics.fLeading;
++index;
index %= kNumBlobs;
}
// now "fling" a random amount
fIndex += fRand.nextRangeU(5, 20);
fIndex %= kNumBlobs;
}
void load(SkScalar w, SkScalar h) override {
fTypeface = chinese_typeface();
SkFont font(fTypeface, 56);
font.getMetrics(&fMetrics);
SkUnichar glyphs[kWordLength];
for (int32_t i = 0; i < kNumBlobs; ++i) {
this->createRandomWord(glyphs);
SkTextBlobBuilder builder;
ToolUtils::add_to_text_blob_w_len(&builder,
(const char*)glyphs,
kWordLength * 4,
SkTextEncoding::kUTF32,
font,
0,
0);
fBlobs[i] = builder.make();
}
}
// Construct a random kWordLength character 'word' drawing from the full Chinese set
void createRandomWord(SkUnichar glyphs[kWordLength]) {
for (int i = 0; i < kWordLength; ++i) {
glyphs[i] = fRand.nextRangeU(0x4F00, 0x9FA0);
}
}
};
class ChineseZoomSlide : public Slide {
inline static constexpr int kNumBlobs = 8;
inline static constexpr int kParagraphLength = 175;
bool fAfterFirstFrame = false;
sk_sp<SkTypeface> fTypeface;
SkFontMetrics fMetrics;
sk_sp<SkTextBlob> fBlobs[kNumBlobs];
SkRandom fRand;
SkScalar fScale = 15;
SkScalar fTranslate = 0;
public:
ChineseZoomSlide() { fName = "chinese-zoom"; }
bool onChar(SkUnichar uni) override {
if ('>' == uni) {
fScale += 0.125f;
return true;
}
if ('<' == uni) {
fScale -= 0.125f;
return true;
}
return false;
}
void draw(SkCanvas* canvas) override {
canvas->clear(0xFFDDDDDD);
SkPaint paint;
paint.setAntiAlias(true);
paint.setColor(0xDE000000);
if (fAfterFirstFrame) {
#if defined(SK_GANESH)
auto direct = GrAsDirectContext(canvas->recordingContext());
if (direct) {
sk_sp<SkImage> image = direct->priv().testingOnly_getFontAtlasImage(MaskFormat::kA8,
0);
canvas->drawImageRect(image,
SkRect::MakeXYWH(10.0f, 10.0f, 512.0f, 512.0),
SkSamplingOptions(), &paint);
image = direct->priv().testingOnly_getFontAtlasImage(MaskFormat::kA8, 1);
canvas->drawImageRect(image,
SkRect::MakeXYWH(522.0f, 10.0f, 512.f, 512.0f),
SkSamplingOptions(), &paint);
image = direct->priv().testingOnly_getFontAtlasImage(MaskFormat::kA8, 2);
canvas->drawImageRect(image,
SkRect::MakeXYWH(10.0f, 522.0f, 512.0f, 512.0f),
SkSamplingOptions(), &paint);
image = direct->priv().testingOnly_getFontAtlasImage(MaskFormat::kA8, 3);
canvas->drawImageRect(image,
SkRect::MakeXYWH(522.0f, 522.0f, 512.0f, 512.0f),
SkSamplingOptions(), &paint);
}
#endif
}
canvas->scale(fScale, fScale);
canvas->translate(0, fTranslate);
fTranslate -= 0.5f;
// draw a consistent run of the 'words' - one word per line
SkScalar y = 0;
for (int index = 0; index < kNumBlobs; ++index) {
y += -fMetrics.fAscent;
canvas->drawTextBlob(fBlobs[index], 0, y, paint);
y += 3*(fMetrics.fDescent - fMetrics.fAscent + fMetrics.fLeading);
}
if (!fAfterFirstFrame) {
fAfterFirstFrame = true;
}
}
void load(SkScalar w, SkScalar h) override {
fTypeface = chinese_typeface();
SkFont font(fTypeface, 11);
font.getMetrics(&fMetrics);
SkPaint paint;
paint.setColor(0xDE000000);
SkUnichar glyphs[45];
for (int32_t i = 0; i < kNumBlobs; ++i) {
SkTextBlobBuilder builder;
auto paragraphLength = kParagraphLength;
SkScalar y = 0;
while (paragraphLength - 45 > 0) {
auto currentLineLength = std::min(45, paragraphLength - 45);
this->createRandomLine(glyphs, currentLineLength);
ToolUtils::add_to_text_blob_w_len(&builder,
(const char*)glyphs,
currentLineLength * 4,
SkTextEncoding::kUTF32,
font,
0,
y);
y += fMetrics.fDescent - fMetrics.fAscent + fMetrics.fLeading;
paragraphLength -= 45;
}
fBlobs[i] = builder.make();
}
}
// Construct a random kWordLength character 'word' drawing from the full Chinese set
void createRandomLine(SkUnichar glyphs[45], int lineLength) {
for (auto i = 0; i < lineLength; ++i) {
glyphs[i] = fRand.nextRangeU(0x4F00, 0x9FA0);
}
}
};
/**
* Scrolls a large number of unique glyphs so the set of glyphs that need to be active in the cache
* evolves over time and glyphs age out of being needed in current frame.
*/
class ChineseScrollSlide : public Slide {
static constexpr int kNumBlobs = 50;
static constexpr int kLinesPerBlob = 6;
static constexpr int kLineLength = 45;
static constexpr float kFontSize = 25;
static constexpr int kDefaultScrollUnitsPerSecondLog2 = 7;
sk_sp<SkTypeface> fTypeface;
sk_sp<SkTextBlob> fBlobs[kNumBlobs];
float fVerticalPeriod;
SkRandom fRand;
float fTranslate = 0;
double fPrevNs = std::numeric_limits<double>::infinity();
// The scroll speed is sign(value)*2^abs(value) where sign is tri-state (-1, 0, or 1)
int fScrollUnitsPerSecondLog2 = kDefaultScrollUnitsPerSecondLog2;
public:
ChineseScrollSlide() { fName = "chinese-scroll"; }
bool onChar(SkUnichar uni) override {
if (',' == uni) {
--fScrollUnitsPerSecondLog2;
return true;
}
if ('.' == uni) {
++fScrollUnitsPerSecondLog2;
return true;
}
return false;
}
bool animate(double ns) override {
if (!fScrollUnitsPerSecondLog2) {
fPrevNs = std::numeric_limits<double>::infinity();
return false;
}
if (!std::isfinite(fPrevNs)) {
fPrevNs = ns;
return true;
}
float scrollUnitsPerSecond = std::pow(2, std::abs(fScrollUnitsPerSecondLog2));
if (fScrollUnitsPerSecondLog2 < 0) {
scrollUnitsPerSecond = -scrollUnitsPerSecond;
}
float dt = TimeUtils::NanosToSeconds(ns - fPrevNs) * scrollUnitsPerSecond;
fTranslate -= dt;
fPrevNs = ns;
return true;
}
void draw(SkCanvas* canvas) override {
canvas->clear(0xFFDDDDDD);
SkPaint paint;
paint.setAntiAlias(true);
paint.setColor(0xDE000000);
canvas->translate(0, fTranslate);
// We want the text to repeat in y. If we assume the total height of the text
// is larger than the window height then we need at most 2 repetitions to
// cover the window.
float n = std::floor(-fTranslate / fVerticalPeriod);
float y = n * fVerticalPeriod;
for (int i = 0; i < 2; ++i) {
for (int b = 0; b < kNumBlobs; ++b) {
canvas->drawTextBlob(fBlobs[b], 0, y, paint);
}
y += fVerticalPeriod;
}
}
void load(SkScalar w, SkScalar h) override {
fTypeface = chinese_typeface();
SkFont font(fTypeface, kFontSize);
SkFontMetrics metrics;
font.getMetrics(&metrics);
SkPaint paint;
paint.setColor(0xDE000000);
skia_private::THashSet<SkUnichar> usedGlyphs;
float y = 0;
for (int i = 0; i < kNumBlobs; ++i) {
SkTextBlobBuilder builder;
for (int j = 0; j < kLinesPerBlob; ++j) {
SkUnichar glyphs[kLineLength];
this->createRandomLine(glyphs, kLineLength, usedGlyphs);
ToolUtils::add_to_text_blob_w_len(&builder,
(const char*)glyphs,
kLineLength * 4,
SkTextEncoding::kUTF32,
font,
0,
y);
y += metrics.fDescent - metrics.fAscent + metrics.fLeading;
}
fBlobs[i] = builder.make();
}
fVerticalPeriod = std::ceil(y);
}
// Construct a random lineLength character 'word' drawing from the full Chinese set
// but not using glyphs in usedGlyphs.
void createRandomLine(SkUnichar glyphs[], int lineLength, auto& usedGlyphs) {
for (auto i = 0; i < lineLength; ++i) {
do {
glyphs[i] = fRand.nextRangeU(0x4F00, 0x9FA0);
} while (usedGlyphs.contains(glyphs[i]));
usedGlyphs.add(glyphs[i]);
}
}
};
//////////////////////////////////////////////////////////////////////////////
DEF_SLIDE( return new ChineseFlingSlide(); )
DEF_SLIDE( return new ChineseZoomSlide(); )
DEF_SLIDE( return new ChineseScrollSlide(); )