ScaledFont.cpp 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. /*
  2. * Copyright (c) 2022, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Utf32View.h>
  7. #include <AK/Utf8View.h>
  8. #include <LibGfx/Font/ScaledFont.h>
  9. namespace Gfx {
  10. int ScaledFont::width(StringView view) const { return unicode_view_width(Utf8View(view)); }
  11. int ScaledFont::width(Utf8View const& view) const { return unicode_view_width(view); }
  12. int ScaledFont::width(Utf32View const& view) const { return unicode_view_width(view); }
  13. template<typename T>
  14. ALWAYS_INLINE int ScaledFont::unicode_view_width(T const& view) const
  15. {
  16. if (view.is_empty())
  17. return 0;
  18. float width = 0;
  19. float longest_width = 0;
  20. u32 last_code_point = 0;
  21. for (auto code_point : view) {
  22. if (code_point == '\n' || code_point == '\r') {
  23. longest_width = max(width, longest_width);
  24. width = 0;
  25. last_code_point = code_point;
  26. continue;
  27. }
  28. u32 glyph_id = glyph_id_for_code_point(code_point);
  29. auto kerning = glyphs_horizontal_kerning(last_code_point, code_point);
  30. width += kerning + glyph_metrics(glyph_id).advance_width;
  31. last_code_point = code_point;
  32. }
  33. longest_width = max(width, longest_width);
  34. return longest_width;
  35. }
  36. RefPtr<Gfx::Bitmap> ScaledFont::rasterize_glyph(u32 glyph_id) const
  37. {
  38. auto glyph_iterator = m_cached_glyph_bitmaps.find(glyph_id);
  39. if (glyph_iterator != m_cached_glyph_bitmaps.end())
  40. return glyph_iterator->value;
  41. auto glyph_bitmap = m_font->rasterize_glyph(glyph_id, m_x_scale, m_y_scale);
  42. m_cached_glyph_bitmaps.set(glyph_id, glyph_bitmap);
  43. return glyph_bitmap;
  44. }
  45. Gfx::Glyph ScaledFont::glyph(u32 code_point) const
  46. {
  47. auto id = glyph_id_for_code_point(code_point);
  48. auto bitmap = rasterize_glyph(id);
  49. auto metrics = glyph_metrics(id);
  50. return Gfx::Glyph(bitmap, metrics.left_side_bearing, metrics.advance_width, metrics.ascender);
  51. }
  52. u8 ScaledFont::glyph_width(u32 code_point) const
  53. {
  54. auto id = glyph_id_for_code_point(code_point);
  55. auto metrics = glyph_metrics(id);
  56. return metrics.advance_width;
  57. }
  58. int ScaledFont::glyph_or_emoji_width(u32 code_point) const
  59. {
  60. auto id = glyph_id_for_code_point(code_point);
  61. auto metrics = glyph_metrics(id);
  62. return metrics.advance_width;
  63. }
  64. float ScaledFont::glyphs_horizontal_kerning(u32 left_code_point, u32 right_code_point) const
  65. {
  66. if (left_code_point == 0 || right_code_point == 0)
  67. return 0.f;
  68. auto left_glyph_id = glyph_id_for_code_point(left_code_point);
  69. auto right_glyph_id = glyph_id_for_code_point(right_code_point);
  70. if (left_glyph_id == 0 || right_glyph_id == 0)
  71. return 0.f;
  72. return m_font->glyphs_horizontal_kerning(left_glyph_id, right_glyph_id, m_x_scale);
  73. }
  74. u8 ScaledFont::glyph_fixed_width() const
  75. {
  76. return glyph_metrics(glyph_id_for_code_point(' ')).advance_width;
  77. }
  78. Gfx::FontPixelMetrics ScaledFont::pixel_metrics() const
  79. {
  80. auto metrics = m_font->metrics(m_x_scale, m_y_scale);
  81. return Gfx::FontPixelMetrics {
  82. .size = (float)pixel_size(),
  83. .x_height = (float)x_height(),
  84. .advance_of_ascii_zero = (float)glyph_width('0'),
  85. .glyph_spacing = (float)glyph_spacing(),
  86. .ascent = metrics.ascender,
  87. .descent = -metrics.descender,
  88. .line_gap = metrics.line_gap,
  89. };
  90. }
  91. }