/* * Copyright (c) 2018-2020, Andreas Kling * Copyright (c) 2021, sin-ack * * SPDX-License-Identifier: BSD-2-Clause */ #include "TextLayout.h" #include "Font/Emoji.h" #include #include #include #include namespace Gfx { static DrawGlyphOrEmoji construct_glyph_or_emoji(size_t index, FloatPoint const& position, Gfx::Font const& font, Span glyph_info, Span input_glyph_info) { if (font.has_color_bitmaps()) { auto cluster_start = glyph_info[index].cluster; auto cluster_end = [&]() -> u32 { if (index + 1 < glyph_info.size()) return glyph_info[index + 1].cluster; return input_glyph_info.last().cluster + 1; }(); Vector cluster; for (size_t j = 0; j < input_glyph_info.size(); ++j) { auto const& glyph = input_glyph_info[j]; if (glyph.cluster >= cluster_end) break; if (glyph.cluster >= cluster_start) cluster.append(glyph.codepoint); } if (auto const* emoji = Emoji::emoji_for_code_points(cluster)) { return DrawEmoji { .position = position, .emoji = emoji, }; } } return DrawGlyph { .position = position, .glyph_id = glyph_info[index].codepoint, }; } void for_each_glyph_position(FloatPoint baseline_start, Utf8View string, Gfx::Font const& font, Function callback, IncludeLeftBearing include_left_bearing, Optional width) { hb_buffer_t* buffer = hb_buffer_create(); ScopeGuard destroy_buffer = [&]() { hb_buffer_destroy(buffer); }; hb_buffer_add_utf8(buffer, reinterpret_cast(string.bytes()), string.byte_length(), 0, -1); hb_buffer_guess_segment_properties(buffer); u32 glyph_count; auto* glyph_info = hb_buffer_get_glyph_infos(buffer, &glyph_count); Vector const input_glyph_info({ glyph_info, glyph_count }); if (input_glyph_info.is_empty()) return; auto* hb_font = font.harfbuzz_font(); hb_shape(hb_font, buffer, nullptr, 0); glyph_info = hb_buffer_get_glyph_infos(buffer, &glyph_count); auto* positions = hb_buffer_get_glyph_positions(buffer, &glyph_count); FloatPoint point = baseline_start; for (size_t i = 0; i < glyph_count; ++i) { auto position = point - FloatPoint { 0, font.pixel_metrics().ascent } + FloatPoint { positions[i].x_offset, positions[i].y_offset } / text_shaping_resolution; if (include_left_bearing == IncludeLeftBearing::Yes) { VERIFY(is(font)); auto bearing = static_cast(font).glyph_metrics(glyph_info[i].codepoint).left_side_bearing; position += FloatPoint { bearing, 0 }; } callback(construct_glyph_or_emoji(i, position, font, { glyph_info, glyph_count }, input_glyph_info.span())); point += FloatPoint { positions[i].x_advance, positions[i].y_advance } / text_shaping_resolution; } if (width.has_value()) *width = point.x(); } }