| /* |
| * 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 "Resources.h" |
| #include "Sample.h" |
| #include "ToolUtils.h" |
| |
| #include "SkCanvas.h" |
| #include "SkFontMetrics.h" |
| #include "SkFontMgr.h" |
| #include "SkRandom.h" |
| #include "SkTypeface.h" |
| #include "SkTextBlob.h" |
| |
| #if SK_SUPPORT_GPU |
| #include "GrContext.h" |
| #include "GrContextPriv.h" |
| #endif |
| |
| static sk_sp<SkTypeface> chinese_typeface() { |
| #ifdef SK_BUILD_FOR_ANDROID |
| return MakeResourceAsTypeface("fonts/NotoSansCJK-Regular.ttc"); |
| #elif defined(SK_BUILD_FOR_WIN) |
| return SkTypeface::MakeFromName("SimSun", SkFontStyle()); |
| #elif defined(SK_BUILD_FOR_MAC) |
| return SkTypeface::MakeFromName("Hiragino Sans GB W3", SkFontStyle()); |
| #elif defined(SK_BUILD_FOR_IOS) |
| return SkTypeface::MakeFromName("Hiragino Sans GB W3", SkFontStyle()); |
| #elif defined(SK_BUILD_FOR_UNIX) |
| return SkTypeface::MakeFromName("Noto Sans CJK SC", SkFontStyle()); |
| #else |
| return nullptr; |
| #endif |
| } |
| |
| class ChineseFlingView : public Sample { |
| public: |
| ChineseFlingView() : fBlobs(kNumBlobs) {} |
| |
| protected: |
| bool onQuery(Sample::Event* evt) override { |
| if (Sample::TitleQ(*evt)) { |
| Sample::TitleR(evt, "chinese-fling"); |
| return true; |
| } |
| return this->INHERITED::onQuery(evt); |
| } |
| |
| void onDrawContent(SkCanvas* canvas) override { |
| if (!fInitialized) { |
| this->init(); |
| fInitialized = true; |
| } |
| |
| 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 %= fBlobs.count(); |
| } |
| // now "fling" a random amount |
| fIndex += fRand.nextRangeU(5, 20); |
| fIndex %= fBlobs.count(); |
| } |
| |
| private: |
| static constexpr auto kNumBlobs = 200; |
| static constexpr auto kWordLength = 16; |
| |
| void init() { |
| 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, |
| kUTF32_SkTextEncoding, |
| font, |
| 0, |
| 0); |
| |
| fBlobs.emplace_back(builder.make()); |
| } |
| |
| fIndex = 0; |
| } |
| |
| // 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); |
| } |
| } |
| |
| bool fInitialized = false; |
| sk_sp<SkTypeface> fTypeface; |
| SkFontMetrics fMetrics; |
| SkTArray<sk_sp<SkTextBlob>> fBlobs; |
| SkRandom fRand; |
| int fIndex; |
| |
| typedef Sample INHERITED; |
| }; |
| |
| class ChineseZoomView : public Sample { |
| public: |
| ChineseZoomView() : fBlobs(kNumBlobs), fScale(15.0f), fTranslate(0.0f) {} |
| |
| protected: |
| bool onQuery(Sample::Event* evt) override { |
| if (Sample::TitleQ(*evt)) { |
| Sample::TitleR(evt, "chinese-zoom"); |
| return true; |
| } |
| SkUnichar uni; |
| if (Sample::CharQ(*evt, &uni)) { |
| if ('>' == uni) { |
| fScale += 0.125f; |
| return true; |
| } |
| if ('<' == uni) { |
| fScale -= 0.125f; |
| return true; |
| } |
| } |
| return this->INHERITED::onQuery(evt); |
| } |
| |
| void onDrawContent(SkCanvas* canvas) override { |
| bool afterFirstFrame = fInitialized; |
| if (!fInitialized) { |
| this->init(); |
| fInitialized = true; |
| } |
| |
| canvas->clear(0xFFDDDDDD); |
| |
| SkPaint paint; |
| paint.setAntiAlias(true); |
| paint.setColor(0xDE000000); |
| |
| if (afterFirstFrame) { |
| #if SK_SUPPORT_GPU |
| GrContext* grContext = canvas->getGrContext(); |
| if (grContext) { |
| sk_sp<SkImage> image = |
| grContext->priv().testingOnly_getFontAtlasImage( |
| GrMaskFormat::kA8_GrMaskFormat, 0); |
| canvas->drawImageRect(image, |
| SkRect::MakeXYWH(10.0f, 10.0f, 512.0f, 512.0), &paint); |
| image = grContext->priv().testingOnly_getFontAtlasImage( |
| GrMaskFormat::kA8_GrMaskFormat, 1); |
| canvas->drawImageRect(image, |
| SkRect::MakeXYWH(522.0f, 10.0f, 512.f, 512.0f), &paint); |
| image = grContext->priv().testingOnly_getFontAtlasImage( |
| GrMaskFormat::kA8_GrMaskFormat, 2); |
| canvas->drawImageRect(image, |
| SkRect::MakeXYWH(10.0f, 522.0f, 512.0f, 512.0f), &paint); |
| image = grContext->priv().testingOnly_getFontAtlasImage( |
| GrMaskFormat::kA8_GrMaskFormat, 3); |
| canvas->drawImageRect(image, |
| SkRect::MakeXYWH(522.0f, 522.0f, 512.0f, 512.0f), &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); |
| } |
| } |
| |
| private: |
| static constexpr auto kNumBlobs = 8; |
| static constexpr auto kParagraphLength = 175; |
| |
| void init() { |
| 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 = SkTMin(45, paragraphLength - 45); |
| this->createRandomLine(glyphs, currentLineLength); |
| |
| ToolUtils::add_to_text_blob_w_len(&builder, |
| (const char*)glyphs, |
| currentLineLength * 4, |
| kUTF32_SkTextEncoding, |
| font, |
| 0, |
| y); |
| y += fMetrics.fDescent - fMetrics.fAscent + fMetrics.fLeading; |
| paragraphLength -= 45; |
| } |
| fBlobs.emplace_back(builder.make()); |
| } |
| |
| fIndex = 0; |
| } |
| |
| // 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); |
| } |
| } |
| |
| bool fInitialized = false; |
| sk_sp<SkTypeface> fTypeface; |
| SkFontMetrics fMetrics; |
| SkTArray<sk_sp<SkTextBlob>> fBlobs; |
| SkRandom fRand; |
| SkScalar fScale; |
| SkScalar fTranslate; |
| int fIndex; |
| |
| typedef Sample INHERITED; |
| }; |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| DEF_SAMPLE( return new ChineseFlingView(); ) |
| DEF_SAMPLE( return new ChineseZoomView(); ) |