// Copyright 2019 Google LLC. #include "include/core/SkCanvas.h" #include "include/core/SkColorFilter.h" #include "include/core/SkColorPriv.h" #include "include/core/SkGraphics.h" #include "include/core/SkPath.h" #include "include/core/SkRegion.h" #include "include/core/SkShader.h" #include "include/core/SkStream.h" #include "include/core/SkTextBlob.h" #include "include/core/SkTime.h" #include "include/core/SkTypeface.h" #include "include/effects/SkBlurMaskFilter.h" #include "include/effects/SkGradientShader.h" #include "include/utils/SkRandom.h" #include "modules/skparagraph/include/Paragraph.h" #include "modules/skparagraph/include/TypefaceFontProvider.h" #include "modules/skparagraph/src/ParagraphBuilderImpl.h" #include "modules/skparagraph/src/ParagraphImpl.h" #include "modules/skparagraph/utils/TestFontCollection.h" #include "samplecode/Sample.h" #include "src/core/SkOSFile.h" #include "src/shaders/SkColorShader.h" #include "src/utils/SkOSPath.h" #include "src/utils/SkUTF.h" #include "tools/Resources.h" using namespace skia::textlayout; namespace { class ParagraphView_Base : public Sample { protected: sk_sp getFontCollection() { // If we reset font collection we need to reset paragraph cache static sk_sp fFC = nullptr; if (fFC == nullptr) { fFC = sk_make_sp(GetResourcePath("fonts").c_str(), false, true); } return fFC; } }; sk_sp setgrad(const SkRect& r, SkColor c0, SkColor c1) { SkColor colors[] = {c0, c1}; SkPoint pts[] = {{r.fLeft, r.fTop}, {r.fRight, r.fTop}}; return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp); } /* void writeHtml(const char* name, Paragraph* paragraph) { SkString tmpDir = skiatest::GetTmpDir(); if (!tmpDir.isEmpty()) { SkString path = SkOSPath::Join(tmpDir.c_str(), name); SkFILEWStream file(path.c_str()); file.write(nullptr, 0); } } */ } // namespace class ParagraphView1 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph1"); } void drawTest(SkCanvas* canvas, SkScalar w, SkScalar h, SkColor fg, SkColor bg) { const std::vector< std::tuple> gParagraph = {{"monospace", true, false, 14, SK_ColorWHITE, SK_ColorRED, true, TextDecorationStyle::kDashed}, {"Assyrian", false, false, 20, SK_ColorWHITE, SK_ColorBLUE, false, TextDecorationStyle::kDotted}, {"serif", true, true, 10, SK_ColorWHITE, SK_ColorRED, true, TextDecorationStyle::kDouble}, {"Arial", false, true, 16, SK_ColorGRAY, SK_ColorGREEN, true, TextDecorationStyle::kSolid}, {"sans-serif", false, false, 8, SK_ColorWHITE, SK_ColorRED, false, TextDecorationStyle::kWavy}}; SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); canvas->drawColor(SK_ColorWHITE); SkScalar margin = 20; SkPaint paint; paint.setAntiAlias(true); paint.setColor(fg); SkPaint blue; blue.setColor(SK_ColorBLUE); TextStyle defaultStyle; defaultStyle.setBackgroundColor(blue); defaultStyle.setForegroundColor(paint); ParagraphStyle paraStyle; auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); for (auto i = 1; i < 5; ++i) { defaultStyle.setFontSize(24 * i); paraStyle.setTextStyle(defaultStyle); ParagraphBuilderImpl builder(paraStyle, fontCollection); std::string name = "Paragraph: " + std::to_string(24 * i); builder.addText(name.c_str(), name.length()); for (auto para : gParagraph) { TextStyle style; style.setFontFamilies({SkString(std::get<0>(para).c_str())}); SkFontStyle fontStyle(std::get<1>(para) ? SkFontStyle::Weight::kBold_Weight : SkFontStyle::Weight::kNormal_Weight, SkFontStyle::Width::kNormal_Width, std::get<2>(para) ? SkFontStyle::Slant::kItalic_Slant : SkFontStyle::Slant::kUpright_Slant); style.setFontStyle(fontStyle); style.setFontSize(std::get<3>(para) * i); SkPaint background; background.setColor(std::get<4>(para)); style.setBackgroundColor(background); SkPaint foreground; foreground.setColor(std::get<5>(para)); foreground.setAntiAlias(true); style.setForegroundColor(foreground); if (std::get<6>(para)) { style.addShadow(TextShadow(SK_ColorBLACK, SkPoint::Make(5, 5), 2)); } auto decoration = (i % 4); if (decoration == 3) { decoration = 4; } bool test = (TextDecoration)decoration != TextDecoration::kNoDecoration; std::string deco = std::to_string((int)decoration); if (test) { style.setDecoration((TextDecoration)decoration); style.setDecorationStyle(std::get<7>(para)); style.setDecorationColor(std::get<5>(para)); } builder.pushStyle(style); std::string name = " " + std::get<0>(para) + " " + (std::get<1>(para) ? ", bold" : "") + (std::get<2>(para) ? ", italic" : "") + " " + std::to_string(std::get<3>(para) * i) + (std::get<4>(para) != bg ? ", background" : "") + (std::get<5>(para) != fg ? ", foreground" : "") + (std::get<6>(para) ? ", shadow" : "") + (test ? ", decorations " + deco : "") + ";"; builder.addText(name.c_str(), name.length()); builder.pop(); } auto paragraph = builder.Build(); paragraph->layout(w - margin * 2); paragraph->paint(canvas, margin, margin); canvas->translate(0, paragraph->getHeight()); } } void onDrawContent(SkCanvas* canvas) override { drawTest(canvas, this->width(), this->height(), SK_ColorRED, SK_ColorWHITE); } private: typedef Sample INHERITED; }; class ParagraphView2 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph2"); } void drawCode(SkCanvas* canvas, SkScalar w, SkScalar h) { SkPaint comment; comment.setColor(SK_ColorGRAY); SkPaint constant; constant.setColor(SK_ColorMAGENTA); SkPaint null; null.setColor(SK_ColorMAGENTA); SkPaint literal; literal.setColor(SK_ColorGREEN); SkPaint code; code.setColor(SK_ColorDKGRAY); SkPaint number; number.setColor(SK_ColorBLUE); SkPaint name; name.setColor(SK_ColorRED); SkPaint white; white.setColor(SK_ColorWHITE); TextStyle defaultStyle; defaultStyle.setBackgroundColor(white); defaultStyle.setForegroundColor(code); defaultStyle.setFontFamilies({SkString("monospace")}); defaultStyle.setFontSize(30); ParagraphStyle paraStyle; paraStyle.setTextStyle(defaultStyle); auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); ParagraphBuilderImpl builder(paraStyle, fontCollection); const char* text1 = "RaisedButton"; const char* text2 = "(\n"; const char* text3 = " child: "; const char* text4 = "const"; const char* text5 = "Text"; const char* text6 = "'BUTTON TITLE'"; const char* text7 = "),\n"; builder.pushStyle(style(name)); builder.addText(text1, strlen(text1)); builder.pop(); builder.addText(text2, strlen(text2)); builder.addText(text3, strlen(text3)); builder.pushStyle(style(constant)); builder.addText(text4, strlen(text4)); builder.pop(); builder.addText(" ", 1); builder.pushStyle(style(name)); builder.addText(text5, strlen(text5)); builder.pop(); builder.addText("(", 1); builder.pushStyle(style(literal)); builder.addText(text6, strlen(text6)); builder.pop(); builder.addText(text7, strlen(text7)); auto paragraph = builder.Build(); paragraph->layout(w - 20); paragraph->paint(canvas, 20, 20); } TextStyle style(SkPaint paint) { TextStyle style; paint.setAntiAlias(true); style.setForegroundColor(paint); style.setFontFamilies({SkString("monospace")}); style.setFontSize(30); return style; } void drawText(SkCanvas* canvas, SkScalar w, SkScalar h, std::vector& text, SkColor fg = SK_ColorDKGRAY, SkColor bg = SK_ColorWHITE, const char* ff = "sans-serif", SkScalar fs = 24, size_t lineLimit = 30, const std::u16string& ellipsis = u"\u2026") { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); canvas->drawColor(bg); SkScalar margin = 20; SkPaint paint; paint.setAntiAlias(true); paint.setColor(fg); SkPaint blue; blue.setColor(SK_ColorBLUE); SkPaint background; background.setColor(bg); TextStyle style; style.setBackgroundColor(blue); style.setForegroundColor(paint); style.setFontFamilies({SkString(ff)}); style.setFontStyle(SkFontStyle(SkFontStyle::kMedium_Weight, SkFontStyle::kNormal_Width, SkFontStyle::kUpright_Slant)); style.setFontSize(fs); ParagraphStyle paraStyle; paraStyle.setTextStyle(style); paraStyle.setMaxLines(lineLimit); paraStyle.setEllipsis(ellipsis); TextStyle defaultStyle; defaultStyle.setFontSize(20); paraStyle.setTextStyle(defaultStyle); ParagraphBuilderImpl builder(paraStyle, getFontCollection()); SkPaint foreground; foreground.setColor(fg); style.setForegroundColor(foreground); style.setBackgroundColor(background); for (auto& part : text) { builder.pushStyle(style); builder.addText(part, strlen(part)); builder.pop(); } auto paragraph = builder.Build(); paragraph->layout(w - margin * 2); paragraph->paint(canvas, margin, margin); canvas->translate(0, paragraph->getHeight() + margin); } void drawLine(SkCanvas* canvas, SkScalar w, SkScalar h, const std::string& text, TextAlign align) { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); canvas->drawColor(SK_ColorWHITE); SkScalar margin = 20; SkPaint paint; paint.setAntiAlias(true); paint.setColor(SK_ColorBLUE); SkPaint gray; gray.setColor(SK_ColorLTGRAY); TextStyle style; style.setBackgroundColor(gray); style.setForegroundColor(paint); style.setFontFamilies({SkString("Arial")}); style.setFontSize(30); ParagraphStyle paraStyle; paraStyle.setTextStyle(style); paraStyle.setTextAlign(align); auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); ParagraphBuilderImpl builder(paraStyle, fontCollection); builder.addText(text.c_str(), text.length()); auto paragraph = builder.Build(); paragraph->layout(w - margin * 2); paragraph->layout(w - margin); paragraph->paint(canvas, margin, margin); canvas->translate(0, paragraph->getHeight() + margin); } void onDrawContent(SkCanvas* canvas) override { std::vector cupertino = { "google_logogoogle_gsuper_g_logo 1 " "google_logogoogle_gsuper_g_logo 12 " "google_logogoogle_gsuper_g_logo 123 " "google_logogoogle_gsuper_g_logo 1234 " "google_logogoogle_gsuper_g_logo 12345 " "google_logogoogle_gsuper_g_logo 123456 " "google_logogoogle_gsuper_g_logo 1234567 " "google_logogoogle_gsuper_g_logo 12345678 " "google_logogoogle_gsuper_g_logo 123456789 " "google_logogoogle_gsuper_g_logo 1234567890 " "google_logogoogle_gsuper_g_logo 123456789 " "google_logogoogle_gsuper_g_logo 12345678 " "google_logogoogle_gsuper_g_logo 1234567 " "google_logogoogle_gsuper_g_logo 123456 " "google_logogoogle_gsuper_g_logo 12345 " "google_logogoogle_gsuper_g_logo 1234 " "google_logogoogle_gsuper_g_logo 123 " "google_logogoogle_gsuper_g_logo 12 " "google_logogoogle_gsuper_g_logo 1 " "google_logogoogle_gsuper_g_logo " "google_logogoogle_gsuper_g_logo " "google_logogoogle_gsuper_g_logo " "google_logogoogle_gsuper_g_logo " "google_logogoogle_gsuper_g_logo " "google_logogoogle_gsuper_g_logo"}; std::vector text = { "My neighbor came over to say,\n" "Although not in a neighborly way,\n\n" "That he'd knock me around,\n\n\n" "If I didn't stop the sound,\n\n\n\n" "Of the classical music I play."}; std::vector long_word = { "A_very_very_very_very_very_very_very_very_very_very_very_very_very_very_very_very_" "very_very_very_very_very_very_very_very_very_very_very_very_very_very_very_very_" "very_very_very_very_very_very_very_very_very_very_very_very_very_very_very_very_" "very_very_very_very_very_very_very_long_text"}; std::vector very_long = { "A very very very very very very very very very very very very very very very very " "very very very very very very very very very very very very very very very very " "very very very very very very very very very very very very very very very very " "very very very very very very very long text"}; std::vector very_word = { "A very_very_very_very_very_very_very_very_very_very " "very_very_very_very_very_very_very_very_very_very very very very very very very " "very very very very very very very very very very very very very very very very " "very very very very very very very very very very very very very long text"}; SkScalar width = this->width() / 5; SkScalar height = this->height(); drawText(canvas, width, height, long_word, SK_ColorBLACK, SK_ColorWHITE, "Google Sans", 30); canvas->translate(width, 0); drawText(canvas, width, height, very_long, SK_ColorBLACK, SK_ColorWHITE, "Google Sans", 30); canvas->translate(width, 0); drawText(canvas, width, height, very_word, SK_ColorBLACK, SK_ColorWHITE, "Google Sans", 30); canvas->translate(width, 0); drawText(canvas, width, height / 2, text, SK_ColorBLACK, SK_ColorWHITE, "Roboto", 20, 100, u"\u2026"); canvas->translate(0, height / 2); drawCode(canvas, width, height / 2); canvas->translate(width, -height / 2); drawText(canvas, width, height, cupertino, SK_ColorBLACK, SK_ColorWHITE, "Google Sans", 30); } private: typedef Sample INHERITED; }; class ParagraphView3 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph3"); } void drawLine(SkCanvas* canvas, SkScalar w, SkScalar h, const std::string& text, TextAlign align, size_t lineLimit = std::numeric_limits::max(), bool RTL = false, SkColor background = SK_ColorGRAY, const std::u16string& ellipsis = u"\u2026") { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); canvas->drawColor(SK_ColorWHITE); SkScalar margin = 20; SkPaint paint; paint.setAntiAlias(true); paint.setColor(SK_ColorBLACK); SkPaint gray; gray.setColor(background); SkPaint yellow; yellow.setColor(SK_ColorYELLOW); TextStyle style; style.setBackgroundColor(gray); style.setForegroundColor(paint); style.setFontFamilies({SkString("sans-serif")}); style.setFontSize(30); ParagraphStyle paraStyle; paraStyle.setTextStyle(style); paraStyle.setTextAlign(align); paraStyle.setMaxLines(lineLimit); paraStyle.setEllipsis(ellipsis); // paraStyle.setTextDirection(RTL ? SkTextDirection::rtl : SkTextDirection::ltr); auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); ParagraphBuilderImpl builder(paraStyle, fontCollection); if (RTL) { builder.addText(mirror(text)); } else { builder.addText(normal(text)); } canvas->drawRect(SkRect::MakeXYWH(margin, margin, w - margin * 2, h - margin * 2), yellow); auto paragraph = builder.Build(); paragraph->layout(w - margin * 2); paragraph->paint(canvas, margin, margin); } std::u16string mirror(const std::string& text) { std::u16string result; result += u"\u202E"; // for (auto i = text.size(); i > 0; --i) { // result += text[i - 1]; //} for (auto i = text.size(); i > 0; --i) { auto ch = text[i - 1]; if (ch == ',') { result += u"!"; } else if (ch == '.') { result += u"!"; } else { result += ch; } } result += u"\u202C"; return result; } std::u16string normal(const std::string& text) { std::u16string result; result += u"\u202D"; for (auto ch : text) { result += ch; } result += u"\u202C"; return result; } void onDrawContent(SkCanvas* canvas) override { const std::string options = // { "open-source open-source open-source open-source" }; {"Flutter is an open-source project to help developers " "build high-performance, high-fidelity, mobile apps for " "iOS and Android " "from a single codebase. This design lab is a playground " "and showcase of Flutter's many widgets, behaviors, " "animations, layouts, and more."}; canvas->drawColor(SK_ColorDKGRAY); SkScalar width = this->width() / 4; SkScalar height = this->height() / 2; const std::string line = "World domination is such an ugly phrase - I prefer to call it world optimisation"; drawLine(canvas, width, height, line, TextAlign::kLeft, 1, false, SK_ColorLTGRAY); canvas->translate(width, 0); drawLine(canvas, width, height, line, TextAlign::kRight, 2, false, SK_ColorLTGRAY); canvas->translate(width, 0); drawLine(canvas, width, height, line, TextAlign::kCenter, 3, false, SK_ColorLTGRAY); canvas->translate(width, 0); drawLine(canvas, width, height, line, TextAlign::kJustify, 4, false, SK_ColorLTGRAY); canvas->translate(-width * 3, height); drawLine(canvas, width, height, line, TextAlign::kLeft, 1, true, SK_ColorLTGRAY); canvas->translate(width, 0); drawLine(canvas, width, height, line, TextAlign::kRight, 2, true, SK_ColorLTGRAY); canvas->translate(width, 0); drawLine(canvas, width, height, line, TextAlign::kCenter, 3, true, SK_ColorLTGRAY); canvas->translate(width, 0); drawLine(canvas, width, height, line, TextAlign::kJustify, 4, true, SK_ColorLTGRAY); canvas->translate(width, 0); } private: typedef Sample INHERITED; }; class ParagraphView4 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph4"); } void drawFlutter(SkCanvas* canvas, SkScalar w, SkScalar h, const char* ff = "Google Sans", SkScalar fs = 30, size_t lineLimit = std::numeric_limits::max(), const std::u16string& ellipsis = u"\u2026") { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); SkScalar margin = 20; SkPaint black; black.setAntiAlias(true); black.setColor(SK_ColorBLACK); SkPaint blue; blue.setAntiAlias(true); blue.setColor(SK_ColorBLUE); SkPaint red; red.setAntiAlias(true); red.setColor(SK_ColorRED); SkPaint green; green.setAntiAlias(true); green.setColor(SK_ColorGREEN); SkPaint gray; gray.setColor(SK_ColorLTGRAY); SkPaint yellow; yellow.setColor(SK_ColorYELLOW); SkPaint magenta; magenta.setAntiAlias(true); magenta.setColor(SK_ColorMAGENTA); TextStyle style; style.setFontFamilies({SkString(ff)}); style.setFontSize(fs); TextStyle style0; style0.setForegroundColor(black); style0.setBackgroundColor(gray); style0.setFontFamilies({SkString(ff)}); style0.setFontSize(fs); style0.setDecoration(TextDecoration::kUnderline); style0.setDecorationStyle(TextDecorationStyle::kDouble); style0.setDecorationColor(SK_ColorBLACK); TextStyle style1; style1.setForegroundColor(blue); style1.setBackgroundColor(yellow); style1.setFontFamilies({SkString(ff)}); style1.setFontSize(fs); style1.setDecoration(TextDecoration::kOverline); style1.setDecorationStyle(TextDecorationStyle::kWavy); style1.setDecorationColor(SK_ColorBLACK); TextStyle style2; style2.setForegroundColor(red); style2.setFontFamilies({SkString(ff)}); style2.setFontSize(fs); TextStyle style3; style3.setForegroundColor(green); style3.setFontFamilies({SkString(ff)}); style3.setFontSize(fs); TextStyle style4; style4.setForegroundColor(magenta); style4.setFontFamilies({SkString(ff)}); style4.setFontSize(fs); ParagraphStyle paraStyle; paraStyle.setTextStyle(style); paraStyle.setMaxLines(lineLimit); paraStyle.setEllipsis(ellipsis); const char* logo1 = "google_"; const char* logo2 = "logo"; const char* logo3 = "go"; const char* logo4 = "ogle_logo"; const char* logo5 = "google_lo"; const char* logo6 = "go"; { ParagraphBuilderImpl builder(paraStyle, getFontCollection()); builder.pushStyle(style0); builder.addText(logo1, strlen(logo1)); builder.pop(); builder.pushStyle(style1); builder.addText(logo2, strlen(logo2)); builder.pop(); builder.addText(" ", 1); builder.pushStyle(style0); builder.addText(logo3, strlen(logo3)); builder.pop(); builder.pushStyle(style1); builder.addText(logo4, strlen(logo4)); builder.pop(); builder.addText(" ", 1); builder.pushStyle(style0); builder.addText(logo5, strlen(logo5)); builder.pop(); builder.pushStyle(style1); builder.addText(logo6, strlen(logo6)); builder.pop(); auto paragraph = builder.Build(); paragraph->layout(w - margin * 2); paragraph->paint(canvas, margin, margin); canvas->translate(0, h + margin); } } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); SkScalar width = this->width(); SkScalar height = this->height(); drawFlutter(canvas, width, height / 2); } private: typedef Sample INHERITED; }; class ParagraphView5 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph5"); } void bidi(SkCanvas* canvas, SkScalar w, SkScalar h, const std::u16string& text, const std::u16string& expected, size_t lineLimit = std::numeric_limits::max(), const char* ff = "Roboto", SkScalar fs = 30, const std::u16string& ellipsis = u"\u2026") { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); SkScalar margin = 20; SkPaint black; black.setColor(SK_ColorBLACK); SkPaint gray; gray.setColor(SK_ColorLTGRAY); TextStyle style; style.setForegroundColor(black); style.setFontFamilies({SkString(ff)}); style.setFontSize(fs); TextStyle style0; style0.setForegroundColor(black); style0.setFontFamilies({SkString(ff)}); style0.setFontSize(fs); style0.setFontStyle(SkFontStyle(SkFontStyle::kNormal_Weight, SkFontStyle::kNormal_Width, SkFontStyle::kItalic_Slant)); TextStyle style1; style1.setForegroundColor(gray); style1.setFontFamilies({SkString(ff)}); style1.setFontSize(fs); style1.setFontStyle(SkFontStyle(SkFontStyle::kBold_Weight, SkFontStyle::kNormal_Width, SkFontStyle::kUpright_Slant)); ParagraphStyle paraStyle; paraStyle.setTextStyle(style); paraStyle.setMaxLines(lineLimit); paraStyle.setEllipsis(ellipsis); ParagraphBuilderImpl builder(paraStyle, getFontCollection()); if (text.empty()) { const std::u16string text0 = u"\u202Dabc"; const std::u16string text1 = u"\u202EFED"; const std::u16string text2 = u"\u202Dghi"; const std::u16string text3 = u"\u202ELKJ"; const std::u16string text4 = u"\u202Dmno"; builder.pushStyle(style0); builder.addText(text0); builder.pop(); builder.pushStyle(style1); builder.addText(text1); builder.pop(); builder.pushStyle(style0); builder.addText(text2); builder.pop(); builder.pushStyle(style1); builder.addText(text3); builder.pop(); builder.pushStyle(style0); builder.addText(text4); builder.pop(); } else { // icu::UnicodeString unicode((UChar*) text.data(), SkToS32(text.size())); // std::string str; // unicode.toUTF8String(str); // SkDebugf("Text: %s\n", str.c_str()); builder.addText(text + expected); } auto paragraph = builder.Build(); paragraph->layout(w - margin * 2); paragraph->paint(canvas, margin, margin); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); SkScalar width = this->width(); SkScalar height = this->height() / 8; const std::u16string text1 = u"A \u202ENAC\u202Cner, exceedingly \u202ENAC\u202Cny,\n" "One morning remarked to his granny:\n" "A \u202ENAC\u202Cner \u202ENAC\u202C \u202ENAC\u202C,\n" "Anything that he \u202ENAC\u202C,\n" "But a \u202ENAC\u202Cner \u202ENAC\u202C't \u202ENAC\u202C a \u202ENAC\u202C, " "\u202ENAC\u202C he?"; bidi(canvas, width, height * 3, text1, u"", 5); canvas->translate(0, height * 3); bidi(canvas, width, height, u"\u2067DETALOSI\u2069", u""); canvas->translate(0, height); bidi(canvas, width, height, u"\u202BDEDDEBME\u202C", u""); canvas->translate(0, height); bidi(canvas, width, height, u"\u202EEDIRREVO\u202C", u""); canvas->translate(0, height); bidi(canvas, width, height, u"\u200FTICILPMI\u200E", u""); canvas->translate(0, height); bidi(canvas, width, height, u"123 456 7890 \u202EZYXWV UTS RQP ONM LKJ IHG FED CBA\u202C.", u"", 2); canvas->translate(0, height); // bidi(canvas, width, height, u"", u""); // canvas->translate(0, height); } private: typedef Sample INHERITED; }; class ParagraphView6 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph6"); } void hangingS(SkCanvas* canvas, SkScalar w, SkScalar h, SkScalar fs = 60.0) { auto ff = "HangingS"; canvas->drawColor(SK_ColorLTGRAY); SkPaint black; black.setAntiAlias(true); black.setColor(SK_ColorBLACK); SkPaint blue; blue.setAntiAlias(true); blue.setColor(SK_ColorBLUE); SkPaint red; red.setAntiAlias(true); red.setColor(SK_ColorRED); SkPaint green; green.setAntiAlias(true); green.setColor(SK_ColorGREEN); SkPaint gray; gray.setColor(SK_ColorCYAN); SkPaint yellow; yellow.setColor(SK_ColorYELLOW); SkPaint magenta; magenta.setAntiAlias(true); magenta.setColor(SK_ColorMAGENTA); SkFontStyle fontStyle(SkFontStyle::kBold_Weight, SkFontStyle::kNormal_Width, SkFontStyle::kItalic_Slant); TextStyle style; style.setFontFamilies({SkString(ff)}); style.setFontSize(fs); style.setFontStyle(fontStyle); TextStyle style0; style0.setForegroundColor(black); style0.setBackgroundColor(gray); style0.setFontFamilies({SkString(ff)}); style0.setFontSize(fs); style0.setFontStyle(fontStyle); TextStyle style1; style1.setForegroundColor(blue); style1.setBackgroundColor(yellow); style1.setFontFamilies({SkString(ff)}); style1.setFontSize(fs); style1.setFontStyle(fontStyle); TextStyle style2; style2.setForegroundColor(red); style2.setFontFamilies({SkString(ff)}); style2.setFontSize(fs); style2.setFontStyle(fontStyle); TextStyle style3; style3.setForegroundColor(green); style3.setFontFamilies({SkString(ff)}); style3.setFontSize(fs); style3.setFontStyle(fontStyle); TextStyle style4; style4.setForegroundColor(magenta); style4.setFontFamilies({SkString(ff)}); style4.setFontSize(fs); style4.setFontStyle(fontStyle); ParagraphStyle paraStyle; paraStyle.setTextStyle(style); const char* logo1 = "S"; const char* logo2 = "kia"; const char* logo3 = "Sk"; const char* logo4 = "ia"; const char* logo5 = "Ski"; const char* logo6 = "a"; { ParagraphBuilderImpl builder(paraStyle, getFontCollection()); builder.pushStyle(style0); builder.addText(logo1, strlen(logo1)); builder.pop(); builder.pushStyle(style1); builder.addText(logo2, strlen(logo2)); builder.pop(); builder.addText(" ", 3); builder.pushStyle(style0); builder.addText(logo3, strlen(logo3)); builder.pop(); builder.pushStyle(style1); builder.addText(logo4, strlen(logo4)); builder.pop(); builder.addText(" ", 3); builder.pushStyle(style0); builder.addText(logo5, strlen(logo5)); builder.pop(); builder.pushStyle(style1); builder.addText(logo6, strlen(logo6)); builder.pop(); auto paragraph = builder.Build(); paragraph->layout(w); paragraph->paint(canvas, 40, 40); canvas->translate(0, h); } const char* logo11 = "S"; const char* logo12 = "S"; const char* logo13 = "S"; const char* logo14 = "S"; const char* logo15 = "S"; const char* logo16 = "S"; { ParagraphBuilderImpl builder(paraStyle, getFontCollection()); builder.pushStyle(style0); builder.addText(logo11, strlen(logo1)); builder.pop(); builder.pushStyle(style1); builder.addText(logo12, strlen(logo2)); builder.pop(); builder.addText(" ", 3); builder.pushStyle(style0); builder.addText(logo13, strlen(logo3)); builder.pop(); builder.pushStyle(style1); builder.addText(logo14, strlen(logo4)); builder.pop(); builder.addText(" ", 3); builder.pushStyle(style0); builder.addText(logo15, strlen(logo5)); builder.pop(); builder.pushStyle(style1); builder.addText(logo16, strlen(logo6)); builder.pop(); auto paragraph = builder.Build(); paragraph->layout(w); paragraph->paint(canvas, 40, h); canvas->translate(0, h); } } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); SkScalar width = this->width(); SkScalar height = this->height() / 4; hangingS(canvas, width, height); } private: typedef Sample INHERITED; }; class ParagraphView7 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph7"); } void drawText(SkCanvas* canvas, SkColor background, SkScalar letterSpace, SkScalar w, SkScalar h) { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); canvas->drawColor(background); const char* line = "World domination is such an ugly phrase - I prefer to call it world optimisation."; ParagraphStyle paragraphStyle; paragraphStyle.setTextAlign(TextAlign::kLeft); paragraphStyle.setMaxLines(10); paragraphStyle.turnHintingOff(); TextStyle textStyle; textStyle.setFontFamilies({SkString("Roboto")}); textStyle.setFontSize(30); textStyle.setLetterSpacing(letterSpace); textStyle.setColor(SK_ColorBLACK); textStyle.setFontStyle(SkFontStyle(SkFontStyle::kMedium_Weight, SkFontStyle::kNormal_Width, SkFontStyle::kUpright_Slant)); ParagraphBuilderImpl builder(paragraphStyle, getFontCollection()); builder.pushStyle(textStyle); builder.addText(line, strlen(line)); builder.pop(); auto paragraph = builder.Build(); paragraph->layout(w - 20); paragraph->paint(canvas, 10, 10); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto h = this->height() / 4; auto w = this->width() / 2; drawText(canvas, SK_ColorGRAY, 1, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorLTGRAY, 2, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorCYAN, 3, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorGRAY, 4, w, h); canvas->translate(w, -3 * h); drawText(canvas, SK_ColorYELLOW, 5, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorGREEN, 10, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorRED, 15, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorBLUE, 20, w, h); canvas->translate(0, h); } private: typedef Sample INHERITED; }; class ParagraphView8 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph8"); } void drawText(SkCanvas* canvas, SkColor background, SkScalar wordSpace, SkScalar w, SkScalar h) { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); canvas->drawColor(background); const char* line = "World domination is such an ugly phrase - I prefer to call it world optimisation."; ParagraphStyle paragraphStyle; paragraphStyle.setTextAlign(TextAlign::kLeft); paragraphStyle.setMaxLines(10); paragraphStyle.turnHintingOff(); TextStyle textStyle; textStyle.setFontFamilies({SkString("Roboto")}); textStyle.setFontSize(30); textStyle.setWordSpacing(wordSpace); textStyle.setColor(SK_ColorBLACK); textStyle.setFontStyle(SkFontStyle(SkFontStyle::kMedium_Weight, SkFontStyle::kNormal_Width, SkFontStyle::kUpright_Slant)); ParagraphBuilderImpl builder(paragraphStyle, getFontCollection()); builder.pushStyle(textStyle); builder.addText(line, strlen(line)); builder.pop(); auto paragraph = builder.Build(); paragraph->layout(w - 20); paragraph->paint(canvas, 10, 10); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto h = this->height() / 4; auto w = this->width() / 2; drawText(canvas, SK_ColorGRAY, 1, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorLTGRAY, 2, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorCYAN, 3, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorGRAY, 4, w, h); canvas->translate(w, -3 * h); drawText(canvas, SK_ColorYELLOW, 5, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorGREEN, 10, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorRED, 15, w, h); canvas->translate(0, h); drawText(canvas, SK_ColorBLUE, 20, w, h); canvas->translate(0, h); } private: typedef Sample INHERITED; }; class ParagraphView9 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph9"); } bool onChar(SkUnichar uni) override { switch (uni) { case 'w': ++wordSpacing; return true; case 'q': if (wordSpacing > 0) --wordSpacing; return true; case 'l': ++letterSpacing; return true; case 'k': if (letterSpacing > 0) --letterSpacing; return true; default: break; } return false; } void drawText(SkCanvas* canvas, SkColor background, SkScalar w, SkScalar h) { SkAutoCanvasRestore acr(canvas, true); canvas->clipRect(SkRect::MakeWH(w, h)); canvas->drawColor(background); auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); fontCollection->enableFontFallback(); const char* text = "( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)(" " ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)(" " ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)( ´・‿・`)"; auto multiplier = 5.67; ParagraphStyle paragraphStyle; paragraphStyle.setTextAlign(TextAlign::kLeft); paragraphStyle.setMaxLines(10); paragraphStyle.turnHintingOff(); TextStyle textStyle; textStyle.setFontFamilies({SkString("Roboto")}); textStyle.setFontSize(5 * multiplier); textStyle.setHeight(1.3f); textStyle.setColor(SK_ColorBLACK); textStyle.setFontStyle(SkFontStyle(SkFontStyle::kMedium_Weight, SkFontStyle::kNormal_Width, SkFontStyle::kUpright_Slant)); ParagraphBuilderImpl builder(paragraphStyle, fontCollection); builder.pushStyle(textStyle); builder.addText(text, strlen(text)); builder.pop(); auto paragraph = builder.Build(); paragraph->layout(200 * multiplier); std::vector sizes = {0, 1, 2, 8, 19, 21, 22, 30, 150}; std::vector colors = {SK_ColorBLUE, SK_ColorCYAN, SK_ColorLTGRAY, SK_ColorGREEN, SK_ColorRED, SK_ColorWHITE, SK_ColorYELLOW, SK_ColorMAGENTA}; RectHeightStyle rect_height_style = RectHeightStyle::kTight; RectWidthStyle rect_width_style = RectWidthStyle::kTight; for (size_t i = 0; i < sizes.size() - 1; ++i) { size_t from = sizes[i]; size_t to = sizes[i + 1]; auto boxes = paragraph->getRectsForRange(from, to, rect_height_style, rect_width_style); if (boxes.empty()) { continue; } for (auto& box : boxes) { SkPaint paint; paint.setColor(colors[i % colors.size()]); paint.setShader(setgrad(box.rect, colors[i % colors.size()], SK_ColorWHITE)); canvas->drawRect(box.rect, paint); } } paragraph->paint(canvas, 0, 0); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto h = this->height(); auto w = this->width(); drawText(canvas, SK_ColorGRAY, w, h); } private: typedef Sample INHERITED; SkScalar letterSpacing; SkScalar wordSpacing; }; class ParagraphView10 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph10"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto multiplier = 5.67; const char* text = "English English 字典 字典 😀😃😄 😀😃😄"; auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); fontCollection->enableFontFallback(); ParagraphStyle paragraph_style; paragraph_style.turnHintingOff(); ParagraphBuilderImpl builder(paragraph_style, fontCollection); TextStyle text_style; text_style.setFontFamilies({SkString("Roboto"), SkString("Noto Color Emoji"), SkString("Noto Serif CJK JP")}); text_style.setFontSize(10 * multiplier); text_style.setLetterSpacing(0); text_style.setWordSpacing(0); text_style.setColor(SK_ColorBLACK); text_style.setHeight(1); builder.pushStyle(text_style); builder.addText(text, strlen(text)); builder.pop(); auto paragraph = builder.Build(); paragraph->layout(width()); paragraph->paint(canvas, 0, 0); } private: typedef Sample INHERITED; }; class ParagraphView11 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph11"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto text = "\U0001f469\U0000200D\U0001f469\U0000200D\U0001f466\U0001f469\U0000200D\U0001f469\U0000200D\U0001f467\U0000200D\U0001f467\U0001f1fa\U0001f1f8"; TextStyle text_style; text_style.setFontFamilies({SkString("Ahem")}); text_style.setColor(SK_ColorBLACK); text_style.setFontSize(60); text_style.setLetterSpacing(0); text_style.setWordSpacing(0); ParagraphStyle paragraph_style; paragraph_style.setTextStyle(text_style); auto fontCollection = sk_make_sp(GetResourcePath("fonts").c_str(), true, true); ParagraphBuilderImpl builder(paragraph_style, fontCollection); builder.addText(text, strlen(text)); auto paragraph = builder.Build(); paragraph->layout(1000); paragraph->paint(canvas, 0, 0); struct pair { unsigned fX; unsigned fY; }; pair hit1[] = {{ 0, 8},{1, 33}, {2, 34}, { 3, 19}, {4, 20}, { 5, 21}, { 6, 22 }, { 7, 23 }, {8, 24 }, { 9, 25}, { 10, 26}, { 11, 27}, {12, 28}, { 13, 21}, {14, 22 }, { 15, 23}, {16, 24}, {17, 21}, { 18, 22}, {19, 21}, { 20, 24}, { 21, 23}, }; pair miss[] = {{ 0, 4},{1, 17}, {2, 18}, { 3, 11}, {4, 12}, { 5, 13}, { 6, 14 }, { 7, 15 }, {8, 16 }, { 9, 17}, { 10, 18}, { 11, 19}, {12, 20}, { 13, 17}, {14, 18 }, { 15, 19}, {16, 20}, {17, 19}, { 18, 20}, { 20, 22}, }; auto rects = paragraph->getRectsForRange(7, 9, RectHeightStyle::kTight, RectWidthStyle::kTight); SkPaint paint; paint.setColor(SK_ColorRED); paint.setStyle(SkPaint::kStroke_Style); paint.setAntiAlias(true); paint.setStrokeWidth(1); if (!rects.empty()) { canvas->drawRect(rects[0].rect, paint); } for (auto& query : hit1) { auto rects = paragraph->getRectsForRange(query.fX, query.fY, RectHeightStyle::kTight, RectWidthStyle::kTight); if (rects.size() >= 1 && rects[0].rect.width() > 0) { } else { SkDebugf("+[%d:%d): Bad\n", query.fX, query.fY); } } for (auto& query : miss) { auto miss = paragraph->getRectsForRange(query.fX, query.fY, RectHeightStyle::kTight, RectWidthStyle::kTight); if (miss.empty()) { } else { SkDebugf("-[%d:%d): Bad\n", query.fX, query.fY); } } } private: typedef Sample INHERITED; }; class ParagraphView12 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph12"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); const char* text = "Atwater Peel Sherbrooke Bonaventure Angrignon Peel Côte-des-Neiges"; TextStyle text_style; text_style.setFontFamilies({SkString("Ahem")}); text_style.setColor(SK_ColorBLACK); text_style.setFontSize(16); //text_style.setLetterSpacing(-0.41); StrutStyle strut_style; strut_style.setStrutEnabled(false); ParagraphStyle paragraph_style; paragraph_style.setStrutStyle(strut_style); paragraph_style.setTextStyle(text_style); ParagraphBuilderImpl builder(paragraph_style, getFontCollection()); builder.addText(text); auto paragraph = builder.Build(); paragraph->layout(1095.000000); auto result = paragraph->getRectsForRange(65, 66, RectHeightStyle::kTight, RectWidthStyle::kTight); paragraph->paint(canvas, 0, 0); SkPaint paint; paint.setColor(SK_ColorRED); paint.setStyle(SkPaint::kStroke_Style); paint.setAntiAlias(true); paint.setStrokeWidth(1); canvas->drawRect(result.front().rect, paint); } private: typedef Sample INHERITED; }; class ParagraphView13 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph13"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); const char* text = "This\n" "is a wrapping test. It should wrap at manual newlines, and if softWrap is true, also at spaces."; TextStyle text_style; text_style.setFontFamilies({SkString("Ahem")}); text_style.setColor(SK_ColorBLACK); text_style.setFontSize(10); auto relayout = [&](size_t lines, bool ellipsis, SkScalar width, SkScalar height, SkScalar minWidth, SkScalar maxWidth, SkColor bg) { ParagraphStyle paragraph_style; SkPaint paint; paint.setColor(bg); text_style.setForegroundColor(paint); paragraph_style.setTextStyle(text_style); paragraph_style.setMaxLines(lines); if (ellipsis) { paragraph_style.setEllipsis(u"\u2026"); } ParagraphBuilderImpl builder(paragraph_style, getFontCollection()); builder.addText(text); auto paragraph = builder.Build(); paragraph->layout(50); paragraph->paint(canvas, 0, 0); canvas->translate(0, paragraph->getHeight() + 10); auto result = paragraph->getRectsForRange(0, strlen(text), RectHeightStyle::kTight, RectWidthStyle::kTight); SkPaint background; background.setColor(SK_ColorRED); background.setStyle(SkPaint::kStroke_Style); background.setAntiAlias(true); background.setStrokeWidth(1); canvas->drawRect(result.front().rect, background); SkASSERT(width == paragraph->getMaxWidth()); SkASSERT(height == paragraph->getHeight()); SkASSERT(minWidth == paragraph->getMinIntrinsicWidth()); SkASSERT(maxWidth == paragraph->getMaxIntrinsicWidth()); }; SkPaint paint; paint.setColor(SK_ColorLTGRAY); canvas->drawRect(SkRect::MakeXYWH(0, 0, 50, 500), paint); relayout(1, false, 50, 10, 950, 950, SK_ColorRED); relayout(3, false, 50, 30, 50, 950, SK_ColorBLUE); relayout(std::numeric_limits::max(), false, 50, 200, 50, 950, SK_ColorGREEN); relayout(1, true, 50, 10, 950, 950, SK_ColorYELLOW); relayout(3, true, 50, 30, 50, 950, SK_ColorMAGENTA); relayout(std::numeric_limits::max(), true, 50, 20, 950, 950, SK_ColorCYAN); relayout(1, false, 50, 10, 950, 950, SK_ColorRED); relayout(3, false, 50, 30, 50, 950, SK_ColorBLUE); relayout(std::numeric_limits::max(), false, 50, 200, 50, 950, SK_ColorGREEN); } private: typedef Sample INHERITED; }; class ParagraphView14 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph14"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); TextStyle text_style; text_style.setFontFamilies({SkString("Ahem")}); text_style.setColor(SK_ColorBLACK); text_style.setFontSize(25); text_style.setDecoration((TextDecoration)(TextDecoration::kUnderline | TextDecoration::kOverline | TextDecoration::kLineThrough)); text_style.setDecorationColor(SK_ColorBLUE); text_style.setDecorationStyle(TextDecorationStyle::kWavy); text_style.setDecorationThicknessMultiplier(4.0f); ParagraphStyle paragraph_style; paragraph_style.setTextStyle(text_style); paragraph_style.setTextDirection(TextDirection::kRtl); ParagraphBuilderImpl builder(paragraph_style, getFontCollection()); builder.pushStyle(text_style); builder.addText("Hello, wor!\nabcd."); auto paragraph = builder.Build(); paragraph->layout(300); paragraph->paint(canvas, 0, 0); SkPaint paint; paint.setColor(SK_ColorRED); paint.setStyle(SkPaint::kStroke_Style); paint.setAntiAlias(true); paint.setStrokeWidth(1); canvas->drawRect(SkRect::MakeXYWH(0, 0, 300, 100), paint); } private: typedef Sample INHERITED; }; class ParagraphView15 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph15"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); TextStyle text_style; text_style.setFontFamilies({SkString("abc.ttf")}); text_style.setFontSize(50); auto fontCollection = sk_make_sp(GetResourcePath("fonts").c_str(), false); fontCollection->addFontFromFile("abc/abc.ttf", "abc"); fontCollection->addFontFromFile("abc/abc+grave.ttf", "abc+grave"); fontCollection->addFontFromFile("abc/abc+agrave.ttf", "abc+agrave"); ParagraphStyle paragraph_style; ParagraphBuilderImpl builder(paragraph_style, fontCollection); text_style.setFontFamilies({SkString("abc"), SkString("abc+grave")}); text_style.setColor(SK_ColorBLUE); builder.pushStyle(text_style); builder.addText(u"a\u0300"); text_style.setColor(SK_ColorMAGENTA); builder.pushStyle(text_style); builder.addText(u"à"); text_style.setFontFamilies({SkString("abc"), SkString("abc+agrave")}); text_style.setColor(SK_ColorRED); builder.pushStyle(text_style); builder.addText(u"a\u0300"); text_style.setColor(SK_ColorGREEN); builder.pushStyle(text_style); builder.addText(u"à"); auto paragraph = builder.Build(); paragraph->layout(800); paragraph->paint(canvas, 50, 50); } private: typedef Sample INHERITED; }; class ParagraphView16 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph16"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); const char* text = "content"; ParagraphStyle paragraph_style; paragraph_style.setMaxLines(1); paragraph_style.setEllipsis(u"\u2026"); //auto fontCollection = sk_make_sp(GetResourcePath("fonts").c_str(), false, true); auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); fontCollection->enableFontFallback(); ParagraphBuilderImpl builder(paragraph_style, fontCollection); TextStyle text_style; text_style.setFontFamilies({SkString(".SF Pro Text")}); text_style.setColor(SK_ColorBLACK); text_style.setFontSize(17.0f * 99.0f); text_style.setLetterSpacing(0.41f); builder.pushStyle(text_style); builder.addText(text); auto paragraph = builder.Build(); paragraph->layout(800); paragraph->paint(canvas, 0, 0); } private: typedef Sample INHERITED; }; class ParagraphView17 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph17"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); fontCollection->enableFontFallback(); auto navy = SkColorSetRGB(0, 0, 139); auto ltgray = SkColorSetRGB(211, 211, 211); auto multiplier = 5.67; const char* text = ">Sͬ͑̀͐̈͒̈́̋̎ͮͩ̽̓ͬ̂̆̔͗́̓ͣͧ͊ͫ͛̉͌̐̑ͪ͗̚͝҉̴͉͢k̡̊̓ͫͭͩ͂͊ͨͪͬ̑ͫ̍̌̄͛̌̂̑̂̋̊̔ͫ͛̽̑ͨ̍ͭ̓̀ͪͪ̉͐͗̌̓̃̚͟͝҉̢͏̫̞̙͇͖̮͕̗̟͕͇͚̻͈̣̻̪͉̰̲̣̫ͅͅP̴̅̍͒̿͗͗̇ͩ̃͆͌̀̽͏̧̡͕͖̝̖̼̺̰̣̬͔͖͔̼͙̞̦̫͓̘͜a̸̴̸̴̢̢̨̨̫͍͓̥̼̭̼̻̤̯̙̤̻̠͚̍̌͋̂ͦͨ̽̇͌͌͆̀̽̎͒̄ͪ̐ͦ̈ͫ͐͗̓̚̚͜ͅr͐͐ͤͫ̐ͥ͂̈́̿́ͮ̃͗̓̏ͫ̀̿͏̸̵̧́͘̕͟͝͠͞͠҉̷̧͚͢͟a̓̽̎̄͗̔͛̄̐͊͛ͫ͂͌̂̂̈̈̓̔̅̅̄͊̉́ͪ̑̄͆ͬ̍͆ͭ͋̐ͬ͏̷̵̨̢̩̹̖͓̥̳̰͔̱̬͖̙͓̙͇̀̀̕͜͟͟͢͟͜͠͡g̨̅̇ͦ͋̂ͦͨͭ̓͐͆̏̂͛̉ͧ̑ͫ̐̒͛ͫ̍̒͛́̚҉̷̨̛̛̀͜͢͞҉̩̘̲͍͎̯̹̝̭̗̱͇͉̲̱͔̯̠̹̥̻͉̲̜̤̰̪̗̺̖̺r̷͌̓̇̅ͭ̀̐̃̃ͭ͑͗̉̈̇̈́ͥ̓ͣ́ͤ͂ͤ͂̏͌̆̚҉̴̸̧̢̢̛̫͉̦̥̤̙͈͉͈͉͓̙̗̟̳̜͈̗̺̟̠̠͖͓̖̪͕̠̕̕͝ͅả̸̴̡̡̧͠͞͡͞҉̛̕͟͏̷̘̪̱͈̲͉̞̠̞̪̫͎̲̬̖̀̀͟͝͞͞͠p̛͂̈͐̚͠҉̵̸̡̢̢̩̹͙̯͖̙̙̮̥̙͚̠͔̥̭̮̞̣̪̬̥̠̖̝̥̪͎́̀̕͜͡͡ͅͅh̵̷̵̡̛ͤ̂͌̐̓̐̋̋͊̒̆̽́̀̀̀͢͠͞͞҉̷̸̢̕҉͚̯͖̫̜̞̟̠̱͉̝̲̹̼͉̟͉̩̮͔̤͖̞̭̙̹̬ͅ<"; ParagraphStyle paragraph_style; ParagraphBuilderImpl builder(paragraph_style, fontCollection); SkPaint paint; paint.setColor(ltgray); TextStyle text_style; text_style.setBackgroundColor(paint); text_style.setColor(navy); text_style.setFontFamilies({SkString("Roboto")}); text_style.setFontSize(20 * multiplier); builder.pushStyle(text_style); builder.addText(text); auto paragraph = builder.Build(); paragraph->layout(10000); paragraph->paint(canvas, 0, 0); } private: typedef Sample INHERITED; }; class Zalgo { private: std::u16string COMBINING_DOWN = u"\u0316\u0317\u0318\u0319\u031c\u031d\u031e\u031f\u0320\u0324\u0325\u0326\u0329\u032a\u032b\u032c\u032d\u032e\u032f\u0330\u0331\u0332\u0333\u0339\u033a\u033b\u033c\u0345\u0347\u0348\u0349\u034d\u034e\u0353\u0354\u0355\u0356\u0359\u035a\u0323"; std::u16string COMBINING_UP = u"\u030d\u030e\u0304\u0305\u033f\u0311\u0306\u0310\u0352\u0357\u0351\u0307\u0308\u030a\u0342\u0343\u0344\u034a\u034b\u034c\u0303\u0302\u030c\u0350\u0300\u0301\u030b\u030f\u0312\u0313\u0314\u033d\u0309\u0363\u0364\u0365\u0366\u0367\u0368\u0369\u036a\u036b\u036c\u036d\u036e\u035b\u0346\u031a"; std::u16string COMBINING_MIDDLE = u"\u0315\u031b\u0340\u0341\u0358\u0321\u0322\u0327\u0328\u0334\u0335\u0336\u034f\u035c\u035d\u035e\u035f\u0360\u0362\u0338\u0337\u0361\u0489"; std::u16string randomMarks(std::u16string& combiningMarks) { std::u16string result; auto num = std::rand() % (combiningMarks.size() / 1); for (size_t i = 0; i < num; ++i) { auto index = std::rand() % combiningMarks.size(); result += combiningMarks[index]; } return result; } public: std::u16string zalgo(std::string victim) { std::u16string result; for (auto& c : victim) { result += c; result += randomMarks(COMBINING_UP); result += randomMarks(COMBINING_MIDDLE); result += randomMarks(COMBINING_DOWN); } return result; } }; class ParagraphView18 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph18"); } bool onChar(SkUnichar uni) override { switch (uni) { case ' ': fLimit = 400; return true; case 's': fLimit += 10; return true; case 'f': if (fLimit > 10) { fLimit -= 10; } return true; default: break; } return false; } bool onAnimate(double nanos) override { if (++fIndex > fLimit) { fRedraw = true; fIndex = 0; } else { fRepeat = true; } return true; } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto navy = SkColorSetRGB(0, 0, 139); auto ltgray = SkColorSetRGB(211, 211, 211); auto multiplier = 5.67; auto fontCollection = sk_make_sp(); fontCollection->setDefaultFontManager(SkFontMgr::RefDefault()); fontCollection->enableFontFallback(); ParagraphStyle paragraph_style; TextStyle text_style; text_style.setFontFamilies({SkString("Roboto")}); text_style.setFontSize(20 * multiplier); text_style.setColor(navy); SkPaint paint; paint.setColor(ltgray); text_style.setBackgroundColor(paint); Zalgo zalgo; if (fRedraw || fRepeat) { if (fRedraw || fParagraph.get() == nullptr) { ParagraphBuilderImpl builder(paragraph_style, fontCollection); builder.pushStyle(text_style); auto utf16text = zalgo.zalgo("SkParagraph"); icu::UnicodeString unicode((UChar*)utf16text.data(), SkToS32(utf16text.size())); std::string str; unicode.toUTF8String(str); SkDebugf("Text:>%s<\n", str.data()); builder.addText(utf16text); fParagraph = builder.Build(); } auto impl = static_cast(fParagraph.get()); impl->setState(InternalState::kUnknown); fParagraph->layout(1000); fParagraph->paint(canvas, 300, 200); for (auto& run : impl->runs()) { SkString fontFamily("unresolved"); if (run.font().getTypeface() != nullptr) { run.font().getTypeface()->getFamilyName(&fontFamily); } if (run.font().getTypeface() != nullptr) { for (size_t i = 0; i < run.size(); ++i) { auto glyph = run.glyphs().begin() + i; if (*glyph == 0) { SkDebugf("Run[%d] @pos=%d\n", run.index(), i); SkASSERT(false); } } } else { SkDebugf("Run[%d]: %s\n", run.index(), fontFamily.c_str()); SkASSERT(false); } } fRedraw = false; fRepeat = false; } } private: bool fRedraw = true; bool fRepeat = false; size_t fIndex = 0; size_t fLimit = 20; std::unique_ptr fParagraph; typedef Sample INHERITED; }; class ParagraphView19 : public ParagraphView_Base { protected: SkString name() override { return SkString("Paragraph19"); } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); auto fontCollection = sk_make_sp(GetResourcePath("fonts").c_str(), false, true); const char* text = "Simple\nMultiline\nText"; ParagraphStyle paragraph_style; ParagraphBuilderImpl builder(paragraph_style, fontCollection); SkPaint paint; paint.setColor(SK_ColorLTGRAY); TextStyle text_style; text_style.setBackgroundColor(paint); text_style.setColor(SK_ColorBLUE); text_style.setFontFamilies({SkString("Roboto")}); text_style.setFontSize(20); builder.pushStyle(text_style); builder.addText(text); auto paragraph = builder.Build(); paragraph->layout(500); // Write down the format //auto impl = static_cast(paragraph.get()); //impl->startFormatRecording(SkString("/tmp/format.txt"), canvas->getLocalClipBounds()); canvas->translate(10, 10); canvas->scale(3, 3); paragraph->paint(canvas, 17, 17); //impl->stopFormatRecording(); } private: typedef Sample INHERITED; }; ////////////////////////////////////////////////////////////////////////////// DEF_SAMPLE(return new ParagraphView1();) DEF_SAMPLE(return new ParagraphView2();) DEF_SAMPLE(return new ParagraphView3();) DEF_SAMPLE(return new ParagraphView4();) DEF_SAMPLE(return new ParagraphView5();) DEF_SAMPLE(return new ParagraphView6();) DEF_SAMPLE(return new ParagraphView7();) DEF_SAMPLE(return new ParagraphView8();) DEF_SAMPLE(return new ParagraphView9();) DEF_SAMPLE(return new ParagraphView10();) DEF_SAMPLE(return new ParagraphView11();) DEF_SAMPLE(return new ParagraphView12();) DEF_SAMPLE(return new ParagraphView13();) DEF_SAMPLE(return new ParagraphView14();) DEF_SAMPLE(return new ParagraphView15();) DEF_SAMPLE(return new ParagraphView16();) DEF_SAMPLE(return new ParagraphView17();) DEF_SAMPLE(return new ParagraphView18();) DEF_SAMPLE(return new ParagraphView19();)