HikoGUI
A low latency retained GUI
Loading...
Searching...
No Matches
text_shaper.hpp
1// Copyright Take Vos 2021-2022.
2// Distributed under the Boost Software License, Version 1.0.
3// (See accompanying file LICENSE_1_0.txt or copy at https://www.boost.org/LICENSE_1_0.txt)
4
5#pragma once
6
7#include "text_shaper_char.hpp"
8#include "text_shaper_line.hpp"
9#include "text_cursor.hpp"
10#include "text_style.hpp"
11#include "../layout/box_constraints.hpp"
12#include "../font/module.hpp"
13#include "../geometry/module.hpp"
14#include "../unicode/module.hpp"
15#include <vector>
16#include <tuple>
17
18namespace hi::inline v1 {
19class font_book;
20
35public:
37 using char_iterator = char_vector::iterator;
38 using char_const_iterator = char_vector::const_iterator;
39 using char_reference = char_vector::reference;
40 using char_const_reference = char_vector::const_reference;
42 using line_iterator = line_vector::iterator;
43 using line_const_iterator = line_vector::const_iterator;
44
45 constexpr text_shaper() noexcept = default;
46 constexpr text_shaper(text_shaper const&) noexcept = default;
47 constexpr text_shaper(text_shaper&&) noexcept = default;
48 constexpr text_shaper& operator=(text_shaper const&) noexcept = default;
49 constexpr text_shaper& operator=(text_shaper&&) noexcept = default;
50
80 [[nodiscard]] text_shaper(
81 hi::font_book& font_book,
82 gstring const& text,
83 text_style const& style,
84 float dpi_scale,
85 hi::alignment alignment,
86 bool left_to_right,
87 iso_15924 script = iso_15924{"Zyyy"}) noexcept;
88
89 [[nodiscard]] text_shaper(
90 hi::font_book& font_book,
91 std::string_view text,
92 text_style const& style,
93 float dpi_scale,
94 hi::alignment alignment,
95 bool left_to_right,
96 iso_15924 script = iso_15924{"Zyyy"}) noexcept;
97
98 [[nodiscard]] bool empty() const noexcept
99 {
100 return _text.empty();
101 }
102
103 [[nodiscard]] size_t size() const noexcept
104 {
105 return _text.size();
106 }
107
108 [[nodiscard]] char_iterator begin() noexcept
109 {
110 return _text.begin();
111 }
112
113 [[nodiscard]] char_const_iterator begin() const noexcept
114 {
115 return _text.begin();
116 }
117
118 [[nodiscard]] char_const_iterator cbegin() const noexcept
119 {
120 return _text.cbegin();
121 }
122
123 [[nodiscard]] char_iterator end() noexcept
124 {
125 return _text.end();
126 }
127
128 [[nodiscard]] char_const_iterator end() const noexcept
129 {
130 return _text.end();
131 }
132
133 [[nodiscard]] char_const_iterator cend() const noexcept
134 {
135 return _text.cend();
136 }
137
138 auto const& lines() const noexcept
139 {
140 return _lines;
141 }
142
161 [[nodiscard]] aarectangle
162 bounding_rectangle(float maximum_line_width, float line_spacing = 1.0f, float paragraph_spacing = 1.5f) noexcept;
163
176 [[nodiscard]] void layout(
177 aarectangle rectangle,
178 float baseline,
179 extent2 sub_pixel_size,
180 float line_spacing = 1.0f,
181 float paragraph_spacing = 1.5f) noexcept;
182
185 [[nodiscard]] aarectangle rectangle() const noexcept
186 {
187 return _rectangle;
188 }
189
192 [[nodiscard]] unicode_bidi_class text_direction() const noexcept
193 {
194 return _text_direction;
195 }
196
202 [[nodiscard]] alignment resolved_alignment() const noexcept
203 {
204 return resolve(_alignment, _text_direction == unicode_bidi_class::L);
205 }
206
214 [[nodiscard]] char_const_iterator get_it(size_t index) const noexcept;
215
223 [[nodiscard]] char_const_iterator get_it(text_cursor cursor) const noexcept
224 {
225 return get_it(cursor.index());
226 }
227
236 [[nodiscard]] char_const_iterator get_it(size_t column, size_t row) const noexcept;
237
245 [[nodiscard]] char_const_iterator get_it(std::pair<size_t, size_t> column_row) const noexcept
246 {
247 return get_it(column_row.first, column_row.second);
248 }
249
255 [[nodiscard]] std::pair<size_t, size_t> get_column_line(text_shaper::char_const_iterator it) const noexcept;
256
262 [[nodiscard]] std::pair<size_t, size_t> get_column_line(size_t index) const noexcept
263 {
264 return get_column_line(get_it(index));
265 }
266
272 [[nodiscard]] std::pair<size_t, size_t> get_column_line(text_cursor cursor) const noexcept
273 {
274 return get_column_line(cursor.index());
275 }
276
282 [[nodiscard]] size_t get_index(text_shaper::char_const_iterator it) const noexcept;
283
288 [[nodiscard]] text_cursor get_begin_cursor() const noexcept;
289
294 [[nodiscard]] text_cursor get_end_cursor() const noexcept;
295
301 [[nodiscard]] text_cursor get_before_cursor(size_t index) const noexcept;
302
308 [[nodiscard]] text_cursor get_after_cursor(size_t index) const noexcept;
309
315 [[nodiscard]] text_cursor get_before_cursor(text_shaper::char_const_iterator it) const noexcept
316 {
317 return get_before_cursor(get_index(it));
318 }
319
325 [[nodiscard]] text_cursor get_after_cursor(text_shaper::char_const_iterator it) const noexcept
326 {
327 return get_after_cursor(get_index(it));
328 }
329
335 [[nodiscard]] text_cursor get_left_cursor(text_shaper::char_const_iterator it) const noexcept;
336
342 [[nodiscard]] text_cursor get_right_cursor(text_shaper::char_const_iterator it) const noexcept;
343
349 [[nodiscard]] bool is_on_left(text_cursor cursor) const noexcept;
350
356 [[nodiscard]] bool is_on_right(text_cursor cursor) const noexcept;
357
363 [[nodiscard]] text_cursor get_nearest_cursor(point2 point) const noexcept;
364
367 [[nodiscard]] std::pair<text_cursor, text_cursor> select_char(text_cursor cursor) const noexcept;
368
371 [[nodiscard]] std::pair<text_cursor, text_cursor> select_word(text_cursor cursor) const noexcept;
372
376
380
384
390 [[nodiscard]] char_const_iterator move_left_char(char_const_iterator it) const noexcept;
391
397 [[nodiscard]] char_const_iterator move_right_char(char_const_iterator it) const noexcept;
398
399 [[nodiscard]] text_cursor move_left_char(text_cursor cursor, bool overwrite_mode) const noexcept;
400 [[nodiscard]] text_cursor move_right_char(text_cursor cursor, bool overwrite_mode) const noexcept;
401 [[nodiscard]] text_cursor move_down_char(text_cursor cursor, float& x) const noexcept;
402 [[nodiscard]] text_cursor move_up_char(text_cursor cursor, float& x) const noexcept;
403 [[nodiscard]] text_cursor move_left_word(text_cursor cursor, bool overwrite_mode) const noexcept;
404 [[nodiscard]] text_cursor move_right_word(text_cursor cursor, bool overwrite_mode) const noexcept;
405 [[nodiscard]] text_cursor move_begin_line(text_cursor cursor) const noexcept;
406 [[nodiscard]] text_cursor move_end_line(text_cursor cursor) const noexcept;
407 [[nodiscard]] text_cursor move_begin_sentence(text_cursor cursor) const noexcept;
408 [[nodiscard]] text_cursor move_end_sentence(text_cursor cursor) const noexcept;
409 [[nodiscard]] text_cursor move_begin_paragraph(text_cursor cursor) const noexcept;
410 [[nodiscard]] text_cursor move_end_paragraph(text_cursor cursor) const noexcept;
411 [[nodiscard]] text_cursor move_begin_document(text_cursor cursor) const noexcept;
412 [[nodiscard]] text_cursor move_end_document(text_cursor cursor) const noexcept;
413
414private:
415 font_book *_font_book = nullptr;
416
419 float _dpi_scale;
420
427 char_vector _text;
428
429 hi::alignment _alignment;
430
433 unicode_break_vector _line_break_opportunities;
434
437 std::vector<float> _line_break_widths;
438
441 unicode_break_vector _word_break_opportunities;
442
445 unicode_break_vector _sentence_break_opportunities;
446
449 unicode_bidi_context _bidi_context;
450
453 unicode_bidi_class _text_direction;
454
457 iso_15924 _script;
458
463 line_vector _lines;
464
467 font_metrics _initial_line_metrics;
468
471 aarectangle _rectangle;
472
481 [[nodiscard]] line_vector make_lines(
482 aarectangle rectangle,
483 float baseline,
484 extent2 sub_pixel_size,
485 float line_spacing,
486 float paragraph_spacing) noexcept;
487
494 void position_glyphs(aarectangle rectangle, extent2 sub_pixel_size) noexcept;
495
498 void resolve_script() noexcept;
499
500 [[nodiscard]] std::pair<text_cursor, text_cursor>
501 get_selection_from_break(text_cursor cursor, unicode_break_vector const& break_opportunities) const noexcept;
502
503 [[nodiscard]] std::pair<font_metrics, unicode_general_category>
504 get_line_metrics(text_shaper::char_const_iterator first, text_shaper::char_const_iterator last) const noexcept;
505
512 [[nodiscard]] float get_text_height(std::vector<size_t> const& lines) const noexcept;
513};
514
515} // namespace hi::inline v1
STL namespace.
DOXYGEN BUG.
Definition algorithm.hpp:13
font_book keeps track of multiple fonts.
Definition font_book.hpp:31
The metrics of a font.
Definition font_metrics.hpp:15
Horizontal/Vertical alignment combination.
Definition alignment.hpp:239
A high-level geometric point Part of the high-level vec, point, mat and color types.
Definition point2.hpp:23
ISO-15924 script code.
Definition iso_15924.hpp:17
A variant of text.
Definition label.hpp:31
A cursor-position in text.
Definition text_cursor.hpp:22
Text shaper.
Definition text_shaper.hpp:34
char_const_iterator get_it(size_t column, size_t row) const noexcept
Get the character at column and row in display order.
std::pair< text_cursor, text_cursor > select_paragraph(text_cursor cursor) const noexcept
Get the selection for a paragraph at the cursor.
std::pair< size_t, size_t > get_column_line(size_t index) const noexcept
Get the column and line of a character.
Definition text_shaper.hpp:262
char_const_iterator get_it(text_cursor cursor) const noexcept
Get the character at the cursor.
Definition text_shaper.hpp:223
std::pair< text_cursor, text_cursor > select_sentence(text_cursor cursor) const noexcept
Get the selection for the sentence at the cursor.
bool is_on_left(text_cursor cursor) const noexcept
Check if the cursor is on the left side of the character in display order.
bool is_on_right(text_cursor cursor) const noexcept
Check if the cursor is on the right side of the character in display order.
aarectangle bounding_rectangle(float maximum_line_width, float line_spacing=1.0f, float paragraph_spacing=1.5f) noexcept
Get bounding rectangle.
text_cursor get_after_cursor(text_shaper::char_const_iterator it) const noexcept
Get the cursor after the character in logical order.
Definition text_shaper.hpp:325
unicode_bidi_class text_direction() const noexcept
Get the text-direction as a whole.
Definition text_shaper.hpp:192
std::pair< text_cursor, text_cursor > select_char(text_cursor cursor) const noexcept
Get the selection for the character at the cursor.
std::pair< size_t, size_t > get_column_line(text_shaper::char_const_iterator it) const noexcept
Get the column and line of a character.
text_cursor get_left_cursor(text_shaper::char_const_iterator it) const noexcept
Get the cursor left of the character in display order.
std::pair< text_cursor, text_cursor > select_word(text_cursor cursor) const noexcept
Get the selection for the word at the cursor.
size_t get_index(text_shaper::char_const_iterator it) const noexcept
Get the index of the character in logical order.
char_const_iterator get_it(size_t index) const noexcept
Get the character at index in logical order.
std::pair< text_cursor, text_cursor > select_document(text_cursor cursor) const noexcept
Get the selection for a paragraph at the cursor.
text_cursor get_nearest_cursor(point2 point) const noexcept
find the nearest character.
char_const_iterator move_right_char(char_const_iterator it) const noexcept
Get the character to the right.
char_const_iterator get_it(std::pair< size_t, size_t > column_row) const noexcept
Get the character at column and row in display order.
Definition text_shaper.hpp:245
text_cursor get_begin_cursor() const noexcept
Get the cursor at the beginning of the document.
alignment resolved_alignment() const noexcept
Get the resolved alignment of the text.
Definition text_shaper.hpp:202
std::pair< size_t, size_t > get_column_line(text_cursor cursor) const noexcept
Get the column and line of a character.
Definition text_shaper.hpp:272
text_cursor get_right_cursor(text_shaper::char_const_iterator it) const noexcept
Get the cursor right of the character in display order.
char_const_iterator move_left_char(char_const_iterator it) const noexcept
Get the character to the left.
text_shaper(hi::font_book &font_book, gstring const &text, text_style const &style, float dpi_scale, hi::alignment alignment, bool left_to_right, iso_15924 script=iso_15924{"Zyyy"}) noexcept
Construct a text_shaper with a text and alignment.
Definition text_style.hpp:172
Definition unicode_bidi.hpp:18
T begin(T... args)
T end(T... args)