mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-26 09:59:43 +00:00 
			
		
		
		
	Previously, it was possible for an up/down arrow press to place the cursor in the middle of a multi-code point grapheme cluster. We want to prevent this in a way that matches the behavior of other browsers. Both Chrome and Firefox will map the starting position to a visually equivalent position in the target line with harfbuzz and ICU segmenters. The need for this is explained in a code comment. The result is a much more natural feeling of text navigation.
		
			
				
	
	
		
			84 lines
		
	
	
	
		
			2.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			84 lines
		
	
	
	
		
			2.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <andreas@ladybird.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/Utf16String.h>
 | |
| #include <AK/Utf16View.h>
 | |
| #include <LibGfx/TextLayout.h>
 | |
| #include <LibUnicode/Segmenter.h>
 | |
| #include <LibWeb/DOM/Text.h>
 | |
| #include <LibWeb/Layout/Node.h>
 | |
| 
 | |
| namespace Web::Layout {
 | |
| 
 | |
| class LineBoxFragment;
 | |
| 
 | |
| class TextNode final : public Node {
 | |
|     GC_CELL(TextNode, Node);
 | |
|     GC_DECLARE_ALLOCATOR(TextNode);
 | |
| 
 | |
| public:
 | |
|     TextNode(DOM::Document&, DOM::Text&);
 | |
|     virtual ~TextNode() override;
 | |
| 
 | |
|     const DOM::Text& dom_node() const { return static_cast<const DOM::Text&>(*Node::dom_node()); }
 | |
| 
 | |
|     Utf16String const& text_for_rendering() const;
 | |
| 
 | |
|     struct Chunk {
 | |
|         Utf16View view;
 | |
|         NonnullRefPtr<Gfx::Font const> font;
 | |
|         size_t start { 0 };
 | |
|         size_t length { 0 };
 | |
|         bool has_breaking_newline { false };
 | |
|         bool has_breaking_tab { false };
 | |
|         bool is_all_whitespace { false };
 | |
|         Gfx::GlyphRun::TextType text_type;
 | |
|     };
 | |
| 
 | |
|     class ChunkIterator {
 | |
|     public:
 | |
|         ChunkIterator(TextNode const&, bool wrap_lines, bool respect_linebreaks);
 | |
|         ChunkIterator(TextNode const&, Utf16View const&, Unicode::Segmenter&, bool wrap_lines, bool respect_linebreaks);
 | |
| 
 | |
|         Optional<Chunk> next();
 | |
|         Optional<Chunk> peek(size_t);
 | |
| 
 | |
|     private:
 | |
|         Optional<Chunk> next_without_peek();
 | |
|         Optional<Chunk> try_commit_chunk(size_t start, size_t end, bool has_breaking_newline, bool has_breaking_tab, Gfx::Font const&, Gfx::GlyphRun::TextType) const;
 | |
| 
 | |
|         bool const m_wrap_lines;
 | |
|         bool const m_respect_linebreaks;
 | |
|         Utf16View m_view;
 | |
|         Gfx::FontCascadeList const& m_font_cascade_list;
 | |
| 
 | |
|         Unicode::Segmenter& m_grapheme_segmenter;
 | |
|         size_t m_current_index { 0 };
 | |
| 
 | |
|         Vector<Chunk> m_peek_queue;
 | |
|     };
 | |
| 
 | |
|     void invalidate_text_for_rendering();
 | |
| 
 | |
|     Unicode::Segmenter& grapheme_segmenter() const;
 | |
| 
 | |
|     virtual GC::Ptr<Painting::Paintable> create_paintable() const override;
 | |
| 
 | |
| private:
 | |
|     virtual bool is_text_node() const final { return true; }
 | |
| 
 | |
|     void compute_text_for_rendering();
 | |
| 
 | |
|     Optional<Utf16String> m_text_for_rendering;
 | |
|     mutable OwnPtr<Unicode::Segmenter> m_grapheme_segmenter;
 | |
| };
 | |
| 
 | |
| template<>
 | |
| inline bool Node::fast_is<TextNode>() const { return is_text_node(); }
 | |
| 
 | |
| }
 |