xref: /aosp_15_r20/external/skia/modules/skottie/tests/Shaper.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2022 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 "include/core/SkFontMgr.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skottie/include/TextShaper.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skshaper/utils/FactoryHelpers.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "tools/ToolUtils.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "tools/fonts/FontToolUtils.h"
14*c8dee2aaSAndroid Build Coastguard Worker 
15*c8dee2aaSAndroid Build Coastguard Worker using namespace skottie;
16*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(Skottie_Shaper_Clusters,r)17*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Skottie_Shaper_Clusters, r) {
18*c8dee2aaSAndroid Build Coastguard Worker     const SkString text("Foo \rbar \rBaz.");
19*c8dee2aaSAndroid Build Coastguard Worker 
20*c8dee2aaSAndroid Build Coastguard Worker     auto check_clusters = [](skiatest::Reporter* r, const SkString& text, Shaper::Flags flags,
21*c8dee2aaSAndroid Build Coastguard Worker                              const std::vector<size_t>& expected_clusters) {
22*c8dee2aaSAndroid Build Coastguard Worker         const Shaper::TextDesc desc = {
23*c8dee2aaSAndroid Build Coastguard Worker             ToolUtils::CreatePortableTypeface("Serif", SkFontStyle()),
24*c8dee2aaSAndroid Build Coastguard Worker             18,
25*c8dee2aaSAndroid Build Coastguard Worker             0, 18,
26*c8dee2aaSAndroid Build Coastguard Worker             18,
27*c8dee2aaSAndroid Build Coastguard Worker              0,
28*c8dee2aaSAndroid Build Coastguard Worker              0,
29*c8dee2aaSAndroid Build Coastguard Worker             SkTextUtils::Align::kCenter_Align,
30*c8dee2aaSAndroid Build Coastguard Worker             Shaper::VAlign::kTop,
31*c8dee2aaSAndroid Build Coastguard Worker             Shaper::ResizePolicy::kNone,
32*c8dee2aaSAndroid Build Coastguard Worker             Shaper::LinebreakPolicy::kParagraph,
33*c8dee2aaSAndroid Build Coastguard Worker             Shaper::Direction::kLTR,
34*c8dee2aaSAndroid Build Coastguard Worker             Shaper::Capitalization::kNone,
35*c8dee2aaSAndroid Build Coastguard Worker             0,
36*c8dee2aaSAndroid Build Coastguard Worker             flags,
37*c8dee2aaSAndroid Build Coastguard Worker             nullptr,
38*c8dee2aaSAndroid Build Coastguard Worker         };
39*c8dee2aaSAndroid Build Coastguard Worker         const auto result =
40*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::Shape(text, desc, SkRect::MakeWH(1000, 1000), ToolUtils::TestFontMgr(),
41*c8dee2aaSAndroid Build Coastguard Worker                     SkShapers::BestAvailable());
42*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !result.fFragments.empty());
43*c8dee2aaSAndroid Build Coastguard Worker 
44*c8dee2aaSAndroid Build Coastguard Worker         size_t i = 0;
45*c8dee2aaSAndroid Build Coastguard Worker         for (const auto& frag : result.fFragments) {
46*c8dee2aaSAndroid Build Coastguard Worker             const auto& glyphs = frag.fGlyphs;
47*c8dee2aaSAndroid Build Coastguard Worker 
48*c8dee2aaSAndroid Build Coastguard Worker             if (flags & Shaper::kClusters) {
49*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(r, glyphs.fClusters.size() == glyphs.fGlyphIDs.size());
50*c8dee2aaSAndroid Build Coastguard Worker             }
51*c8dee2aaSAndroid Build Coastguard Worker 
52*c8dee2aaSAndroid Build Coastguard Worker             for (const auto& utf_cluster : glyphs.fClusters) {
53*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(r, i < expected_clusters.size());
54*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(r, utf_cluster == expected_clusters[i++]);
55*c8dee2aaSAndroid Build Coastguard Worker             }
56*c8dee2aaSAndroid Build Coastguard Worker         }
57*c8dee2aaSAndroid Build Coastguard Worker 
58*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, i == expected_clusters.size());
59*c8dee2aaSAndroid Build Coastguard Worker     };
60*c8dee2aaSAndroid Build Coastguard Worker 
61*c8dee2aaSAndroid Build Coastguard Worker     check_clusters(r, text, Shaper::kNone, {});
62*c8dee2aaSAndroid Build Coastguard Worker     check_clusters(r, text, Shaper::kFragmentGlyphs, {});
63*c8dee2aaSAndroid Build Coastguard Worker     check_clusters(r, text, Shaper::kClusters,
64*c8dee2aaSAndroid Build Coastguard Worker                    {0, 1, 2, 3,    5, 6, 7, 8,    10, 11, 12, 13});
65*c8dee2aaSAndroid Build Coastguard Worker     check_clusters(r, text, (Shaper::Flags)(Shaper::kClusters | Shaper::kFragmentGlyphs),
66*c8dee2aaSAndroid Build Coastguard Worker                    {0, 1, 2, 3,    5, 6, 7, 8,    10, 11, 12, 13});
67*c8dee2aaSAndroid Build Coastguard Worker }
68*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(Skottie_Shaper_HAlign,reporter)69*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Skottie_Shaper_HAlign, reporter) {
70*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkTypeface> typeface = ToolUtils::DefaultTypeface();
71*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, typeface);
72*c8dee2aaSAndroid Build Coastguard Worker 
73*c8dee2aaSAndroid Build Coastguard Worker     static constexpr struct {
74*c8dee2aaSAndroid Build Coastguard Worker         SkScalar text_size,
75*c8dee2aaSAndroid Build Coastguard Worker                  tolerance;
76*c8dee2aaSAndroid Build Coastguard Worker     } kTestSizes[] = {
77*c8dee2aaSAndroid Build Coastguard Worker         // These gross tolerances are required for the test to pass on NativeFonts bots.
78*c8dee2aaSAndroid Build Coastguard Worker         // Might be worth investigating why we need so much slack.
79*c8dee2aaSAndroid Build Coastguard Worker         {  5, 2.0f },
80*c8dee2aaSAndroid Build Coastguard Worker         { 10, 2.0f },
81*c8dee2aaSAndroid Build Coastguard Worker         { 15, 2.4f },
82*c8dee2aaSAndroid Build Coastguard Worker         { 25, 4.4f },
83*c8dee2aaSAndroid Build Coastguard Worker     };
84*c8dee2aaSAndroid Build Coastguard Worker 
85*c8dee2aaSAndroid Build Coastguard Worker     static constexpr struct {
86*c8dee2aaSAndroid Build Coastguard Worker         SkTextUtils::Align align;
87*c8dee2aaSAndroid Build Coastguard Worker         SkScalar           l_selector,
88*c8dee2aaSAndroid Build Coastguard Worker                            r_selector;
89*c8dee2aaSAndroid Build Coastguard Worker     } kTestAligns[] = {
90*c8dee2aaSAndroid Build Coastguard Worker         { SkTextUtils::  kLeft_Align, 0.0f, 1.0f },
91*c8dee2aaSAndroid Build Coastguard Worker         { SkTextUtils::kCenter_Align, 0.5f, 0.5f },
92*c8dee2aaSAndroid Build Coastguard Worker         { SkTextUtils:: kRight_Align, 1.0f, 0.0f },
93*c8dee2aaSAndroid Build Coastguard Worker     };
94*c8dee2aaSAndroid Build Coastguard Worker 
95*c8dee2aaSAndroid Build Coastguard Worker     const SkString text("Foo, bar.\rBaz.");
96*c8dee2aaSAndroid Build Coastguard Worker     const SkPoint  text_point = SkPoint::Make(100, 100);
97*c8dee2aaSAndroid Build Coastguard Worker 
98*c8dee2aaSAndroid Build Coastguard Worker     for (const auto& tsize : kTestSizes) {
99*c8dee2aaSAndroid Build Coastguard Worker         for (const auto& talign : kTestAligns) {
100*c8dee2aaSAndroid Build Coastguard Worker             const skottie::Shaper::TextDesc desc = {
101*c8dee2aaSAndroid Build Coastguard Worker                 typeface,
102*c8dee2aaSAndroid Build Coastguard Worker                 tsize.text_size,
103*c8dee2aaSAndroid Build Coastguard Worker                 0, tsize.text_size,
104*c8dee2aaSAndroid Build Coastguard Worker                 tsize.text_size,
105*c8dee2aaSAndroid Build Coastguard Worker                 0,
106*c8dee2aaSAndroid Build Coastguard Worker                 0,
107*c8dee2aaSAndroid Build Coastguard Worker                 talign.align,
108*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::VAlign::kTopBaseline,
109*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::ResizePolicy::kNone,
110*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::LinebreakPolicy::kExplicit,
111*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::Direction::kLTR,
112*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::Capitalization::kNone,
113*c8dee2aaSAndroid Build Coastguard Worker                 0,
114*c8dee2aaSAndroid Build Coastguard Worker                 0,
115*c8dee2aaSAndroid Build Coastguard Worker                 nullptr
116*c8dee2aaSAndroid Build Coastguard Worker             };
117*c8dee2aaSAndroid Build Coastguard Worker 
118*c8dee2aaSAndroid Build Coastguard Worker             const auto shape_result =
119*c8dee2aaSAndroid Build Coastguard Worker                     Shaper::Shape(text, desc, text_point, ToolUtils::TestFontMgr(), SkShapers::BestAvailable());
120*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, shape_result.fFragments.size() == 1ul);
121*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, !shape_result.fFragments[0].fGlyphs.fRuns.empty());
122*c8dee2aaSAndroid Build Coastguard Worker 
123*c8dee2aaSAndroid Build Coastguard Worker             const auto shape_bounds = shape_result.computeVisualBounds();
124*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, !shape_bounds.isEmpty());
125*c8dee2aaSAndroid Build Coastguard Worker 
126*c8dee2aaSAndroid Build Coastguard Worker             const auto expected_l = text_point.x() - shape_bounds.width() * talign.l_selector;
127*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter,
128*c8dee2aaSAndroid Build Coastguard Worker                             std::fabs(shape_bounds.left() - expected_l) < tsize.tolerance,
129*c8dee2aaSAndroid Build Coastguard Worker                             "%f %f %f %f %d", shape_bounds.left(), expected_l, tsize.tolerance,
130*c8dee2aaSAndroid Build Coastguard Worker                                               tsize.text_size, talign.align);
131*c8dee2aaSAndroid Build Coastguard Worker 
132*c8dee2aaSAndroid Build Coastguard Worker             const auto expected_r = text_point.x() + shape_bounds.width() * talign.r_selector;
133*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter,
134*c8dee2aaSAndroid Build Coastguard Worker                             std::fabs(shape_bounds.right() - expected_r) < tsize.tolerance,
135*c8dee2aaSAndroid Build Coastguard Worker                             "%f %f %f %f %d", shape_bounds.right(), expected_r, tsize.tolerance,
136*c8dee2aaSAndroid Build Coastguard Worker                                               tsize.text_size, talign.align);
137*c8dee2aaSAndroid Build Coastguard Worker 
138*c8dee2aaSAndroid Build Coastguard Worker         }
139*c8dee2aaSAndroid Build Coastguard Worker     }
140*c8dee2aaSAndroid Build Coastguard Worker }
141*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(Skottie_Shaper_VAlign,reporter)142*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Skottie_Shaper_VAlign, reporter) {
143*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkTypeface> typeface = ToolUtils::DefaultTypeface();
144*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, typeface);
145*c8dee2aaSAndroid Build Coastguard Worker 
146*c8dee2aaSAndroid Build Coastguard Worker     static constexpr struct {
147*c8dee2aaSAndroid Build Coastguard Worker         SkScalar text_size,
148*c8dee2aaSAndroid Build Coastguard Worker                  tolerance;
149*c8dee2aaSAndroid Build Coastguard Worker     } kTestSizes[] = {
150*c8dee2aaSAndroid Build Coastguard Worker         // These gross tolerances are required for the test to pass on NativeFonts bots.
151*c8dee2aaSAndroid Build Coastguard Worker         // Might be worth investigating why we need so much slack.
152*c8dee2aaSAndroid Build Coastguard Worker         {  5, 2.0f },
153*c8dee2aaSAndroid Build Coastguard Worker         { 10, 4.0f },
154*c8dee2aaSAndroid Build Coastguard Worker         { 15, 5.5f },
155*c8dee2aaSAndroid Build Coastguard Worker         { 25, 8.0f },
156*c8dee2aaSAndroid Build Coastguard Worker     };
157*c8dee2aaSAndroid Build Coastguard Worker 
158*c8dee2aaSAndroid Build Coastguard Worker     struct {
159*c8dee2aaSAndroid Build Coastguard Worker         skottie::Shaper::VAlign align;
160*c8dee2aaSAndroid Build Coastguard Worker         SkScalar                topFactor;
161*c8dee2aaSAndroid Build Coastguard Worker     } kTestAligns[] = {
162*c8dee2aaSAndroid Build Coastguard Worker         { skottie::Shaper::VAlign::kHybridTop   , 0.0f },
163*c8dee2aaSAndroid Build Coastguard Worker         { skottie::Shaper::VAlign::kHybridCenter, 0.5f },
164*c8dee2aaSAndroid Build Coastguard Worker         // TODO: any way to test kTopBaseline?
165*c8dee2aaSAndroid Build Coastguard Worker     };
166*c8dee2aaSAndroid Build Coastguard Worker 
167*c8dee2aaSAndroid Build Coastguard Worker     const SkString text("Foo, bar.\rBaz.");
168*c8dee2aaSAndroid Build Coastguard Worker     const auto text_box = SkRect::MakeXYWH(100, 100, 1000, 1000); // large-enough to avoid breaks.
169*c8dee2aaSAndroid Build Coastguard Worker 
170*c8dee2aaSAndroid Build Coastguard Worker 
171*c8dee2aaSAndroid Build Coastguard Worker     for (const auto& tsize : kTestSizes) {
172*c8dee2aaSAndroid Build Coastguard Worker         for (const auto& talign : kTestAligns) {
173*c8dee2aaSAndroid Build Coastguard Worker             const skottie::Shaper::TextDesc desc = {
174*c8dee2aaSAndroid Build Coastguard Worker                 typeface,
175*c8dee2aaSAndroid Build Coastguard Worker                 tsize.text_size,
176*c8dee2aaSAndroid Build Coastguard Worker                 0, tsize.text_size,
177*c8dee2aaSAndroid Build Coastguard Worker                 tsize.text_size,
178*c8dee2aaSAndroid Build Coastguard Worker                 0,
179*c8dee2aaSAndroid Build Coastguard Worker                 0,
180*c8dee2aaSAndroid Build Coastguard Worker                 SkTextUtils::Align::kCenter_Align,
181*c8dee2aaSAndroid Build Coastguard Worker                 talign.align,
182*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::ResizePolicy::kNone,
183*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::LinebreakPolicy::kParagraph,
184*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::Direction::kLTR,
185*c8dee2aaSAndroid Build Coastguard Worker                 Shaper::Capitalization::kNone,
186*c8dee2aaSAndroid Build Coastguard Worker                 0,
187*c8dee2aaSAndroid Build Coastguard Worker                 0,
188*c8dee2aaSAndroid Build Coastguard Worker                 nullptr
189*c8dee2aaSAndroid Build Coastguard Worker             };
190*c8dee2aaSAndroid Build Coastguard Worker 
191*c8dee2aaSAndroid Build Coastguard Worker             const auto shape_result = Shaper::Shape(text, desc, text_box, ToolUtils::TestFontMgr(), SkShapers::BestAvailable());
192*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, shape_result.fFragments.size() == 1ul);
193*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, !shape_result.fFragments[0].fGlyphs.fRuns.empty());
194*c8dee2aaSAndroid Build Coastguard Worker 
195*c8dee2aaSAndroid Build Coastguard Worker             const auto shape_bounds = shape_result.computeVisualBounds();
196*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, !shape_bounds.isEmpty());
197*c8dee2aaSAndroid Build Coastguard Worker 
198*c8dee2aaSAndroid Build Coastguard Worker             const auto v_diff = text_box.height() - shape_bounds.height();
199*c8dee2aaSAndroid Build Coastguard Worker 
200*c8dee2aaSAndroid Build Coastguard Worker             const auto expected_t = text_box.top() + v_diff * talign.topFactor;
201*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter,
202*c8dee2aaSAndroid Build Coastguard Worker                             std::fabs(shape_bounds.top() - expected_t) < tsize.tolerance,
203*c8dee2aaSAndroid Build Coastguard Worker                             "%f %f %f %f %u", shape_bounds.top(), expected_t, tsize.tolerance,
204*c8dee2aaSAndroid Build Coastguard Worker                                               tsize.text_size, SkToU32(talign.align));
205*c8dee2aaSAndroid Build Coastguard Worker 
206*c8dee2aaSAndroid Build Coastguard Worker             const auto expected_b = text_box.bottom() - v_diff * (1 - talign.topFactor);
207*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter,
208*c8dee2aaSAndroid Build Coastguard Worker                             std::fabs(shape_bounds.bottom() - expected_b) < tsize.tolerance,
209*c8dee2aaSAndroid Build Coastguard Worker                             "%f %f %f %f %u", shape_bounds.bottom(), expected_b, tsize.tolerance,
210*c8dee2aaSAndroid Build Coastguard Worker                                               tsize.text_size, SkToU32(talign.align));
211*c8dee2aaSAndroid Build Coastguard Worker         }
212*c8dee2aaSAndroid Build Coastguard Worker     }
213*c8dee2aaSAndroid Build Coastguard Worker }
214*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(Skottie_Shaper_FragmentGlyphs,reporter)215*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Skottie_Shaper_FragmentGlyphs, reporter) {
216*c8dee2aaSAndroid Build Coastguard Worker     skottie::Shaper::TextDesc desc = {
217*c8dee2aaSAndroid Build Coastguard Worker         ToolUtils::DefaultTypeface(),
218*c8dee2aaSAndroid Build Coastguard Worker         18,
219*c8dee2aaSAndroid Build Coastguard Worker         0, 18,
220*c8dee2aaSAndroid Build Coastguard Worker         18,
221*c8dee2aaSAndroid Build Coastguard Worker          0,
222*c8dee2aaSAndroid Build Coastguard Worker          0,
223*c8dee2aaSAndroid Build Coastguard Worker         SkTextUtils::Align::kCenter_Align,
224*c8dee2aaSAndroid Build Coastguard Worker         Shaper::VAlign::kTop,
225*c8dee2aaSAndroid Build Coastguard Worker         Shaper::ResizePolicy::kNone,
226*c8dee2aaSAndroid Build Coastguard Worker         Shaper::LinebreakPolicy::kParagraph,
227*c8dee2aaSAndroid Build Coastguard Worker         Shaper::Direction::kLTR,
228*c8dee2aaSAndroid Build Coastguard Worker         Shaper::Capitalization::kNone,
229*c8dee2aaSAndroid Build Coastguard Worker         0,
230*c8dee2aaSAndroid Build Coastguard Worker         0,
231*c8dee2aaSAndroid Build Coastguard Worker         nullptr
232*c8dee2aaSAndroid Build Coastguard Worker     };
233*c8dee2aaSAndroid Build Coastguard Worker 
234*c8dee2aaSAndroid Build Coastguard Worker     const SkString text("Foo bar baz");
235*c8dee2aaSAndroid Build Coastguard Worker     const auto text_box = SkRect::MakeWH(100, 100);
236*c8dee2aaSAndroid Build Coastguard Worker 
237*c8dee2aaSAndroid Build Coastguard Worker     {
238*c8dee2aaSAndroid Build Coastguard Worker         const auto shape_result = Shaper::Shape(text, desc, text_box, ToolUtils::TestFontMgr(), SkShapers::BestAvailable());
239*c8dee2aaSAndroid Build Coastguard Worker         // Default/consolidated mode => single blob result.
240*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, shape_result.fFragments.size() == 1ul);
241*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!shape_result.fFragments.empty());
242*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, !shape_result.fFragments[0].fGlyphs.fRuns.empty());
243*c8dee2aaSAndroid Build Coastguard Worker     }
244*c8dee2aaSAndroid Build Coastguard Worker 
245*c8dee2aaSAndroid Build Coastguard Worker     {
246*c8dee2aaSAndroid Build Coastguard Worker         desc.fFlags = Shaper::Flags::kFragmentGlyphs;
247*c8dee2aaSAndroid Build Coastguard Worker         const auto shape_result =
248*c8dee2aaSAndroid Build Coastguard Worker                 skottie::Shaper::Shape(text, desc, text_box, ToolUtils::TestFontMgr(), SkShapers::BestAvailable());
249*c8dee2aaSAndroid Build Coastguard Worker         // Fragmented mode => one blob per glyph.
250*c8dee2aaSAndroid Build Coastguard Worker         const size_t expectedSize = text.size();
251*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, shape_result.fFragments.size() == expectedSize);
252*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!shape_result.fFragments.empty());
253*c8dee2aaSAndroid Build Coastguard Worker         for (size_t i = 0; i < expectedSize; ++i) {
254*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, !shape_result.fFragments[i].fGlyphs.fRuns.empty());
255*c8dee2aaSAndroid Build Coastguard Worker         }
256*c8dee2aaSAndroid Build Coastguard Worker     }
257*c8dee2aaSAndroid Build Coastguard Worker }
258*c8dee2aaSAndroid Build Coastguard Worker 
259*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_SHAPER_HARFBUZZ_AVAILABLE) && !defined(SK_BUILD_FOR_WIN)
260*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(Skottie_Shaper_ExplicitFontMgr,reporter)261*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Skottie_Shaper_ExplicitFontMgr, reporter) {
262*c8dee2aaSAndroid Build Coastguard Worker     class CountingFontMgr : public SkFontMgr {
263*c8dee2aaSAndroid Build Coastguard Worker     public:
264*c8dee2aaSAndroid Build Coastguard Worker         size_t fallbackCount() const { return fFallbackCount; }
265*c8dee2aaSAndroid Build Coastguard Worker 
266*c8dee2aaSAndroid Build Coastguard Worker     protected:
267*c8dee2aaSAndroid Build Coastguard Worker         int onCountFamilies() const override { return 0; }
268*c8dee2aaSAndroid Build Coastguard Worker         void onGetFamilyName(int index, SkString* familyName) const override {
269*c8dee2aaSAndroid Build Coastguard Worker             SkDEBUGFAIL("onGetFamilyName called with bad index");
270*c8dee2aaSAndroid Build Coastguard Worker         }
271*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkFontStyleSet> onCreateStyleSet(int index) const override {
272*c8dee2aaSAndroid Build Coastguard Worker             SkDEBUGFAIL("onCreateStyleSet called with bad index");
273*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
274*c8dee2aaSAndroid Build Coastguard Worker         }
275*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkFontStyleSet> onMatchFamily(const char[]) const override {
276*c8dee2aaSAndroid Build Coastguard Worker             return SkFontStyleSet::CreateEmpty();
277*c8dee2aaSAndroid Build Coastguard Worker         }
278*c8dee2aaSAndroid Build Coastguard Worker 
279*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkTypeface> onMatchFamilyStyle(const char[], const SkFontStyle&) const override {
280*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
281*c8dee2aaSAndroid Build Coastguard Worker         }
282*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkTypeface> onMatchFamilyStyleCharacter(const char familyName[],
283*c8dee2aaSAndroid Build Coastguard Worker                                                       const SkFontStyle& style,
284*c8dee2aaSAndroid Build Coastguard Worker                                                       const char* bcp47[],
285*c8dee2aaSAndroid Build Coastguard Worker                                                       int bcp47Count,
286*c8dee2aaSAndroid Build Coastguard Worker                                                       SkUnichar character) const override {
287*c8dee2aaSAndroid Build Coastguard Worker             fFallbackCount++;
288*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
289*c8dee2aaSAndroid Build Coastguard Worker         }
290*c8dee2aaSAndroid Build Coastguard Worker 
291*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkTypeface> onMakeFromData(sk_sp<SkData>, int) const override {
292*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
293*c8dee2aaSAndroid Build Coastguard Worker         }
294*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkTypeface> onMakeFromStreamIndex(std::unique_ptr<SkStreamAsset>, int) const override {
295*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
296*c8dee2aaSAndroid Build Coastguard Worker         }
297*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkTypeface> onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset>,
298*c8dee2aaSAndroid Build Coastguard Worker                                                const SkFontArguments&) const override {
299*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
300*c8dee2aaSAndroid Build Coastguard Worker         }
301*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkTypeface> onMakeFromFile(const char[], int) const override {
302*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
303*c8dee2aaSAndroid Build Coastguard Worker         }
304*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkTypeface> onLegacyMakeTypeface(const char [], SkFontStyle) const override {
305*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
306*c8dee2aaSAndroid Build Coastguard Worker         }
307*c8dee2aaSAndroid Build Coastguard Worker     private:
308*c8dee2aaSAndroid Build Coastguard Worker         mutable size_t fFallbackCount = 0;
309*c8dee2aaSAndroid Build Coastguard Worker     };
310*c8dee2aaSAndroid Build Coastguard Worker 
311*c8dee2aaSAndroid Build Coastguard Worker     auto fontmgr = sk_make_sp<CountingFontMgr>();
312*c8dee2aaSAndroid Build Coastguard Worker 
313*c8dee2aaSAndroid Build Coastguard Worker     skottie::Shaper::TextDesc desc = {
314*c8dee2aaSAndroid Build Coastguard Worker         ToolUtils::DefaultPortableTypeface(),
315*c8dee2aaSAndroid Build Coastguard Worker         18,
316*c8dee2aaSAndroid Build Coastguard Worker         0, 18,
317*c8dee2aaSAndroid Build Coastguard Worker         18,
318*c8dee2aaSAndroid Build Coastguard Worker          0,
319*c8dee2aaSAndroid Build Coastguard Worker          0,
320*c8dee2aaSAndroid Build Coastguard Worker         SkTextUtils::Align::kCenter_Align,
321*c8dee2aaSAndroid Build Coastguard Worker         Shaper::VAlign::kTop,
322*c8dee2aaSAndroid Build Coastguard Worker         Shaper::ResizePolicy::kNone,
323*c8dee2aaSAndroid Build Coastguard Worker         Shaper::LinebreakPolicy::kParagraph,
324*c8dee2aaSAndroid Build Coastguard Worker         Shaper::Direction::kLTR,
325*c8dee2aaSAndroid Build Coastguard Worker         Shaper::Capitalization::kNone,
326*c8dee2aaSAndroid Build Coastguard Worker         0,
327*c8dee2aaSAndroid Build Coastguard Worker         0,
328*c8dee2aaSAndroid Build Coastguard Worker         nullptr
329*c8dee2aaSAndroid Build Coastguard Worker     };
330*c8dee2aaSAndroid Build Coastguard Worker 
331*c8dee2aaSAndroid Build Coastguard Worker     const auto text_box = SkRect::MakeWH(100, 100);
332*c8dee2aaSAndroid Build Coastguard Worker 
333*c8dee2aaSAndroid Build Coastguard Worker     {
334*c8dee2aaSAndroid Build Coastguard Worker         const auto shape_result = Shaper::Shape(SkString("foo bar"), desc, text_box, fontmgr, SkShapers::BestAvailable());
335*c8dee2aaSAndroid Build Coastguard Worker 
336*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, shape_result.fFragments.size() == 1ul);
337*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, !shape_result.fFragments[0].fGlyphs.fRuns.empty());
338*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, fontmgr->fallbackCount() == 0ul);
339*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, shape_result.fMissingGlyphCount == 0);
340*c8dee2aaSAndroid Build Coastguard Worker     }
341*c8dee2aaSAndroid Build Coastguard Worker 
342*c8dee2aaSAndroid Build Coastguard Worker     {
343*c8dee2aaSAndroid Build Coastguard Worker         // An unassigned codepoint should trigger fallback.
344*c8dee2aaSAndroid Build Coastguard Worker         const auto shape_result = skottie::Shaper::Shape(SkString("foo\U000DFFFFbar"),
345*c8dee2aaSAndroid Build Coastguard Worker                                                          desc, text_box, fontmgr, SkShapers::BestAvailable());
346*c8dee2aaSAndroid Build Coastguard Worker 
347*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, shape_result.fFragments.size() == 1ul);
348*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, !shape_result.fFragments[0].fGlyphs.fRuns.empty());
349*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, fontmgr->fallbackCount() == 1ul);
350*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, shape_result.fMissingGlyphCount == 1ul);
351*c8dee2aaSAndroid Build Coastguard Worker     }
352*c8dee2aaSAndroid Build Coastguard Worker }
353*c8dee2aaSAndroid Build Coastguard Worker 
354*c8dee2aaSAndroid Build Coastguard Worker #endif
355*c8dee2aaSAndroid Build Coastguard Worker 
356