Flutter Impeller
aiks_dl_text_unittests.cc
Go to the documentation of this file.
1 // Copyright 2013 The Flutter Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "display_list/display_list.h"
6 #include "display_list/dl_blend_mode.h"
7 #include "display_list/dl_tile_mode.h"
8 #include "display_list/effects/dl_color_source.h"
9 #include "display_list/effects/dl_mask_filter.h"
10 #include "flutter/display_list/dl_builder.h"
11 #include "flutter/display_list/dl_color.h"
12 #include "flutter/display_list/dl_paint.h"
13 #include "flutter/fml/build_config.h"
15 #include "flutter/testing/testing.h"
17 #include "impeller/entity/entity.h"
20 #include "include/core/SkMatrix.h"
21 #include "include/core/SkRect.h"
22 
24 #include "txt/platform.h"
25 
26 using namespace flutter;
27 /////////////////////////////////////////////////////
28 
29 namespace impeller {
30 namespace testing {
31 
33  bool stroke = false;
34  Scalar font_size = 50;
36  DlColor color = DlColor::kYellow();
37  SkPoint position = SkPoint::Make(100, 200);
38  std::shared_ptr<DlMaskFilter> filter;
39  bool is_subpixel = false;
40 };
41 
42 bool RenderTextInCanvasSkia(const std::shared_ptr<Context>& context,
43  DisplayListBuilder& canvas,
44  const std::string& text,
45  const std::string_view& font_fixture,
46  const TextRenderOptions& options = {}) {
47  // Draw the baseline.
48  DlPaint paint;
49  paint.setColor(DlColor::kAqua().withAlpha(255 * 0.25));
50  canvas.DrawRect(SkRect::MakeXYWH(options.position.x() - 50,
51  options.position.y(), 900, 10),
52  paint);
53 
54  // Mark the point at which the text is drawn.
55  paint.setColor(DlColor::kRed().withAlpha(255 * 0.25));
56  canvas.DrawCircle(options.position, 5.0, paint);
57 
58  // Construct the text blob.
59  auto c_font_fixture = std::string(font_fixture);
60  auto mapping = flutter::testing::OpenFixtureAsSkData(c_font_fixture.c_str());
61  if (!mapping) {
62  return false;
63  }
64  sk_sp<SkFontMgr> font_mgr = txt::GetDefaultFontManager();
65  SkFont sk_font(font_mgr->makeFromData(mapping), options.font_size);
66  if (options.is_subpixel) {
67  sk_font.setSubpixel(true);
68  }
69  auto blob = SkTextBlob::MakeFromString(text.c_str(), sk_font);
70  if (!blob) {
71  return false;
72  }
73 
74  // Create the Impeller text frame and draw it at the designated baseline.
75  auto frame = MakeTextFrameFromTextBlobSkia(blob);
76 
77  DlPaint text_paint;
78  text_paint.setColor(options.color);
79  text_paint.setMaskFilter(options.filter);
80  text_paint.setStrokeWidth(options.stroke_width);
81  text_paint.setDrawStyle(options.stroke ? DlDrawStyle::kStroke
82  : DlDrawStyle::kFill);
83  canvas.DrawTextFrame(frame, options.position.x(), options.position.y(),
84  text_paint);
85  return true;
86 }
87 
88 TEST_P(AiksTest, CanRenderTextFrame) {
89  DisplayListBuilder builder;
90 
91  DlPaint paint;
92  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
93  builder.DrawPaint(paint);
94  ASSERT_TRUE(RenderTextInCanvasSkia(
95  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
96  "Roboto-Regular.ttf"));
97 
98  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
99 }
100 
101 TEST_P(AiksTest, CanRenderTextFrameWithInvertedTransform) {
102  DisplayListBuilder builder;
103 
104  DlPaint paint;
105  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
106  builder.DrawPaint(paint);
107  builder.Translate(1000, 0);
108  builder.Scale(-1, 1);
109 
110  ASSERT_TRUE(RenderTextInCanvasSkia(
111  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
112  "Roboto-Regular.ttf"));
113 
114  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
115 }
116 
117 TEST_P(AiksTest, CanRenderStrokedTextFrame) {
118  DisplayListBuilder builder;
119 
120  DlPaint paint;
121  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
122  builder.DrawPaint(paint);
123 
124  ASSERT_TRUE(RenderTextInCanvasSkia(
125  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
126  "Roboto-Regular.ttf",
127  {
128  .stroke = true,
129  }));
130  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
131 }
132 
133 TEST_P(AiksTest, CanRenderTextStrokeWidth) {
134  DisplayListBuilder builder;
135 
136  DlPaint paint;
137  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
138  builder.DrawPaint(paint);
139 
140  ASSERT_TRUE(RenderTextInCanvasSkia(GetContext(), builder, "LMNOP VWXYZ",
141  "Roboto-Medium.ttf",
142  {
143  .stroke = true,
144  .stroke_width = 4,
145  }));
146  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
147 }
148 
149 TEST_P(AiksTest, CanRenderTextFrameWithHalfScaling) {
150  DisplayListBuilder builder;
151 
152  DlPaint paint;
153  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
154  builder.DrawPaint(paint);
155  builder.Scale(0.5, 0.5);
156 
157  ASSERT_TRUE(RenderTextInCanvasSkia(
158  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
159  "Roboto-Regular.ttf"));
160  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
161 }
162 
163 TEST_P(AiksTest, CanRenderTextFrameWithFractionScaling) {
164  Scalar fine_scale = 0.f;
165  bool is_subpixel = false;
166  auto callback = [&]() -> sk_sp<DisplayList> {
167  if (AiksTest::ImGuiBegin("Controls", nullptr,
168  ImGuiWindowFlags_AlwaysAutoResize)) {
169  ImGui::SliderFloat("Fine Scale", &fine_scale, -1, 1);
170  ImGui::Checkbox("subpixel", &is_subpixel);
171  ImGui::End();
172  }
173 
174  DisplayListBuilder builder;
175  DlPaint paint;
176  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
177  builder.DrawPaint(paint);
178  Scalar scale = 2.625 + fine_scale;
179  builder.Scale(scale, scale);
180  RenderTextInCanvasSkia(GetContext(), builder,
181  "the quick brown fox jumped over the lazy dog!.?",
182  "Roboto-Regular.ttf",
183  TextRenderOptions{.is_subpixel = is_subpixel});
184  return builder.Build();
185  };
186 
187  ASSERT_TRUE(OpenPlaygroundHere(callback));
188 }
189 
190 // https://github.com/flutter/flutter/issues/164958
191 TEST_P(AiksTest, TextRotated180Degrees) {
192  float fpivot[2] = {200 + 30, 200 - 20};
193  float rotation = 180;
194  float foffset[2] = {200, 200};
195 
196  auto callback = [&]() -> sk_sp<DisplayList> {
197  if (AiksTest::ImGuiBegin("Controls", nullptr,
198  ImGuiWindowFlags_AlwaysAutoResize)) {
199  ImGui::SliderFloat("pivotx", &fpivot[0], 0, 300);
200  ImGui::SliderFloat("pivoty", &fpivot[1], 0, 300);
201  ImGui::SliderFloat("rotation", &rotation, 0, 360);
202  ImGui::SliderFloat("foffsetx", &foffset[0], 0, 300);
203  ImGui::SliderFloat("foffsety", &foffset[1], 0, 300);
204  ImGui::End();
205  }
206  DisplayListBuilder builder;
207  builder.Scale(GetContentScale().x, GetContentScale().y);
208  builder.DrawPaint(DlPaint().setColor(DlColor(0xffffeeff)));
209 
210  builder.Save();
211  DlPoint pivot = Point(fpivot[0], fpivot[1]);
212  builder.Translate(pivot.x, pivot.y);
213  builder.Rotate(rotation);
214  builder.Translate(-pivot.x, -pivot.y);
215 
217  GetContext(), builder, "test", "Roboto-Regular.ttf",
219  .color = DlColor::kBlack(),
220  .position = SkPoint::Make(foffset[0], foffset[1]),
221  });
222 
223  builder.Restore();
224  return builder.Build();
225  };
226  ASSERT_TRUE(OpenPlaygroundHere(callback));
227 }
228 
229 TEST_P(AiksTest, TextFrameSubpixelAlignment) {
230  // "Random" numbers between 0 and 1. Hardcoded to avoid flakiness in goldens.
231  std::array<Scalar, 20> phase_offsets = {
232  7.82637e-06, 0.131538, 0.755605, 0.45865, 0.532767,
233  0.218959, 0.0470446, 0.678865, 0.679296, 0.934693,
234  0.383502, 0.519416, 0.830965, 0.0345721, 0.0534616,
235  0.5297, 0.671149, 0.00769819, 0.383416, 0.0668422};
236  auto callback = [&]() -> sk_sp<DisplayList> {
237  static float font_size = 20;
238  static float phase_variation = 0.2;
239  static float speed = 0.5;
240  static float magnitude = 100;
241  if (AiksTest::ImGuiBegin("Controls", nullptr,
242  ImGuiWindowFlags_AlwaysAutoResize)) {
243  ImGui::SliderFloat("Font size", &font_size, 5, 50);
244  ImGui::SliderFloat("Phase variation", &phase_variation, 0, 1);
245  ImGui::SliderFloat("Oscillation speed", &speed, 0, 2);
246  ImGui::SliderFloat("Oscillation magnitude", &magnitude, 0, 300);
247  ImGui::End();
248  }
249 
250  DisplayListBuilder builder;
251  builder.Scale(GetContentScale().x, GetContentScale().y);
252 
253  for (size_t i = 0; i < phase_offsets.size(); i++) {
254  SkPoint position = SkPoint::Make(
255  200 +
256  magnitude * std::sin((-phase_offsets[i] * k2Pi * phase_variation +
257  GetSecondsElapsed() * speed)), //
258  200 + i * font_size * 1.1 //
259  );
261  GetContext(), builder,
262  "the quick brown fox jumped over "
263  "the lazy dog!.?",
264  "Roboto-Regular.ttf",
265  {.font_size = font_size, .position = position})) {
266  return nullptr;
267  }
268  }
269  return builder.Build();
270  };
271 
272  ASSERT_TRUE(OpenPlaygroundHere(callback));
273 }
274 
275 TEST_P(AiksTest, CanRenderItalicizedText) {
276  DisplayListBuilder builder;
277 
278  DlPaint paint;
279  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
280  builder.DrawPaint(paint);
281 
282  ASSERT_TRUE(RenderTextInCanvasSkia(
283  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
284  "HomemadeApple.ttf"));
285  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
286 }
287 
288 static constexpr std::string_view kFontFixture =
289 #if FML_OS_MACOSX
290  "Apple Color Emoji.ttc";
291 #else
292  "NotoColorEmoji.ttf";
293 #endif
294 
295 TEST_P(AiksTest, CanRenderEmojiTextFrame) {
296  DisplayListBuilder builder;
297 
298  DlPaint paint;
299  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
300  builder.DrawPaint(paint);
301 
302  ASSERT_TRUE(RenderTextInCanvasSkia(
303  GetContext(), builder, "😀 😃 😄 😁 😆 😅 😂 🤣 🥲 😊", kFontFixture));
304  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
305 }
306 
307 TEST_P(AiksTest, CanRenderEmojiTextFrameWithBlur) {
308  DisplayListBuilder builder;
309 
310  builder.Scale(GetContentScale().x, GetContentScale().y);
311  DlPaint paint;
312  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
313  builder.DrawPaint(paint);
314 
315  ASSERT_TRUE(RenderTextInCanvasSkia(
316  GetContext(), builder, "😀 😃 😄 😁 😆 😅 😂 🤣 🥲 😊", kFontFixture,
318  .color = DlColor::kBlue(),
319  .filter = DlBlurMaskFilter::Make(DlBlurStyle::kNormal, 4)}));
320  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
321 }
322 
323 TEST_P(AiksTest, CanRenderEmojiTextFrameWithAlpha) {
324  DisplayListBuilder builder;
325 
326  DlPaint paint;
327  paint.setColor(DlColor::ARGB(1, 0.1, 0.1, 0.1));
328  builder.DrawPaint(paint);
329 
330  ASSERT_TRUE(RenderTextInCanvasSkia(
331  GetContext(), builder, "😀 😃 😄 😁 😆 😅 😂 🤣 🥲 😊", kFontFixture,
332  {.color = DlColor::kBlack().modulateOpacity(0.5)}));
333  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
334 }
335 
336 TEST_P(AiksTest, CanRenderTextInSaveLayer) {
337  DisplayListBuilder builder;
338 
339  DlPaint paint;
340  paint.setColor(DlColor::ARGB(0.1, 0.1, 0.1, 0.1));
341  builder.DrawPaint(paint);
342 
343  builder.Translate(100, 100);
344  builder.Scale(0.5, 0.5);
345 
346  // Blend the layer with the parent pass using kClear to expose the coverage.
347  paint.setBlendMode(DlBlendMode::kClear);
348  builder.SaveLayer(nullptr, &paint);
349  ASSERT_TRUE(RenderTextInCanvasSkia(
350  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
351  "Roboto-Regular.ttf"));
352  builder.Restore();
353 
354  // Render the text again over the cleared coverage rect.
355  ASSERT_TRUE(RenderTextInCanvasSkia(
356  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
357  "Roboto-Regular.ttf"));
358 
359  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
360 }
361 
362 TEST_P(AiksTest, CanRenderTextOutsideBoundaries) {
363  DisplayListBuilder builder;
364  builder.Translate(200, 150);
365 
366  // Construct the text blob.
367  auto mapping = flutter::testing::OpenFixtureAsSkData("wtf.otf");
368  ASSERT_NE(mapping, nullptr);
369 
370  Scalar font_size = 80;
371  sk_sp<SkFontMgr> font_mgr = txt::GetDefaultFontManager();
372  SkFont sk_font(font_mgr->makeFromData(mapping), font_size);
373 
374  DlPaint text_paint;
375  text_paint.setColor(DlColor::kBlue().withAlpha(255 * 0.8));
376 
377  struct {
378  SkPoint position;
379  const char* text;
380  } text[] = {{SkPoint::Make(0, 0), "0F0F0F0"},
381  {SkPoint::Make(1, 2), "789"},
382  {SkPoint::Make(1, 3), "456"},
383  {SkPoint::Make(1, 4), "123"},
384  {SkPoint::Make(0, 6), "0F0F0F0"}};
385  for (auto& t : text) {
386  builder.Save();
387  builder.Translate(t.position.x() * font_size * 2,
388  t.position.y() * font_size * 1.1);
389  {
390  auto blob = SkTextBlob::MakeFromString(t.text, sk_font);
391  ASSERT_NE(blob, nullptr);
392  auto frame = MakeTextFrameFromTextBlobSkia(blob);
393  builder.DrawTextFrame(frame, 0, 0, text_paint);
394  }
395  builder.Restore();
396  }
397 
398  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
399 }
400 
401 TEST_P(AiksTest, TextRotated) {
402  DisplayListBuilder builder;
403 
404  builder.Scale(GetContentScale().x, GetContentScale().y);
405  DlPaint paint;
406  paint.setColor(DlColor::ARGB(0.1, 0.1, 0.1, 1.0));
407  builder.DrawPaint(paint);
408 
409  builder.Transform(SkM44::ColMajor(Matrix(0.25, -0.3, 0, -0.002, //
410  0, 0.5, 0, 0, //
411  0, 0, 0.3, 0, //
412  100, 100, 0, 1.3)
413  .m));
414  ASSERT_TRUE(RenderTextInCanvasSkia(
415  GetContext(), builder, "the quick brown fox jumped over the lazy dog!.?",
416  "Roboto-Regular.ttf"));
417 
418  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
419 }
420 
421 TEST_P(AiksTest, DrawScaledTextWithPerspectiveNoSaveLayer) {
422  DisplayListBuilder builder;
423 
424  Matrix matrix = Matrix(1.0, 0.0, 0.0, 0.0, //
425  0.0, 1.0, 0.0, 0.0, //
426  0.0, 0.0, 1.0, 0.01, //
427  0.0, 0.0, 0.0, 1.0) * //
428  Matrix::MakeRotationY({Degrees{10}});
429 
430  builder.Transform(SkM44::ColMajor(matrix.m));
431 
432  ASSERT_TRUE(RenderTextInCanvasSkia(GetContext(), builder, "Hello world",
433  "Roboto-Regular.ttf"));
434  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
435 }
436 
437 TEST_P(AiksTest, DrawScaledTextWithPerspectiveSaveLayer) {
438  DisplayListBuilder builder;
439 
440  Matrix matrix = Matrix(1.0, 0.0, 0.0, 0.0, //
441  0.0, 1.0, 0.0, 0.0, //
442  0.0, 0.0, 1.0, 0.01, //
443  0.0, 0.0, 0.0, 1.0) * //
444  Matrix::MakeRotationY({Degrees{10}});
445 
446  DlPaint save_paint;
447  SkRect window_bounds =
448  SkRect::MakeXYWH(0, 0, GetWindowSize().width, GetWindowSize().height);
449  // Note: bounds were not needed by the AIKS version, which may indicate a bug.
450  builder.SaveLayer(&window_bounds, &save_paint);
451  builder.Transform(SkM44::ColMajor(matrix.m));
452 
453  ASSERT_TRUE(RenderTextInCanvasSkia(GetContext(), builder, "Hello world",
454  "Roboto-Regular.ttf"));
455 
456  builder.Restore();
457  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
458 }
459 
460 TEST_P(AiksTest, CanRenderTextWithLargePerspectiveTransform) {
461  // Verifies that text scales are clamped to work around
462  // https://github.com/flutter/flutter/issues/136112 .
463 
464  DisplayListBuilder builder;
465 
466  DlPaint save_paint;
467  builder.SaveLayer(nullptr, &save_paint);
468  builder.Transform(SkM44::ColMajor(Matrix(2000, 0, 0, 0, //
469  0, 2000, 0, 0, //
470  0, 0, -1, 9000, //
471  0, 0, -1, 7000 //
472  )
473  .m));
474 
475  ASSERT_TRUE(RenderTextInCanvasSkia(GetContext(), builder, "Hello world",
476  "Roboto-Regular.ttf"));
477  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
478 }
479 
480 TEST_P(AiksTest, CanRenderTextWithPerspectiveTransformInSublist) {
481  DisplayListBuilder text_builder;
482  ASSERT_TRUE(RenderTextInCanvasSkia(GetContext(), text_builder, "Hello world",
483  "Roboto-Regular.ttf"));
484  auto text_display_list = text_builder.Build();
485 
486  DisplayListBuilder builder;
487 
488  Matrix matrix = Matrix::MakeRow(2.0, 0.0, 0.0, 0.0, //
489  0.0, 2.0, 0.0, 0.0, //
490  0.0, 0.0, 1.0, 0.0, //
491  0.0, 0.002, 0.0, 1.0);
492 
493  DlPaint save_paint;
494  SkRect window_bounds =
495  SkRect::MakeXYWH(0, 0, GetWindowSize().width, GetWindowSize().height);
496  builder.SaveLayer(&window_bounds, &save_paint);
497  builder.Transform(SkM44::ColMajor(matrix.m));
498  builder.DrawDisplayList(text_display_list, 1.0f);
499  builder.Restore();
500 
501  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
502 }
503 
504 // This currently renders solid blue, as the support for text color sources was
505 // moved into DLDispatching. Path data requires the SkTextBlobs which are not
506 // used in impeller::TextFrames.
507 TEST_P(AiksTest, TextForegroundShaderWithTransform) {
508  auto mapping = flutter::testing::OpenFixtureAsSkData("Roboto-Regular.ttf");
509  ASSERT_NE(mapping, nullptr);
510 
511  Scalar font_size = 100;
512  sk_sp<SkFontMgr> font_mgr = txt::GetDefaultFontManager();
513  SkFont sk_font(font_mgr->makeFromData(mapping), font_size);
514 
515  DlPaint text_paint;
516  text_paint.setColor(DlColor::kBlue());
517 
518  std::vector<DlColor> colors = {DlColor::RGBA(0.9568, 0.2627, 0.2118, 1.0),
519  DlColor::RGBA(0.1294, 0.5882, 0.9529, 1.0)};
520  std::vector<Scalar> stops = {
521  0.0,
522  1.0,
523  };
524  text_paint.setColorSource(DlColorSource::MakeLinear(
525  /*start_point=*/DlPoint(0, 0), //
526  /*end_point=*/DlPoint(100, 100), //
527  /*stop_count=*/2, //
528  /*colors=*/colors.data(), //
529  /*stops=*/stops.data(), //
530  /*tile_mode=*/DlTileMode::kRepeat //
531  ));
532 
533  DisplayListBuilder builder;
534  builder.Translate(100, 100);
535  builder.Rotate(45);
536 
537  auto blob = SkTextBlob::MakeFromString("Hello", sk_font);
538  ASSERT_NE(blob, nullptr);
539  auto frame = MakeTextFrameFromTextBlobSkia(blob);
540  builder.DrawTextFrame(frame, 0, 0, text_paint);
541 
542  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
543 }
544 
545 // Regression test for https://github.com/flutter/flutter/issues/157885.
546 TEST_P(AiksTest, DifferenceClipsMustRenderIdenticallyAcrossBackends) {
547  DisplayListBuilder builder;
548 
549  DlPaint paint;
550  DlColor clear_color(1.0, 0.5, 0.5, 0.5, DlColorSpace::kSRGB);
551  paint.setColor(clear_color);
552  builder.DrawPaint(paint);
553 
554  DlMatrix identity = {
555  1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0,
556  0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0,
557  };
558  builder.Save();
559  builder.Transform(identity);
560 
561  DlRect frame = DlRect::MakeLTRB(1.0, 1.0, 1278.0, 763.0);
562  DlColor white(1.0, 1.0, 1.0, 1.0, DlColorSpace::kSRGB);
563  paint.setColor(white);
564  builder.DrawRect(frame, paint);
565 
566  builder.Save();
567  builder.ClipRect(frame, DlCanvas::ClipOp::kIntersect);
568 
569  DlMatrix rect_xform = {
570  0.8241262, 0.56640625, 0.0, 0.0, -0.56640625, 0.8241262, 0.0, 0.0,
571  0.0, 0.0, 1.0, 0.0, 271.1137, 489.4733, 0.0, 1.0,
572  };
573  builder.Save();
574  builder.Transform(rect_xform);
575 
576  DlRect rect = DlRect::MakeLTRB(0.0, 0.0, 100.0, 100.0);
577  DlColor bluish(1.0, 0.184, 0.501, 0.929, DlColorSpace::kSRGB);
578  paint.setColor(bluish);
579  DlRoundRect rrect = DlRoundRect::MakeRectRadius(rect, 18.0);
580  builder.DrawRoundRect(rrect, paint);
581 
582  builder.Save();
583  builder.ClipRect(rect, DlCanvas::ClipOp::kIntersect);
584  builder.Restore();
585 
586  builder.Restore();
587 
588  DlMatrix path_xform = {
589  1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0,
590  0.0, 0.0, 1.0, 0.0, 675.0, 279.5, 0.0, 1.0,
591  };
592  builder.Save();
593  builder.Transform(path_xform);
594 
595  SkPath path;
596  path.moveTo(87.5, 349.5);
597  path.lineTo(25.0, 29.5);
598  path.lineTo(150.0, 118.0);
599  path.lineTo(25.0, 118.0);
600  path.lineTo(150.0, 29.5);
601  path.close();
602 
603  DlColor fill_color(1.0, 1.0, 0.0, 0.0, DlColorSpace::kSRGB);
604  DlColor stroke_color(1.0, 0.0, 0.0, 0.0, DlColorSpace::kSRGB);
605  paint.setColor(fill_color);
606  paint.setDrawStyle(DlDrawStyle::kFill);
607  builder.DrawPath(DlPath(path), paint);
608 
609  paint.setColor(stroke_color);
610  paint.setStrokeWidth(2.0);
611  paint.setDrawStyle(DlDrawStyle::kStroke);
612  builder.DrawPath(path, paint);
613 
614  builder.Restore();
615  builder.Restore();
616  builder.Restore();
617 
618  ASSERT_TRUE(OpenPlaygroundHere(builder.Build()));
619 }
620 
621 TEST_P(AiksTest, TextContentsMismatchedTransformTest) {
622  AiksContext aiks_context(GetContext(),
623  std::make_shared<TypographerContextSkia>());
624 
625  // Verifies that TextContents only use the scale/transform that is
626  // computed during preroll.
627  constexpr const char* font_fixture = "Roboto-Regular.ttf";
628 
629  // Construct the text blob.
630  auto c_font_fixture = std::string(font_fixture);
631  auto mapping = flutter::testing::OpenFixtureAsSkData(c_font_fixture.c_str());
632  ASSERT_TRUE(mapping);
633 
634  sk_sp<SkFontMgr> font_mgr = txt::GetDefaultFontManager();
635  SkFont sk_font(font_mgr->makeFromData(mapping), 16);
636 
637  auto blob = SkTextBlob::MakeFromString("Hello World", sk_font);
638  ASSERT_TRUE(blob);
639 
640  auto text_frame = MakeTextFrameFromTextBlobSkia(blob);
641 
642  // Simulate recording the text frame during preroll.
643  Matrix preroll_matrix =
644  Matrix::MakeTranslateScale({1.5, 1.5, 1}, {100, 50, 0});
645  Point preroll_point = Point{23, 45};
646  {
647  auto scale = TextFrame::RoundScaledFontSize(
648  (preroll_matrix * Matrix::MakeTranslation(preroll_point))
649  .GetMaxBasisLengthXY());
650 
651  aiks_context.GetContentContext().GetLazyGlyphAtlas()->AddTextFrame(
652  text_frame, //
653  scale, //
654  preroll_point, //
655  preroll_matrix,
656  std::nullopt //
657  );
658  }
659 
660  // Now simulate rendering with a slightly different scale factor.
661  RenderTarget render_target =
662  aiks_context.GetContentContext()
664  ->CreateOffscreenMSAA(*aiks_context.GetContext(), {100, 100}, 1);
665 
666  TextContents text_contents;
667  text_contents.SetTextFrame(text_frame);
668  text_contents.SetOffset(preroll_point);
669  text_contents.SetScale(1.6);
670  text_contents.SetColor(Color::Aqua());
671 
672  Matrix not_preroll_matrix =
673  Matrix::MakeTranslateScale({1.5, 1.5, 1}, {100, 50, 0});
674 
675  Entity entity;
676  entity.SetTransform(not_preroll_matrix);
677 
678  std::shared_ptr<CommandBuffer> command_buffer =
679  aiks_context.GetContext()->CreateCommandBuffer();
680  std::shared_ptr<RenderPass> render_pass =
681  command_buffer->CreateRenderPass(render_target);
682 
683  EXPECT_TRUE(text_contents.Render(aiks_context.GetContentContext(), entity,
684  *render_pass));
685 }
686 
687 } // namespace testing
688 } // namespace impeller
ContentContext & GetContentContext() const
Definition: aiks_context.cc:42
std::shared_ptr< Context > GetContext() const
Definition: aiks_context.cc:38
const std::shared_ptr< LazyGlyphAtlas > & GetLazyGlyphAtlas() const
const std::shared_ptr< RenderTargetAllocator > & GetRenderTargetCache() const
void SetTransform(const Matrix &transform)
Set the global transform matrix for this Entity.
Definition: entity.cc:60
void SetOffset(Vector2 offset)
bool Render(const ContentContext &renderer, const Entity &entity, RenderPass &pass) const override
void SetTextFrame(const std::shared_ptr< TextFrame > &frame)
void SetScale(Scalar scale)
Definition: text_contents.h:58
void SetColor(Color color)
int32_t x
TEST_P(AiksTest, TextContentsMismatchedTransformTest)
bool RenderTextInCanvasSkia(const std::shared_ptr< Context > &context, DisplayListBuilder &canvas, const std::string &text, const std::string_view &font_fixture, const TextRenderOptions &options={})
static constexpr std::string_view kFontFixture
constexpr float k2Pi
Definition: constants.h:29
flutter::DlRect DlRect
Definition: dl_dispatcher.h:25
float Scalar
Definition: scalar.h:18
flutter::DlRoundRect DlRoundRect
Definition: dl_dispatcher.h:27
TPoint< Scalar > Point
Definition: point.h:327
flutter::DlPoint DlPoint
Definition: dl_dispatcher.h:24
flutter::DlPath DlPath
Definition: dl_dispatcher.h:28
std::shared_ptr< TextFrame > MakeTextFrameFromTextBlobSkia(const sk_sp< SkTextBlob > &blob)
const Scalar stroke_width
const Scalar scale
A 4x4 matrix using column-major storage.
Definition: matrix.h:37
Scalar m[16]
Definition: matrix.h:39
constexpr Quad Transform(const Quad &quad) const
Definition: matrix.h:543
std::shared_ptr< DlMaskFilter > filter