/* * Copyright (c) 2024, Aliaksandr Kalenik * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include namespace Web::Painting { PaintableFragment::PaintableFragment(Layout::LineBoxFragment const& fragment) : m_layout_node(fragment.layout_node()) , m_offset(fragment.offset()) , m_size(fragment.size()) , m_baseline(fragment.baseline()) , m_start(fragment.start()) , m_length(fragment.length()) , m_glyph_run(fragment.glyph_run()) { } CSSPixelRect const PaintableFragment::absolute_rect() const { CSSPixelRect rect { {}, size() }; auto const* containing_block = paintable().containing_block(); if (containing_block) rect.set_location(containing_block->absolute_position()); rect.translate_by(offset()); return rect; } int PaintableFragment::text_index_at(CSSPixels x) const { if (!is(*m_layout_node)) return 0; auto& layout_text = verify_cast(layout_node()); auto& font = layout_text.first_available_font(); Utf8View view(layout_text.text_for_rendering().bytes_as_string_view().substring_view(m_start, m_length)); CSSPixels relative_x = x - absolute_rect().x(); CSSPixels glyph_spacing = font.glyph_spacing(); if (relative_x < 0) return 0; CSSPixels width_so_far = 0; for (auto it = view.begin(); it != view.end(); ++it) { auto previous_it = it; CSSPixels glyph_width = CSSPixels::nearest_value_for(font.glyph_or_emoji_width(it)); if ((width_so_far + glyph_width + glyph_spacing / 2) > relative_x) return m_start + view.byte_offset_of(previous_it); width_so_far += glyph_width + glyph_spacing; } return m_start + m_length; } CSSPixelRect PaintableFragment::selection_rect(Gfx::Font const& font) const { if (layout_node().selection_state() == Layout::Node::SelectionState::None) return {}; if (layout_node().selection_state() == Layout::Node::SelectionState::Full) return absolute_rect(); if (!is(layout_node())) return {}; auto selection = layout_node().root().selection(); if (!selection) return {}; auto range = selection->range(); if (!range) return {}; // FIXME: m_start and m_length should be unsigned and then we won't need these casts. auto const start_index = static_cast(m_start); auto const end_index = static_cast(m_start) + static_cast(m_length); auto& layout_text = verify_cast(layout_node()); auto text = layout_text.text_for_rendering().bytes_as_string_view().substring_view(m_start, m_length); if (layout_node().selection_state() == Layout::Node::SelectionState::StartAndEnd) { // we are in the start/end node (both the same) if (start_index > range->end_offset()) return {}; if (end_index < range->start_offset()) return {}; if (range->start_offset() == range->end_offset()) return {}; auto selection_start_in_this_fragment = max(0, range->start_offset() - m_start); auto selection_end_in_this_fragment = min(m_length, range->end_offset() - m_start); auto pixel_distance_to_first_selected_character = CSSPixels::nearest_value_for(font.width(text.substring_view(0, selection_start_in_this_fragment))); auto pixel_width_of_selection = CSSPixels::nearest_value_for(font.width(text.substring_view(selection_start_in_this_fragment, selection_end_in_this_fragment - selection_start_in_this_fragment))) + 1; auto rect = absolute_rect(); rect.set_x(rect.x() + pixel_distance_to_first_selected_character); rect.set_width(pixel_width_of_selection); return rect; } if (layout_node().selection_state() == Layout::Node::SelectionState::Start) { // we are in the start node if (end_index < range->start_offset()) return {}; auto selection_start_in_this_fragment = max(0, range->start_offset() - m_start); auto selection_end_in_this_fragment = m_length; auto pixel_distance_to_first_selected_character = CSSPixels::nearest_value_for(font.width(text.substring_view(0, selection_start_in_this_fragment))); auto pixel_width_of_selection = CSSPixels::nearest_value_for(font.width(text.substring_view(selection_start_in_this_fragment, selection_end_in_this_fragment - selection_start_in_this_fragment))) + 1; auto rect = absolute_rect(); rect.set_x(rect.x() + pixel_distance_to_first_selected_character); rect.set_width(pixel_width_of_selection); return rect; } if (layout_node().selection_state() == Layout::Node::SelectionState::End) { // we are in the end node if (start_index > range->end_offset()) return {}; auto selection_start_in_this_fragment = 0; auto selection_end_in_this_fragment = min(range->end_offset() - m_start, m_length); auto pixel_distance_to_first_selected_character = CSSPixels::nearest_value_for(font.width(text.substring_view(0, selection_start_in_this_fragment))); auto pixel_width_of_selection = CSSPixels::nearest_value_for(font.width(text.substring_view(selection_start_in_this_fragment, selection_end_in_this_fragment - selection_start_in_this_fragment))) + 1; auto rect = absolute_rect(); rect.set_x(rect.x() + pixel_distance_to_first_selected_character); rect.set_width(pixel_width_of_selection); return rect; } return {}; } }