mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 12:32:43 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			151 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			151 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2022, the SerenityOS developers.
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/Utf32View.h>
 | |
| #include <AK/Utf8View.h>
 | |
| #include <LibGfx/Font/Emoji.h>
 | |
| #include <LibGfx/Font/ScaledFont.h>
 | |
| 
 | |
| namespace Gfx {
 | |
| 
 | |
| ScaledFont::ScaledFont(NonnullRefPtr<VectorFont> font, float point_width, float point_height, unsigned dpi_x, unsigned dpi_y)
 | |
|     : m_font(move(font))
 | |
|     , m_point_width(point_width)
 | |
|     , m_point_height(point_height)
 | |
| {
 | |
|     float units_per_em = m_font->units_per_em();
 | |
|     m_x_scale = (point_width * dpi_x) / (POINTS_PER_INCH * units_per_em);
 | |
|     m_y_scale = (point_height * dpi_y) / (POINTS_PER_INCH * units_per_em);
 | |
| 
 | |
|     auto metrics = m_font->metrics(m_x_scale, m_y_scale);
 | |
| 
 | |
|     m_pixel_metrics = Gfx::FontPixelMetrics {
 | |
|         .size = (float)pixel_size(),
 | |
|         .x_height = (float)x_height(),
 | |
|         .advance_of_ascii_zero = (float)glyph_width('0'),
 | |
|         .glyph_spacing = (float)glyph_spacing(),
 | |
|         .ascent = metrics.ascender,
 | |
|         .descent = metrics.descender,
 | |
|         .line_gap = metrics.line_gap,
 | |
|     };
 | |
| }
 | |
| 
 | |
| float ScaledFont::width(StringView view) const { return unicode_view_width(Utf8View(view)); }
 | |
| float ScaledFont::width(Utf8View const& view) const { return unicode_view_width(view); }
 | |
| float ScaledFont::width(Utf32View const& view) const { return unicode_view_width(view); }
 | |
| 
 | |
| template<typename T>
 | |
| ALWAYS_INLINE float ScaledFont::unicode_view_width(T const& view) const
 | |
| {
 | |
|     if (view.is_empty())
 | |
|         return 0;
 | |
|     float width = 0;
 | |
|     float longest_width = 0;
 | |
|     u32 last_code_point = 0;
 | |
| 
 | |
|     for (auto it = view.begin(); it != view.end(); last_code_point = *it, ++it) {
 | |
|         auto code_point = *it;
 | |
| 
 | |
|         if (code_point == '\n' || code_point == '\r') {
 | |
|             longest_width = max(width, longest_width);
 | |
|             width = 0;
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         auto kerning = glyphs_horizontal_kerning(last_code_point, code_point);
 | |
|         width += kerning + glyph_or_emoji_width(it);
 | |
|     }
 | |
| 
 | |
|     longest_width = max(width, longest_width);
 | |
|     return longest_width;
 | |
| }
 | |
| 
 | |
| RefPtr<Gfx::Bitmap> ScaledFont::rasterize_glyph(u32 glyph_id, GlyphSubpixelOffset subpixel_offset) const
 | |
| {
 | |
|     GlyphIndexWithSubpixelOffset index { glyph_id, subpixel_offset };
 | |
|     auto glyph_iterator = m_cached_glyph_bitmaps.find(index);
 | |
|     if (glyph_iterator != m_cached_glyph_bitmaps.end())
 | |
|         return glyph_iterator->value;
 | |
| 
 | |
|     auto glyph_bitmap = m_font->rasterize_glyph(glyph_id, m_x_scale, m_y_scale, subpixel_offset);
 | |
|     m_cached_glyph_bitmaps.set(index, glyph_bitmap);
 | |
|     return glyph_bitmap;
 | |
| }
 | |
| 
 | |
| Gfx::Glyph ScaledFont::glyph(u32 code_point) const
 | |
| {
 | |
|     return glyph(code_point, GlyphSubpixelOffset { 0, 0 });
 | |
| }
 | |
| 
 | |
| Gfx::Glyph ScaledFont::glyph(u32 code_point, GlyphSubpixelOffset subpixel_offset) const
 | |
| {
 | |
|     auto id = glyph_id_for_code_point(code_point);
 | |
|     auto bitmap = rasterize_glyph(id, subpixel_offset);
 | |
|     auto metrics = glyph_metrics(id);
 | |
|     return Gfx::Glyph(bitmap, metrics.left_side_bearing, metrics.advance_width, metrics.ascender);
 | |
| }
 | |
| 
 | |
| float ScaledFont::glyph_left_bearing(u32 code_point) const
 | |
| {
 | |
|     auto id = glyph_id_for_code_point(code_point);
 | |
|     return glyph_metrics(id).left_side_bearing;
 | |
| }
 | |
| 
 | |
| float ScaledFont::glyph_width(u32 code_point) const
 | |
| {
 | |
|     auto id = glyph_id_for_code_point(code_point);
 | |
|     auto metrics = glyph_metrics(id);
 | |
|     return metrics.advance_width;
 | |
| }
 | |
| 
 | |
| template<typename CodePointIterator>
 | |
| static float glyph_or_emoji_width_impl(ScaledFont const& font, CodePointIterator& it)
 | |
| {
 | |
|     if (auto const* emoji = Emoji::emoji_for_code_point_iterator(it))
 | |
|         return font.pixel_size() * emoji->width() / emoji->height();
 | |
| 
 | |
|     return font.glyph_width(*it);
 | |
| }
 | |
| 
 | |
| float ScaledFont::glyph_or_emoji_width(Utf8CodePointIterator& it) const
 | |
| {
 | |
|     return glyph_or_emoji_width_impl(*this, it);
 | |
| }
 | |
| 
 | |
| float ScaledFont::glyph_or_emoji_width(Utf32CodePointIterator& it) const
 | |
| {
 | |
|     return glyph_or_emoji_width_impl(*this, it);
 | |
| }
 | |
| 
 | |
| float ScaledFont::glyphs_horizontal_kerning(u32 left_code_point, u32 right_code_point) const
 | |
| {
 | |
|     if (left_code_point == 0 || right_code_point == 0)
 | |
|         return 0.f;
 | |
| 
 | |
|     auto left_glyph_id = glyph_id_for_code_point(left_code_point);
 | |
|     auto right_glyph_id = glyph_id_for_code_point(right_code_point);
 | |
|     if (left_glyph_id == 0 || right_glyph_id == 0)
 | |
|         return 0.f;
 | |
| 
 | |
|     return m_font->glyphs_horizontal_kerning(left_glyph_id, right_glyph_id, m_x_scale);
 | |
| }
 | |
| 
 | |
| u8 ScaledFont::glyph_fixed_width() const
 | |
| {
 | |
|     return glyph_metrics(glyph_id_for_code_point(' ')).advance_width;
 | |
| }
 | |
| 
 | |
| RefPtr<Font> ScaledFont::with_size(float point_size) const
 | |
| {
 | |
|     return adopt_ref(*new Gfx::ScaledFont(*m_font, point_size, point_size));
 | |
| }
 | |
| 
 | |
| Gfx::FontPixelMetrics ScaledFont::pixel_metrics() const
 | |
| {
 | |
|     return m_pixel_metrics;
 | |
| }
 | |
| 
 | |
| }
 | 
