| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2024-10-04 13:19:50 +02:00
										 |  |  |  * Copyright (c) 2018-2021, Andreas Kling <andreas@ladybird.org> | 
					
						
							| 
									
										
										
										
											2022-01-20 20:30:00 +01:00
										 |  |  |  * Copyright (c) 2022, Tobias Christiansen <tobyase@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-01 21:18:08 +02:00
										 |  |  | #include <AK/CharacterTypes.h>
 | 
					
						
							| 
									
										
										
										
											2019-09-25 12:36:44 +03:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:57:52 +01:00
										 |  |  | #include <LibUnicode/CharacterTypes.h>
 | 
					
						
							| 
									
										
										
										
											2024-10-26 11:23:34 +02:00
										 |  |  | #include <LibUnicode/Locale.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-06 20:02:41 +02:00
										 |  |  | #include <LibWeb/Layout/BlockContainer.h>
 | 
					
						
							| 
									
										
										
										
											2020-12-05 20:10:39 +01:00
										 |  |  | #include <LibWeb/Layout/InlineFormattingContext.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  | #include <LibWeb/Layout/TextNode.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-10 16:46:44 +01:00
										 |  |  | #include <LibWeb/Painting/TextPaintable.h>
 | 
					
						
							| 
									
										
										
										
											2019-06-15 22:49:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  | namespace Web::Layout { | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  | GC_DEFINE_ALLOCATOR(TextNode); | 
					
						
							| 
									
										
										
										
											2024-04-06 10:16:04 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  | TextNode::TextNode(DOM::Document& document, DOM::Text& text) | 
					
						
							|  |  |  |     : Node(document, &text) | 
					
						
							| 
									
										
										
										
											2019-06-15 22:49:44 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 13:21:51 -06:00
										 |  |  | TextNode::~TextNode() = default; | 
					
						
							| 
									
										
										
										
											2019-06-15 23:17:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  | static bool is_all_whitespace(StringView string) | 
					
						
							| 
									
										
										
										
											2019-06-15 23:17:08 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-12-09 17:45:40 +01:00
										 |  |  |     for (size_t i = 0; i < string.length(); ++i) { | 
					
						
							| 
									
										
										
										
											2021-06-01 21:18:08 +02:00
										 |  |  |         if (!is_ascii_space(string[i])) | 
					
						
							| 
									
										
										
										
											2019-06-15 23:17:08 +02:00
										 |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-08 16:01:57 +01:00
										 |  |  | // https://w3c.github.io/mathml-core/#new-text-transform-values
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  | static String apply_math_auto_text_transform(String const& string) | 
					
						
							| 
									
										
										
										
											2023-09-08 16:01:57 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // https://w3c.github.io/mathml-core/#italic-mappings
 | 
					
						
							|  |  |  |     auto map_code_point_to_italic = [](u32 code_point) -> u32 { | 
					
						
							|  |  |  |         switch (code_point) { | 
					
						
							|  |  |  |         case 0x0041: | 
					
						
							|  |  |  |             return 0x1D434; | 
					
						
							|  |  |  |         case 0x0042: | 
					
						
							|  |  |  |             return 0x1D435; | 
					
						
							|  |  |  |         case 0x0043: | 
					
						
							|  |  |  |             return 0x1D436; | 
					
						
							|  |  |  |         case 0x0044: | 
					
						
							|  |  |  |             return 0x1D437; | 
					
						
							|  |  |  |         case 0x0045: | 
					
						
							|  |  |  |             return 0x1D438; | 
					
						
							|  |  |  |         case 0x0046: | 
					
						
							|  |  |  |             return 0x1D439; | 
					
						
							|  |  |  |         case 0x0047: | 
					
						
							|  |  |  |             return 0x1D43A; | 
					
						
							|  |  |  |         case 0x0048: | 
					
						
							|  |  |  |             return 0x1D43B; | 
					
						
							|  |  |  |         case 0x0049: | 
					
						
							|  |  |  |             return 0x1D43C; | 
					
						
							|  |  |  |         case 0x004A: | 
					
						
							|  |  |  |             return 0x1D43D; | 
					
						
							|  |  |  |         case 0x004B: | 
					
						
							|  |  |  |             return 0x1D43E; | 
					
						
							|  |  |  |         case 0x004C: | 
					
						
							|  |  |  |             return 0x1D43F; | 
					
						
							|  |  |  |         case 0x004D: | 
					
						
							|  |  |  |             return 0x1D440; | 
					
						
							|  |  |  |         case 0x004E: | 
					
						
							|  |  |  |             return 0x1D441; | 
					
						
							|  |  |  |         case 0x004F: | 
					
						
							|  |  |  |             return 0x1D442; | 
					
						
							|  |  |  |         case 0x0050: | 
					
						
							|  |  |  |             return 0x1D443; | 
					
						
							|  |  |  |         case 0x0051: | 
					
						
							|  |  |  |             return 0x1D444; | 
					
						
							|  |  |  |         case 0x0052: | 
					
						
							|  |  |  |             return 0x1D445; | 
					
						
							|  |  |  |         case 0x0053: | 
					
						
							|  |  |  |             return 0x1D446; | 
					
						
							|  |  |  |         case 0x0054: | 
					
						
							|  |  |  |             return 0x1D447; | 
					
						
							|  |  |  |         case 0x0055: | 
					
						
							|  |  |  |             return 0x1D448; | 
					
						
							|  |  |  |         case 0x0056: | 
					
						
							|  |  |  |             return 0x1D449; | 
					
						
							|  |  |  |         case 0x0057: | 
					
						
							|  |  |  |             return 0x1D44A; | 
					
						
							|  |  |  |         case 0x0058: | 
					
						
							|  |  |  |             return 0x1D44B; | 
					
						
							|  |  |  |         case 0x0059: | 
					
						
							|  |  |  |             return 0x1D44C; | 
					
						
							|  |  |  |         case 0x005A: | 
					
						
							|  |  |  |             return 0x1D44D; | 
					
						
							|  |  |  |         case 0x0061: | 
					
						
							|  |  |  |             return 0x1D44E; | 
					
						
							|  |  |  |         case 0x0062: | 
					
						
							|  |  |  |             return 0x1D44F; | 
					
						
							|  |  |  |         case 0x0063: | 
					
						
							|  |  |  |             return 0x1D450; | 
					
						
							|  |  |  |         case 0x0064: | 
					
						
							|  |  |  |             return 0x1D451; | 
					
						
							|  |  |  |         case 0x0065: | 
					
						
							|  |  |  |             return 0x1D452; | 
					
						
							|  |  |  |         case 0x0066: | 
					
						
							|  |  |  |             return 0x1D453; | 
					
						
							|  |  |  |         case 0x0067: | 
					
						
							|  |  |  |             return 0x1D454; | 
					
						
							|  |  |  |         case 0x0068: | 
					
						
							|  |  |  |             return 0x0210E; | 
					
						
							|  |  |  |         case 0x0069: | 
					
						
							|  |  |  |             return 0x1D456; | 
					
						
							|  |  |  |         case 0x006A: | 
					
						
							|  |  |  |             return 0x1D457; | 
					
						
							|  |  |  |         case 0x006B: | 
					
						
							|  |  |  |             return 0x1D458; | 
					
						
							|  |  |  |         case 0x006C: | 
					
						
							|  |  |  |             return 0x1D459; | 
					
						
							|  |  |  |         case 0x006D: | 
					
						
							|  |  |  |             return 0x1D45A; | 
					
						
							|  |  |  |         case 0x006E: | 
					
						
							|  |  |  |             return 0x1D45B; | 
					
						
							|  |  |  |         case 0x006F: | 
					
						
							|  |  |  |             return 0x1D45C; | 
					
						
							|  |  |  |         case 0x0070: | 
					
						
							|  |  |  |             return 0x1D45D; | 
					
						
							|  |  |  |         case 0x0071: | 
					
						
							|  |  |  |             return 0x1D45E; | 
					
						
							|  |  |  |         case 0x0072: | 
					
						
							|  |  |  |             return 0x1D45F; | 
					
						
							|  |  |  |         case 0x0073: | 
					
						
							|  |  |  |             return 0x1D460; | 
					
						
							|  |  |  |         case 0x0074: | 
					
						
							|  |  |  |             return 0x1D461; | 
					
						
							|  |  |  |         case 0x0075: | 
					
						
							|  |  |  |             return 0x1D462; | 
					
						
							|  |  |  |         case 0x0076: | 
					
						
							|  |  |  |             return 0x1D463; | 
					
						
							|  |  |  |         case 0x0077: | 
					
						
							|  |  |  |             return 0x1D464; | 
					
						
							|  |  |  |         case 0x0078: | 
					
						
							|  |  |  |             return 0x1D465; | 
					
						
							|  |  |  |         case 0x0079: | 
					
						
							|  |  |  |             return 0x1D466; | 
					
						
							|  |  |  |         case 0x007A: | 
					
						
							|  |  |  |             return 0x1D467; | 
					
						
							|  |  |  |         case 0x0131: | 
					
						
							|  |  |  |             return 0x1D6A4; | 
					
						
							|  |  |  |         case 0x0237: | 
					
						
							|  |  |  |             return 0x1D6A5; | 
					
						
							|  |  |  |         case 0x0391: | 
					
						
							|  |  |  |             return 0x1D6E2; | 
					
						
							|  |  |  |         case 0x0392: | 
					
						
							|  |  |  |             return 0x1D6E3; | 
					
						
							|  |  |  |         case 0x0393: | 
					
						
							|  |  |  |             return 0x1D6E4; | 
					
						
							|  |  |  |         case 0x0394: | 
					
						
							|  |  |  |             return 0x1D6E5; | 
					
						
							|  |  |  |         case 0x0395: | 
					
						
							|  |  |  |             return 0x1D6E6; | 
					
						
							|  |  |  |         case 0x0396: | 
					
						
							|  |  |  |             return 0x1D6E7; | 
					
						
							|  |  |  |         case 0x0397: | 
					
						
							|  |  |  |             return 0x1D6E8; | 
					
						
							|  |  |  |         case 0x0398: | 
					
						
							|  |  |  |             return 0x1D6E9; | 
					
						
							|  |  |  |         case 0x0399: | 
					
						
							|  |  |  |             return 0x1D6EA; | 
					
						
							|  |  |  |         case 0x039A: | 
					
						
							|  |  |  |             return 0x1D6EB; | 
					
						
							|  |  |  |         case 0x039B: | 
					
						
							|  |  |  |             return 0x1D6EC; | 
					
						
							|  |  |  |         case 0x039C: | 
					
						
							|  |  |  |             return 0x1D6ED; | 
					
						
							|  |  |  |         case 0x039D: | 
					
						
							|  |  |  |             return 0x1D6EE; | 
					
						
							|  |  |  |         case 0x039E: | 
					
						
							|  |  |  |             return 0x1D6EF; | 
					
						
							|  |  |  |         case 0x039F: | 
					
						
							|  |  |  |             return 0x1D6F0; | 
					
						
							|  |  |  |         case 0x03A0: | 
					
						
							|  |  |  |             return 0x1D6F1; | 
					
						
							|  |  |  |         case 0x03A1: | 
					
						
							|  |  |  |             return 0x1D6F2; | 
					
						
							|  |  |  |         case 0x03F4: | 
					
						
							|  |  |  |             return 0x1D6F3; | 
					
						
							|  |  |  |         case 0x03A3: | 
					
						
							|  |  |  |             return 0x1D6F4; | 
					
						
							|  |  |  |         case 0x03A4: | 
					
						
							|  |  |  |             return 0x1D6F5; | 
					
						
							|  |  |  |         case 0x03A5: | 
					
						
							|  |  |  |             return 0x1D6F6; | 
					
						
							|  |  |  |         case 0x03A6: | 
					
						
							|  |  |  |             return 0x1D6F7; | 
					
						
							|  |  |  |         case 0x03A7: | 
					
						
							|  |  |  |             return 0x1D6F8; | 
					
						
							|  |  |  |         case 0x03A8: | 
					
						
							|  |  |  |             return 0x1D6F9; | 
					
						
							|  |  |  |         case 0x03A9: | 
					
						
							|  |  |  |             return 0x1D6FA; | 
					
						
							|  |  |  |         case 0x2207: | 
					
						
							|  |  |  |             return 0x1D6FB; | 
					
						
							|  |  |  |         case 0x03B1: | 
					
						
							|  |  |  |             return 0x1D6FC; | 
					
						
							|  |  |  |         case 0x03B2: | 
					
						
							|  |  |  |             return 0x1D6FD; | 
					
						
							|  |  |  |         case 0x03B3: | 
					
						
							|  |  |  |             return 0x1D6FE; | 
					
						
							|  |  |  |         case 0x03B4: | 
					
						
							|  |  |  |             return 0x1D6FF; | 
					
						
							|  |  |  |         case 0x03B5: | 
					
						
							|  |  |  |             return 0x1D700; | 
					
						
							|  |  |  |         case 0x03B6: | 
					
						
							|  |  |  |             return 0x1D701; | 
					
						
							|  |  |  |         case 0x03B7: | 
					
						
							|  |  |  |             return 0x1D702; | 
					
						
							|  |  |  |         case 0x03B8: | 
					
						
							|  |  |  |             return 0x1D703; | 
					
						
							|  |  |  |         case 0x03B9: | 
					
						
							|  |  |  |             return 0x1D704; | 
					
						
							|  |  |  |         case 0x03BA: | 
					
						
							|  |  |  |             return 0x1D705; | 
					
						
							|  |  |  |         case 0x03BB: | 
					
						
							|  |  |  |             return 0x1D706; | 
					
						
							|  |  |  |         case 0x03BC: | 
					
						
							|  |  |  |             return 0x1D707; | 
					
						
							|  |  |  |         case 0x03BD: | 
					
						
							|  |  |  |             return 0x1D708; | 
					
						
							|  |  |  |         case 0x03BE: | 
					
						
							|  |  |  |             return 0x1D709; | 
					
						
							|  |  |  |         case 0x03BF: | 
					
						
							|  |  |  |             return 0x1D70A; | 
					
						
							|  |  |  |         case 0x03C0: | 
					
						
							|  |  |  |             return 0x1D70B; | 
					
						
							|  |  |  |         case 0x03C1: | 
					
						
							|  |  |  |             return 0x1D70C; | 
					
						
							|  |  |  |         case 0x03C2: | 
					
						
							|  |  |  |             return 0x1D70D; | 
					
						
							|  |  |  |         case 0x03C3: | 
					
						
							|  |  |  |             return 0x1D70E; | 
					
						
							|  |  |  |         case 0x03C4: | 
					
						
							|  |  |  |             return 0x1D70F; | 
					
						
							|  |  |  |         case 0x03C5: | 
					
						
							|  |  |  |             return 0x1D710; | 
					
						
							|  |  |  |         case 0x03C6: | 
					
						
							|  |  |  |             return 0x1D711; | 
					
						
							|  |  |  |         case 0x03C7: | 
					
						
							|  |  |  |             return 0x1D712; | 
					
						
							|  |  |  |         case 0x03C8: | 
					
						
							|  |  |  |             return 0x1D713; | 
					
						
							|  |  |  |         case 0x03C9: | 
					
						
							|  |  |  |             return 0x1D714; | 
					
						
							|  |  |  |         case 0x2202: | 
					
						
							|  |  |  |             return 0x1D715; | 
					
						
							|  |  |  |         case 0x03F5: | 
					
						
							|  |  |  |             return 0x1D716; | 
					
						
							|  |  |  |         case 0x03D1: | 
					
						
							|  |  |  |             return 0x1D717; | 
					
						
							|  |  |  |         case 0x03F0: | 
					
						
							|  |  |  |             return 0x1D718; | 
					
						
							|  |  |  |         case 0x03D5: | 
					
						
							|  |  |  |             return 0x1D719; | 
					
						
							|  |  |  |         case 0x03F1: | 
					
						
							|  |  |  |             return 0x1D71A; | 
					
						
							|  |  |  |         case 0x03D6: | 
					
						
							|  |  |  |             return 0x1D71B; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return code_point; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     StringBuilder builder(string.bytes().size()); | 
					
						
							| 
									
										
										
										
											2023-09-08 16:01:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     for (auto code_point : string.code_points()) | 
					
						
							| 
									
										
										
										
											2023-09-08 16:01:57 +01:00
										 |  |  |         builder.append_code_point(map_code_point_to_italic(code_point)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     return MUST(builder.to_string()); | 
					
						
							| 
									
										
										
										
											2023-09-08 16:01:57 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-26 11:23:34 +02:00
										 |  |  | static ErrorOr<String> apply_text_transform(String const& string, CSS::TextTransform text_transform, Optional<StringView> const& locale) | 
					
						
							| 
									
										
										
										
											2022-12-23 19:57:52 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-09-07 17:43:10 +01:00
										 |  |  |     switch (text_transform) { | 
					
						
							|  |  |  |     case CSS::TextTransform::Uppercase: | 
					
						
							| 
									
										
										
										
											2024-10-29 11:35:26 +01:00
										 |  |  |         return string.to_uppercase(locale); | 
					
						
							| 
									
										
										
										
											2023-09-07 17:43:10 +01:00
										 |  |  |     case CSS::TextTransform::Lowercase: | 
					
						
							| 
									
										
										
										
											2024-10-29 12:05:52 +01:00
										 |  |  |         return string.to_lowercase(locale); | 
					
						
							| 
									
										
										
										
											2023-09-07 17:43:10 +01:00
										 |  |  |     case CSS::TextTransform::None: | 
					
						
							|  |  |  |         return string; | 
					
						
							| 
									
										
										
										
											2023-09-08 16:01:57 +01:00
										 |  |  |     case CSS::TextTransform::MathAuto: | 
					
						
							|  |  |  |         return apply_math_auto_text_transform(string); | 
					
						
							| 
									
										
										
										
											2023-09-29 12:32:15 +02:00
										 |  |  |     case CSS::TextTransform::Capitalize: { | 
					
						
							| 
									
										
										
										
											2024-10-26 11:23:34 +02:00
										 |  |  |         return string.to_titlecase(locale, TrailingCodePointTransformation::PreserveExisting); | 
					
						
							| 
									
										
										
										
											2023-09-29 12:32:15 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-11-01 08:01:48 +01:00
										 |  |  |     case CSS::TextTransform::FullSizeKana: { | 
					
						
							|  |  |  |         // FIXME: Implement this!
 | 
					
						
							| 
									
										
										
										
											2023-09-07 17:43:10 +01:00
										 |  |  |         return string; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-11-01 08:01:48 +01:00
										 |  |  |     case CSS::TextTransform::FullWidth: { | 
					
						
							|  |  |  |         return string.to_fullwidth(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-08 16:01:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:57:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-16 11:03:00 +02:00
										 |  |  | void TextNode::invalidate_text_for_rendering() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_text_for_rendering = {}; | 
					
						
							| 
									
										
										
										
											2024-10-09 18:53:50 +02:00
										 |  |  |     m_grapheme_segmenter.clear(); | 
					
						
							| 
									
										
										
										
											2023-08-16 11:03:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  | String const& TextNode::text_for_rendering() const | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-10-10 15:00:58 +03:30
										 |  |  |     if (!m_text_for_rendering.has_value()) | 
					
						
							| 
									
										
										
										
											2023-07-03 10:29:30 +02:00
										 |  |  |         const_cast<TextNode*>(this)->compute_text_for_rendering(); | 
					
						
							| 
									
										
										
										
											2023-10-10 15:00:58 +03:30
										 |  |  |     return *m_text_for_rendering; | 
					
						
							| 
									
										
										
										
											2023-07-03 10:29:30 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NOTE: This collapses whitespace into a single ASCII space if the CSS white-space property tells us to.
 | 
					
						
							|  |  |  | void TextNode::compute_text_for_rendering() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-11-14 18:06:16 +00:00
										 |  |  |     if (dom_node().is_password_input()) { | 
					
						
							|  |  |  |         m_text_for_rendering = MUST(String::repeated('*', dom_node().data().code_points().length())); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 00:31:24 +12:00
										 |  |  |     bool collapse = first_is_one_of(computed_values().white_space_collapse(), CSS::WhiteSpaceCollapse::Collapse, CSS::WhiteSpaceCollapse::PreserveBreaks); | 
					
						
							| 
									
										
										
										
											2023-07-03 10:29:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-18 14:19:19 +12:00
										 |  |  |     auto parent_element = dom_node().parent_element(); | 
					
						
							| 
									
										
										
										
											2024-10-26 11:23:34 +02:00
										 |  |  |     auto const maybe_lang = parent_element ? parent_element->lang() : Optional<String> {}; | 
					
						
							|  |  |  |     auto const lang = maybe_lang.has_value() ? maybe_lang.value() : Optional<StringView> {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto data = apply_text_transform(dom_node().data(), computed_values().text_transform(), lang).release_value_but_fixme_should_propagate_errors(); | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto data_view = data.bytes_as_string_view(); | 
					
						
							| 
									
										
										
										
											2022-11-05 17:19:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |     if (!collapse || data.is_empty()) { | 
					
						
							|  |  |  |         m_text_for_rendering = data; | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |     // NOTE: A couple fast returns to avoid unnecessarily allocating a StringBuilder.
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     if (data_view.length() == 1) { | 
					
						
							|  |  |  |         if (is_ascii_space(data_view[0])) { | 
					
						
							|  |  |  |             static String s_single_space_string = " "_string; | 
					
						
							| 
									
										
										
										
											2022-03-27 21:14:10 +02:00
										 |  |  |             m_text_for_rendering = s_single_space_string; | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             m_text_for_rendering = data; | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool contains_space = false; | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     for (auto c : data_view) { | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |         if (is_ascii_space(c)) { | 
					
						
							|  |  |  |             contains_space = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!contains_space) { | 
					
						
							|  |  |  |         m_text_for_rendering = data; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     StringBuilder builder(data_view.length()); | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |     size_t index = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     auto skip_over_whitespace = [&index, &data_view] { | 
					
						
							|  |  |  |         while (index < data_view.length() && is_ascii_space(data_view[index])) | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |             ++index; | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     while (index < data_view.length()) { | 
					
						
							|  |  |  |         if (is_ascii_space(data_view[index])) { | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  |             builder.append(' '); | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |             ++index; | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  |             skip_over_whitespace(); | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |             builder.append(data_view[index]); | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  |             ++index; | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-03 22:33:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 10:56:29 +13:00
										 |  |  |     m_text_for_rendering = MUST(builder.to_string()); | 
					
						
							| 
									
										
										
										
											2021-04-29 08:51:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-22 10:03:23 -04:00
										 |  |  | Unicode::Segmenter& TextNode::grapheme_segmenter() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_grapheme_segmenter) { | 
					
						
							|  |  |  |         m_grapheme_segmenter = document().grapheme_segmenter().clone(); | 
					
						
							|  |  |  |         m_grapheme_segmenter->set_segmented_text(text_for_rendering()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return *m_grapheme_segmenter; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TextNode::ChunkIterator::ChunkIterator(TextNode const& text_node, bool wrap_lines, bool respect_linebreaks) | 
					
						
							| 
									
										
										
										
											2022-10-14 12:26:18 +02:00
										 |  |  |     : m_wrap_lines(wrap_lines) | 
					
						
							| 
									
										
										
										
											2021-08-28 00:53:59 +00:00
										 |  |  |     , m_respect_linebreaks(respect_linebreaks) | 
					
						
							| 
									
										
										
										
											2024-09-22 10:03:23 -04:00
										 |  |  |     , m_utf8_view(text_node.text_for_rendering()) | 
					
						
							|  |  |  |     , m_font_cascade_list(text_node.computed_values().font_list()) | 
					
						
							|  |  |  |     , m_grapheme_segmenter(text_node.grapheme_segmenter()) | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-18 17:58:05 +01:00
										 |  |  | static Gfx::GlyphRun::TextType text_type_for_code_point(u32 code_point) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (Unicode::bidirectional_class(code_point)) { | 
					
						
							|  |  |  |     case Unicode::BidiClass::WhiteSpaceNeutral: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case Unicode::BidiClass::BlockSeparator: | 
					
						
							|  |  |  |     case Unicode::BidiClass::SegmentSeparator: | 
					
						
							|  |  |  |     case Unicode::BidiClass::CommonNumberSeparator: | 
					
						
							|  |  |  |     case Unicode::BidiClass::DirNonSpacingMark: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case Unicode::BidiClass::ArabicNumber: | 
					
						
							|  |  |  |     case Unicode::BidiClass::EuropeanNumber: | 
					
						
							|  |  |  |     case Unicode::BidiClass::EuropeanNumberSeparator: | 
					
						
							|  |  |  |     case Unicode::BidiClass::EuropeanNumberTerminator: | 
					
						
							|  |  |  |         return Gfx::GlyphRun::TextType::ContextDependent; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case Unicode::BidiClass::BoundaryNeutral: | 
					
						
							|  |  |  |     case Unicode::BidiClass::OtherNeutral: | 
					
						
							|  |  |  |     case Unicode::BidiClass::FirstStrongIsolate: | 
					
						
							|  |  |  |     case Unicode::BidiClass::PopDirectionalFormat: | 
					
						
							|  |  |  |     case Unicode::BidiClass::PopDirectionalIsolate: | 
					
						
							|  |  |  |         return Gfx::GlyphRun::TextType::Common; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case Unicode::BidiClass::LeftToRight: | 
					
						
							|  |  |  |     case Unicode::BidiClass::LeftToRightEmbedding: | 
					
						
							|  |  |  |     case Unicode::BidiClass::LeftToRightIsolate: | 
					
						
							|  |  |  |     case Unicode::BidiClass::LeftToRightOverride: | 
					
						
							|  |  |  |         return Gfx::GlyphRun::TextType::Ltr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case Unicode::BidiClass::RightToLeft: | 
					
						
							|  |  |  |     case Unicode::BidiClass::RightToLeftArabic: | 
					
						
							|  |  |  |     case Unicode::BidiClass::RightToLeftEmbedding: | 
					
						
							|  |  |  |     case Unicode::BidiClass::RightToLeftIsolate: | 
					
						
							|  |  |  |     case Unicode::BidiClass::RightToLeftOverride: | 
					
						
							|  |  |  |         return Gfx::GlyphRun::TextType::Rtl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  | Optional<TextNode::Chunk> TextNode::ChunkIterator::next() | 
					
						
							| 
									
										
										
										
											2024-08-18 17:58:05 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!m_peek_queue.is_empty()) | 
					
						
							|  |  |  |         return m_peek_queue.take_first(); | 
					
						
							|  |  |  |     return next_without_peek(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Optional<TextNode::Chunk> TextNode::ChunkIterator::peek(size_t count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (m_peek_queue.size() <= count) { | 
					
						
							|  |  |  |         auto next = next_without_peek(); | 
					
						
							|  |  |  |         if (!next.has_value()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         m_peek_queue.append(*next); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return m_peek_queue[count]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Optional<TextNode::Chunk> TextNode::ChunkIterator::next_without_peek() | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |     if (m_current_index >= m_utf8_view.byte_length()) | 
					
						
							| 
									
										
										
										
											2021-08-28 00:45:28 +00:00
										 |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |     auto current_code_point = [this]() { | 
					
						
							|  |  |  |         return *m_utf8_view.iterator_at_byte_offset_without_validation(m_current_index); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     auto next_grapheme_boundary = [this]() { | 
					
						
							| 
									
										
										
										
											2024-09-22 10:03:23 -04:00
										 |  |  |         return m_grapheme_segmenter.next_boundary(m_current_index).value_or(m_utf8_view.byte_length()); | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto code_point = current_code_point(); | 
					
						
							|  |  |  |     auto start_of_chunk = m_current_index; | 
					
						
							| 
									
										
										
										
											2021-08-28 00:45:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |     Gfx::Font const& font = m_font_cascade_list.font_for_code_point(code_point); | 
					
						
							|  |  |  |     auto text_type = text_type_for_code_point(code_point); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |     auto broken_on_tab = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |     while (m_current_index < m_utf8_view.byte_length()) { | 
					
						
							|  |  |  |         code_point = current_code_point(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |         if (code_point == '\t') { | 
					
						
							|  |  |  |             if (auto result = try_commit_chunk(start_of_chunk, m_current_index, false, broken_on_tab, font, text_type); result.has_value()) | 
					
						
							|  |  |  |                 return result.release_value(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             broken_on_tab = true; | 
					
						
							|  |  |  |             // consume any consecutive tabs
 | 
					
						
							|  |  |  |             while (m_current_index < m_utf8_view.byte_length() && current_code_point() == '\t') { | 
					
						
							|  |  |  |                 m_current_index = next_grapheme_boundary(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |         if (&font != &m_font_cascade_list.font_for_code_point(code_point)) { | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |             if (auto result = try_commit_chunk(start_of_chunk, m_current_index, false, broken_on_tab, font, text_type); result.has_value()) | 
					
						
							| 
									
										
										
										
											2024-06-29 17:14:23 +02:00
										 |  |  |                 return result.release_value(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |         if (m_respect_linebreaks && code_point == '\n') { | 
					
						
							| 
									
										
										
										
											2022-03-26 19:58:10 +01:00
										 |  |  |             // Newline encountered, and we're supposed to preserve them.
 | 
					
						
							|  |  |  |             // If we have accumulated some code points in the current chunk, commit them now and continue with the newline next time.
 | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |             if (auto result = try_commit_chunk(start_of_chunk, m_current_index, false, broken_on_tab, font, text_type); result.has_value()) | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |                 return result.release_value(); | 
					
						
							| 
									
										
										
										
											2022-03-26 19:58:10 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Otherwise, commit the newline!
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |             m_current_index = next_grapheme_boundary(); | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |             auto result = try_commit_chunk(start_of_chunk, m_current_index, true, broken_on_tab, font, text_type); | 
					
						
							| 
									
										
										
										
											2022-03-26 19:58:10 +01:00
										 |  |  |             VERIFY(result.has_value()); | 
					
						
							|  |  |  |             return result.release_value(); | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-08-28 00:45:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-09 15:17:47 +02:00
										 |  |  |         if (m_wrap_lines) { | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |             if (text_type != text_type_for_code_point(code_point)) { | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |                 if (auto result = try_commit_chunk(start_of_chunk, m_current_index, false, broken_on_tab, font, text_type); result.has_value()) { | 
					
						
							| 
									
										
										
										
											2024-08-18 17:58:05 +01:00
										 |  |  |                     return result.release_value(); | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-08-18 17:58:05 +01:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |             if (is_ascii_space(code_point)) { | 
					
						
							| 
									
										
										
										
											2022-03-26 19:58:10 +01:00
										 |  |  |                 // Whitespace encountered, and we're allowed to break on whitespace.
 | 
					
						
							|  |  |  |                 // If we have accumulated some code points in the current chunk, commit them now and continue with the whitespace next time.
 | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |                 if (auto result = try_commit_chunk(start_of_chunk, m_current_index, false, broken_on_tab, font, text_type); result.has_value()) { | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |                     return result.release_value(); | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-03-26 19:58:10 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 // Otherwise, commit the whitespace!
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |                 m_current_index = next_grapheme_boundary(); | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |                 if (auto result = try_commit_chunk(start_of_chunk, m_current_index, false, broken_on_tab, font, text_type); result.has_value()) | 
					
						
							| 
									
										
										
										
											2022-03-26 19:58:10 +01:00
										 |  |  |                     return result.release_value(); | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-26 19:58:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 12:33:29 +02:00
										 |  |  |         m_current_index = next_grapheme_boundary(); | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |     if (start_of_chunk != m_utf8_view.byte_length()) { | 
					
						
							| 
									
										
										
										
											2021-08-28 00:45:28 +00:00
										 |  |  |         // Try to output whatever's left at the end of the text node.
 | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |         if (auto result = try_commit_chunk(start_of_chunk, m_utf8_view.byte_length(), false, broken_on_tab, font, text_type); result.has_value()) | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |             return result.release_value(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  | Optional<TextNode::Chunk> TextNode::ChunkIterator::try_commit_chunk(size_t start, size_t end, bool has_breaking_newline, bool has_breaking_tab, Gfx::Font const& font, Gfx::GlyphRun::TextType text_type) const | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |     if (auto byte_length = end - start; byte_length > 0) { | 
					
						
							|  |  |  |         auto chunk_view = m_utf8_view.substring_view(start, byte_length); | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |         return Chunk { | 
					
						
							|  |  |  |             .view = chunk_view, | 
					
						
							| 
									
										
										
										
											2024-06-29 17:14:23 +02:00
										 |  |  |             .font = font, | 
					
						
							| 
									
										
										
										
											2024-09-20 17:14:14 -04:00
										 |  |  |             .start = start, | 
					
						
							| 
									
										
										
										
											2021-08-28 00:45:28 +00:00
										 |  |  |             .length = byte_length, | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |             .has_breaking_newline = has_breaking_newline, | 
					
						
							| 
									
										
										
										
											2024-10-15 18:21:49 +01:00
										 |  |  |             .has_breaking_tab = has_breaking_tab, | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |             .is_all_whitespace = is_all_whitespace(chunk_view.as_string()), | 
					
						
							| 
									
										
										
										
											2024-08-18 17:58:05 +01:00
										 |  |  |             .text_type = text_type, | 
					
						
							| 
									
										
										
										
											2021-04-27 13:05:50 +02:00
										 |  |  |         }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  | GC::Ptr<Painting::Paintable> TextNode::create_paintable() const | 
					
						
							| 
									
										
										
										
											2022-03-10 16:46:44 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-03-18 10:25:57 +01:00
										 |  |  |     return Painting::TextPaintable::create(*this, text_for_rendering()); | 
					
						
							| 
									
										
										
										
											2022-03-10 16:46:44 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | } |