1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2020 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "gm/gm.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkCanvas.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkColor.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkFont.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkFontTypes.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPaint.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPoint.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRect.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkScalar.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkShader.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkSize.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypeface.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skparagraph/include/Paragraph.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skparagraph/src/ParagraphBuilderImpl.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "tools/ToolUtils.h"
24*c8dee2aaSAndroid Build Coastguard Worker #include "tools/fonts/FontToolUtils.h"
25*c8dee2aaSAndroid Build Coastguard Worker
26*c8dee2aaSAndroid Build Coastguard Worker static const char* gSpeech = "Five score years ago, a great American, in whose symbolic shadow we stand today, signed the Emancipation Proclamation. This momentous decree came as a great beacon light of hope to millions of Negro slaves who had been seared in the flames of withering injustice. It came as a joyous daybreak to end the long night of their captivity.";
27*c8dee2aaSAndroid Build Coastguard Worker
28*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_UNICODE_ICU_IMPLEMENTATION)
29*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skunicode/include/SkUnicode_icu.h"
30*c8dee2aaSAndroid Build Coastguard Worker #endif
31*c8dee2aaSAndroid Build Coastguard Worker
32*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_UNICODE_LIBGRAPHEME_IMPLEMENTATION)
33*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skunicode/include/SkUnicode_libgrapheme.h"
34*c8dee2aaSAndroid Build Coastguard Worker #endif
35*c8dee2aaSAndroid Build Coastguard Worker
36*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_UNICODE_ICU4X_IMPLEMENTATION)
37*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skunicode/include/SkUnicode_icu4x.h"
38*c8dee2aaSAndroid Build Coastguard Worker #endif
39*c8dee2aaSAndroid Build Coastguard Worker
40*c8dee2aaSAndroid Build Coastguard Worker namespace {
41*c8dee2aaSAndroid Build Coastguard Worker enum ParaFlags {
42*c8dee2aaSAndroid Build Coastguard Worker kTimeLayout = 1 << 0,
43*c8dee2aaSAndroid Build Coastguard Worker kUseUnderline = 1 << 1,
44*c8dee2aaSAndroid Build Coastguard Worker kShowVisitor = 1 << 2,
45*c8dee2aaSAndroid Build Coastguard Worker };
46*c8dee2aaSAndroid Build Coastguard Worker
get_unicode()47*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkUnicode> get_unicode() {
48*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_UNICODE_ICU_IMPLEMENTATION)
49*c8dee2aaSAndroid Build Coastguard Worker if (auto unicode = SkUnicodes::ICU::Make()) {
50*c8dee2aaSAndroid Build Coastguard Worker return unicode;
51*c8dee2aaSAndroid Build Coastguard Worker }
52*c8dee2aaSAndroid Build Coastguard Worker #endif
53*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_UNICODE_LIBGRAPHEME_IMPLEMENTATION)
54*c8dee2aaSAndroid Build Coastguard Worker if (auto unicode = SkUnicodes::Libgrapheme::Make()) {
55*c8dee2aaSAndroid Build Coastguard Worker return unicode;
56*c8dee2aaSAndroid Build Coastguard Worker }
57*c8dee2aaSAndroid Build Coastguard Worker #endif
58*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_UNICODE_ICU4X_IMPLEMENTATION)
59*c8dee2aaSAndroid Build Coastguard Worker if (auto unicode = SkUnicodes::ICU4X::Make()) {
60*c8dee2aaSAndroid Build Coastguard Worker return unicode;
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker #endif
63*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
64*c8dee2aaSAndroid Build Coastguard Worker }
65*c8dee2aaSAndroid Build Coastguard Worker } // namespace
66*c8dee2aaSAndroid Build Coastguard Worker
67*c8dee2aaSAndroid Build Coastguard Worker // TODO: Make it work with ALL possible SkUnicodes
68*c8dee2aaSAndroid Build Coastguard Worker class ParagraphGM : public skiagm::GM {
69*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<skia::textlayout::Paragraph> fPara;
70*c8dee2aaSAndroid Build Coastguard Worker const unsigned fFlags;
71*c8dee2aaSAndroid Build Coastguard Worker
72*c8dee2aaSAndroid Build Coastguard Worker public:
ParagraphGM(unsigned flags)73*c8dee2aaSAndroid Build Coastguard Worker ParagraphGM(unsigned flags) : fFlags(flags) {}
74*c8dee2aaSAndroid Build Coastguard Worker
buildParagraph()75*c8dee2aaSAndroid Build Coastguard Worker void buildParagraph() {
76*c8dee2aaSAndroid Build Coastguard Worker skia::textlayout::TextStyle style;
77*c8dee2aaSAndroid Build Coastguard Worker style.setForegroundColor(SkPaint());
78*c8dee2aaSAndroid Build Coastguard Worker style.setFontFamilies({SkString("sans-serif")});
79*c8dee2aaSAndroid Build Coastguard Worker style.setFontSize(30);
80*c8dee2aaSAndroid Build Coastguard Worker
81*c8dee2aaSAndroid Build Coastguard Worker if (fFlags & kUseUnderline) {
82*c8dee2aaSAndroid Build Coastguard Worker style.setDecoration(skia::textlayout::TextDecoration::kUnderline);
83*c8dee2aaSAndroid Build Coastguard Worker style.setDecorationMode(skia::textlayout::TextDecorationMode::kThrough);
84*c8dee2aaSAndroid Build Coastguard Worker style.setDecorationColor(SK_ColorBLACK);
85*c8dee2aaSAndroid Build Coastguard Worker style.setDecorationThicknessMultiplier(2);
86*c8dee2aaSAndroid Build Coastguard Worker }
87*c8dee2aaSAndroid Build Coastguard Worker
88*c8dee2aaSAndroid Build Coastguard Worker skia::textlayout::ParagraphStyle paraStyle;
89*c8dee2aaSAndroid Build Coastguard Worker paraStyle.setTextStyle(style);
90*c8dee2aaSAndroid Build Coastguard Worker
91*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkFontMgr> fontmgr = ToolUtils::TestFontMgr();
92*c8dee2aaSAndroid Build Coastguard Worker if (fontmgr->countFamilies() == 0) {
93*c8dee2aaSAndroid Build Coastguard Worker fPara = nullptr;
94*c8dee2aaSAndroid Build Coastguard Worker return;
95*c8dee2aaSAndroid Build Coastguard Worker }
96*c8dee2aaSAndroid Build Coastguard Worker auto collection = sk_make_sp<skia::textlayout::FontCollection>();
97*c8dee2aaSAndroid Build Coastguard Worker collection->setDefaultFontManager(std::move(fontmgr));
98*c8dee2aaSAndroid Build Coastguard Worker
99*c8dee2aaSAndroid Build Coastguard Worker auto unicode = get_unicode();
100*c8dee2aaSAndroid Build Coastguard Worker if (!unicode) {
101*c8dee2aaSAndroid Build Coastguard Worker fPara = nullptr;
102*c8dee2aaSAndroid Build Coastguard Worker return;
103*c8dee2aaSAndroid Build Coastguard Worker }
104*c8dee2aaSAndroid Build Coastguard Worker auto builder = skia::textlayout::ParagraphBuilderImpl::make(
105*c8dee2aaSAndroid Build Coastguard Worker paraStyle, collection, unicode);
106*c8dee2aaSAndroid Build Coastguard Worker if (nullptr == builder) {
107*c8dee2aaSAndroid Build Coastguard Worker fPara = nullptr;
108*c8dee2aaSAndroid Build Coastguard Worker return;
109*c8dee2aaSAndroid Build Coastguard Worker }
110*c8dee2aaSAndroid Build Coastguard Worker
111*c8dee2aaSAndroid Build Coastguard Worker builder->addText(gSpeech, strlen(gSpeech));
112*c8dee2aaSAndroid Build Coastguard Worker
113*c8dee2aaSAndroid Build Coastguard Worker fPara = builder->Build();
114*c8dee2aaSAndroid Build Coastguard Worker fPara->layout(400);
115*c8dee2aaSAndroid Build Coastguard Worker }
116*c8dee2aaSAndroid Build Coastguard Worker
117*c8dee2aaSAndroid Build Coastguard Worker protected:
onOnceBeforeDraw()118*c8dee2aaSAndroid Build Coastguard Worker void onOnceBeforeDraw() override {
119*c8dee2aaSAndroid Build Coastguard Worker this->buildParagraph();
120*c8dee2aaSAndroid Build Coastguard Worker }
121*c8dee2aaSAndroid Build Coastguard Worker
getName() const122*c8dee2aaSAndroid Build Coastguard Worker SkString getName() const override {
123*c8dee2aaSAndroid Build Coastguard Worker SkString name;
124*c8dee2aaSAndroid Build Coastguard Worker name.printf("paragraph%s_%s",
125*c8dee2aaSAndroid Build Coastguard Worker fFlags & kTimeLayout ? "_layout" : "",
126*c8dee2aaSAndroid Build Coastguard Worker fFlags & kUseUnderline ? "_underline" : "");
127*c8dee2aaSAndroid Build Coastguard Worker if (fFlags & kShowVisitor) {
128*c8dee2aaSAndroid Build Coastguard Worker name.append("_visitor");
129*c8dee2aaSAndroid Build Coastguard Worker }
130*c8dee2aaSAndroid Build Coastguard Worker return name;
131*c8dee2aaSAndroid Build Coastguard Worker }
132*c8dee2aaSAndroid Build Coastguard Worker
getISize()133*c8dee2aaSAndroid Build Coastguard Worker SkISize getISize() override {
134*c8dee2aaSAndroid Build Coastguard Worker if (fFlags & kShowVisitor) {
135*c8dee2aaSAndroid Build Coastguard Worker return SkISize::Make(810, 420);
136*c8dee2aaSAndroid Build Coastguard Worker }
137*c8dee2aaSAndroid Build Coastguard Worker return SkISize::Make(412, 420);
138*c8dee2aaSAndroid Build Coastguard Worker }
139*c8dee2aaSAndroid Build Coastguard Worker
drawFromVisitor(SkCanvas * canvas,skia::textlayout::Paragraph * para) const140*c8dee2aaSAndroid Build Coastguard Worker void drawFromVisitor(SkCanvas* canvas, skia::textlayout::Paragraph* para) const {
141*c8dee2aaSAndroid Build Coastguard Worker SkPaint p, p2;
142*c8dee2aaSAndroid Build Coastguard Worker p.setColor(0xFF0000FF);
143*c8dee2aaSAndroid Build Coastguard Worker p2.setColor(0xFFFF0000);
144*c8dee2aaSAndroid Build Coastguard Worker p2.setStrokeWidth(4);
145*c8dee2aaSAndroid Build Coastguard Worker p2.setStrokeCap(SkPaint::kSquare_Cap);
146*c8dee2aaSAndroid Build Coastguard Worker SkPaint underp;
147*c8dee2aaSAndroid Build Coastguard Worker underp.setStroke(true);
148*c8dee2aaSAndroid Build Coastguard Worker underp.setStrokeWidth(2);
149*c8dee2aaSAndroid Build Coastguard Worker underp.setAntiAlias(true);
150*c8dee2aaSAndroid Build Coastguard Worker underp.setColor(p.getColor());
151*c8dee2aaSAndroid Build Coastguard Worker const SkScalar GAP = 2;
152*c8dee2aaSAndroid Build Coastguard Worker
153*c8dee2aaSAndroid Build Coastguard Worker para->visit([&](int, const skia::textlayout::Paragraph::VisitorInfo* info) {
154*c8dee2aaSAndroid Build Coastguard Worker if (!info) {
155*c8dee2aaSAndroid Build Coastguard Worker return;
156*c8dee2aaSAndroid Build Coastguard Worker }
157*c8dee2aaSAndroid Build Coastguard Worker canvas->drawGlyphs(info->count, info->glyphs, info->positions, info->origin,
158*c8dee2aaSAndroid Build Coastguard Worker info->font, p);
159*c8dee2aaSAndroid Build Coastguard Worker
160*c8dee2aaSAndroid Build Coastguard Worker if (fFlags & kUseUnderline) {
161*c8dee2aaSAndroid Build Coastguard Worker // Need to modify positions to roll-in the orign
162*c8dee2aaSAndroid Build Coastguard Worker std::vector<SkPoint> pos;
163*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < info->count; ++i) {
164*c8dee2aaSAndroid Build Coastguard Worker pos.push_back({info->origin.fX + info->positions[i].fX,
165*c8dee2aaSAndroid Build Coastguard Worker info->origin.fY + info->positions[i].fY});
166*c8dee2aaSAndroid Build Coastguard Worker }
167*c8dee2aaSAndroid Build Coastguard Worker
168*c8dee2aaSAndroid Build Coastguard Worker const SkScalar X0 = pos[0].fX;
169*c8dee2aaSAndroid Build Coastguard Worker const SkScalar X1 = X0 + info->advanceX;
170*c8dee2aaSAndroid Build Coastguard Worker const SkScalar Y = pos[0].fY;
171*c8dee2aaSAndroid Build Coastguard Worker auto sects = info->font.getIntercepts(info->glyphs, info->count, pos.data(),
172*c8dee2aaSAndroid Build Coastguard Worker Y+1, Y+3);
173*c8dee2aaSAndroid Build Coastguard Worker
174*c8dee2aaSAndroid Build Coastguard Worker SkScalar x0 = X0;
175*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < sects.size(); i += 2) {
176*c8dee2aaSAndroid Build Coastguard Worker SkScalar x1 = sects[i] - GAP;
177*c8dee2aaSAndroid Build Coastguard Worker if (x0 < x1) {
178*c8dee2aaSAndroid Build Coastguard Worker canvas->drawLine(x0, Y+2, x1, Y+2, underp);
179*c8dee2aaSAndroid Build Coastguard Worker }
180*c8dee2aaSAndroid Build Coastguard Worker x0 = sects[i+1] + GAP;
181*c8dee2aaSAndroid Build Coastguard Worker }
182*c8dee2aaSAndroid Build Coastguard Worker canvas->drawLine(x0, Y+2, X1, Y+2, underp);
183*c8dee2aaSAndroid Build Coastguard Worker }
184*c8dee2aaSAndroid Build Coastguard Worker
185*c8dee2aaSAndroid Build Coastguard Worker if ((false)) {
186*c8dee2aaSAndroid Build Coastguard Worker if (info->utf8Starts) {
187*c8dee2aaSAndroid Build Coastguard Worker SkString str;
188*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < info->count; ++i) {
189*c8dee2aaSAndroid Build Coastguard Worker str.appendUnichar(gSpeech[info->utf8Starts[i]]);
190*c8dee2aaSAndroid Build Coastguard Worker }
191*c8dee2aaSAndroid Build Coastguard Worker SkDebugf("'%s'\n", str.c_str());
192*c8dee2aaSAndroid Build Coastguard Worker }
193*c8dee2aaSAndroid Build Coastguard Worker
194*c8dee2aaSAndroid Build Coastguard Worker // show position points
195*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < info->count; ++i) {
196*c8dee2aaSAndroid Build Coastguard Worker auto pos = info->positions[i];
197*c8dee2aaSAndroid Build Coastguard Worker canvas->drawPoint(pos.fX + info->origin.fX, pos.fY + info->origin.fY, p2);
198*c8dee2aaSAndroid Build Coastguard Worker }
199*c8dee2aaSAndroid Build Coastguard Worker }
200*c8dee2aaSAndroid Build Coastguard Worker });
201*c8dee2aaSAndroid Build Coastguard Worker }
202*c8dee2aaSAndroid Build Coastguard Worker
onDraw(SkCanvas * canvas,SkString * errorMsg)203*c8dee2aaSAndroid Build Coastguard Worker DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg) override {
204*c8dee2aaSAndroid Build Coastguard Worker if (nullptr == fPara) {
205*c8dee2aaSAndroid Build Coastguard Worker *errorMsg = "Font manager had no fonts or could not build paragraph.";
206*c8dee2aaSAndroid Build Coastguard Worker return DrawResult::kSkip;
207*c8dee2aaSAndroid Build Coastguard Worker }
208*c8dee2aaSAndroid Build Coastguard Worker
209*c8dee2aaSAndroid Build Coastguard Worker if (fFlags & kShowVisitor) {
210*c8dee2aaSAndroid Build Coastguard Worker canvas->clear(SK_ColorWHITE);
211*c8dee2aaSAndroid Build Coastguard Worker fPara->layout(400);
212*c8dee2aaSAndroid Build Coastguard Worker fPara->paint(canvas, 10, 10);
213*c8dee2aaSAndroid Build Coastguard Worker canvas->translate(400+10, 10);
214*c8dee2aaSAndroid Build Coastguard Worker this->drawFromVisitor(canvas, fPara.get());
215*c8dee2aaSAndroid Build Coastguard Worker return DrawResult::kOk;
216*c8dee2aaSAndroid Build Coastguard Worker }
217*c8dee2aaSAndroid Build Coastguard Worker
218*c8dee2aaSAndroid Build Coastguard Worker const int loop = (this->getMode() == kGM_Mode) ? 1 : 50;
219*c8dee2aaSAndroid Build Coastguard Worker
220*c8dee2aaSAndroid Build Coastguard Worker int parity = 0;
221*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < loop; ++i) {
222*c8dee2aaSAndroid Build Coastguard Worker SkAutoCanvasRestore acr(canvas, true);
223*c8dee2aaSAndroid Build Coastguard Worker
224*c8dee2aaSAndroid Build Coastguard Worker if (fFlags & kTimeLayout) {
225*c8dee2aaSAndroid Build Coastguard Worker fPara->layout(400 + parity);
226*c8dee2aaSAndroid Build Coastguard Worker parity = (parity + 1) & 1;
227*c8dee2aaSAndroid Build Coastguard Worker }
228*c8dee2aaSAndroid Build Coastguard Worker fPara->paint(canvas, 10, 10);
229*c8dee2aaSAndroid Build Coastguard Worker }
230*c8dee2aaSAndroid Build Coastguard Worker // clean up if we've been looping
231*c8dee2aaSAndroid Build Coastguard Worker if (loop > 1) {
232*c8dee2aaSAndroid Build Coastguard Worker canvas->clear(SK_ColorWHITE);
233*c8dee2aaSAndroid Build Coastguard Worker fPara->layout(400);
234*c8dee2aaSAndroid Build Coastguard Worker fPara->paint(canvas, 10, 10);
235*c8dee2aaSAndroid Build Coastguard Worker }
236*c8dee2aaSAndroid Build Coastguard Worker
237*c8dee2aaSAndroid Build Coastguard Worker if ((this->getMode() == kGM_Mode) && (fFlags & kTimeLayout)) {
238*c8dee2aaSAndroid Build Coastguard Worker return DrawResult::kSkip;
239*c8dee2aaSAndroid Build Coastguard Worker }
240*c8dee2aaSAndroid Build Coastguard Worker return DrawResult::kOk;
241*c8dee2aaSAndroid Build Coastguard Worker }
242*c8dee2aaSAndroid Build Coastguard Worker
runAsBench() const243*c8dee2aaSAndroid Build Coastguard Worker bool runAsBench() const override { return true; }
244*c8dee2aaSAndroid Build Coastguard Worker
onAnimate(double)245*c8dee2aaSAndroid Build Coastguard Worker bool onAnimate(double /*nanos*/) override {
246*c8dee2aaSAndroid Build Coastguard Worker return false;
247*c8dee2aaSAndroid Build Coastguard Worker }
248*c8dee2aaSAndroid Build Coastguard Worker
249*c8dee2aaSAndroid Build Coastguard Worker private:
250*c8dee2aaSAndroid Build Coastguard Worker using INHERITED = skiagm::GM;
251*c8dee2aaSAndroid Build Coastguard Worker };
252*c8dee2aaSAndroid Build Coastguard Worker DEF_GM(return new ParagraphGM(0);)
253*c8dee2aaSAndroid Build Coastguard Worker DEF_GM(return new ParagraphGM(kTimeLayout);)
254*c8dee2aaSAndroid Build Coastguard Worker DEF_GM(return new ParagraphGM(kUseUnderline);)
255*c8dee2aaSAndroid Build Coastguard Worker DEF_GM(return new ParagraphGM(kShowVisitor);)
256*c8dee2aaSAndroid Build Coastguard Worker DEF_GM(return new ParagraphGM(kShowVisitor | kUseUnderline);)
257