| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2023, Ali Mohammad Pur <mpfard@serenityos.org> | 
					
						
							| 
									
										
										
										
											2025-02-05 16:29:59 +00:00
										 |  |  |  * Copyright (c) 2023-2025, Sam Atkins <sam@ladybird.org> | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-19 12:21:15 +01:00
										 |  |  | #include "ShorthandStyleValue.h"
 | 
					
						
							| 
									
										
										
										
											2025-02-05 16:29:59 +00:00
										 |  |  | #include <LibGfx/Font/FontWeight.h>
 | 
					
						
							| 
									
										
										
										
											2023-09-19 16:26:23 +01:00
										 |  |  | #include <LibWeb/CSS/PropertyID.h>
 | 
					
						
							| 
									
										
										
										
											2025-06-10 17:38:55 +12:00
										 |  |  | #include <LibWeb/CSS/StyleComputer.h>
 | 
					
						
							| 
									
										
										
										
											2023-09-19 16:01:44 +01:00
										 |  |  | #include <LibWeb/CSS/StyleValues/BorderRadiusStyleValue.h>
 | 
					
						
							| 
									
										
										
										
											2023-09-20 14:50:44 +01:00
										 |  |  | #include <LibWeb/CSS/StyleValues/GridTemplateAreaStyleValue.h>
 | 
					
						
							| 
									
										
										
										
											2023-09-20 14:47:37 +01:00
										 |  |  | #include <LibWeb/CSS/StyleValues/GridTrackPlacementStyleValue.h>
 | 
					
						
							| 
									
										
										
										
											2023-09-20 14:50:44 +01:00
										 |  |  | #include <LibWeb/CSS/StyleValues/GridTrackSizeListStyleValue.h>
 | 
					
						
							| 
									
										
										
										
											2025-08-08 10:28:41 +01:00
										 |  |  | #include <LibWeb/CSS/StyleValues/KeywordStyleValue.h>
 | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  | #include <LibWeb/CSS/StyleValues/StyleValueList.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Web::CSS { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  | ShorthandStyleValue::ShorthandStyleValue(PropertyID shorthand, Vector<PropertyID> sub_properties, Vector<ValueComparingNonnullRefPtr<StyleValue const>> values) | 
					
						
							| 
									
										
										
										
											2023-09-19 12:21:15 +01:00
										 |  |  |     : StyleValueWithDefaultOperators(Type::Shorthand) | 
					
						
							| 
									
										
										
										
											2023-09-19 16:26:23 +01:00
										 |  |  |     , m_properties { shorthand, move(sub_properties), move(values) } | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  | { | 
					
						
							|  |  |  |     if (m_properties.sub_properties.size() != m_properties.values.size()) { | 
					
						
							| 
									
										
										
										
											2023-09-19 12:21:15 +01:00
										 |  |  |         dbgln("ShorthandStyleValue: sub_properties and values must be the same size! {} != {}", m_properties.sub_properties.size(), m_properties.values.size()); | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-19 12:21:15 +01:00
										 |  |  | ShorthandStyleValue::~ShorthandStyleValue() = default; | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  | ValueComparingRefPtr<StyleValue const> ShorthandStyleValue::longhand(PropertyID longhand) const | 
					
						
							| 
									
										
										
										
											2023-09-19 16:26:23 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     for (auto i = 0u; i < m_properties.sub_properties.size(); ++i) { | 
					
						
							|  |  |  |         if (m_properties.sub_properties[i] == longhand) | 
					
						
							|  |  |  |             return m_properties.values[i]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  | String ShorthandStyleValue::to_string(SerializationMode mode) const | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  |     // If all the longhands are the same CSS-wide keyword, just return that once.
 | 
					
						
							|  |  |  |     Optional<Keyword> built_in_keyword; | 
					
						
							|  |  |  |     bool all_same_keyword = true; | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |     StyleComputer::for_each_property_expanding_shorthands(m_properties.shorthand_property, *this, [&](PropertyID name, StyleValue const& value) { | 
					
						
							| 
									
										
										
										
											2025-06-10 17:38:55 +12:00
										 |  |  |         (void)name; | 
					
						
							|  |  |  |         if (!value.is_css_wide_keyword()) { | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  |             all_same_keyword = false; | 
					
						
							| 
									
										
										
										
											2025-06-10 17:38:55 +12:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-06-10 17:38:55 +12:00
										 |  |  |         auto keyword = value.to_keyword(); | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  |         if (!built_in_keyword.has_value()) { | 
					
						
							|  |  |  |             built_in_keyword = keyword; | 
					
						
							| 
									
										
										
										
											2025-06-10 17:38:55 +12:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (built_in_keyword != keyword) { | 
					
						
							|  |  |  |             all_same_keyword = false; | 
					
						
							| 
									
										
										
										
											2025-06-10 17:38:55 +12:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-06-10 17:38:55 +12:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-11 15:25:28 +12:00
										 |  |  |     if (built_in_keyword.has_value()) { | 
					
						
							|  |  |  |         if (all_same_keyword) | 
					
						
							|  |  |  |             return MUST(String::from_utf8(string_from_keyword(built_in_keyword.value()))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ""_string; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |     auto positional_value_list_shorthand_to_string = [&](Vector<ValueComparingNonnullRefPtr<StyleValue const>> values) -> String { | 
					
						
							| 
									
										
										
										
											2025-07-10 22:23:07 +12:00
										 |  |  |         switch (values.size()) { | 
					
						
							|  |  |  |         case 2: { | 
					
						
							|  |  |  |             auto first_property_serialized = values[0]->to_string(mode); | 
					
						
							|  |  |  |             auto second_property_serialized = values[1]->to_string(mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (first_property_serialized == second_property_serialized) | 
					
						
							|  |  |  |                 return first_property_serialized; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return MUST(String::formatted("{} {}", first_property_serialized, second_property_serialized)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case 4: { | 
					
						
							|  |  |  |             auto first_property_serialized = values[0]->to_string(mode); | 
					
						
							|  |  |  |             auto second_property_serialized = values[1]->to_string(mode); | 
					
						
							|  |  |  |             auto third_property_serialized = values[2]->to_string(mode); | 
					
						
							|  |  |  |             auto fourth_property_serialized = values[3]->to_string(mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (first_is_equal_to_all_of(first_property_serialized, second_property_serialized, third_property_serialized, fourth_property_serialized)) | 
					
						
							|  |  |  |                 return first_property_serialized; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (first_property_serialized == third_property_serialized && second_property_serialized == fourth_property_serialized) | 
					
						
							|  |  |  |                 return MUST(String::formatted("{} {}", first_property_serialized, second_property_serialized)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (second_property_serialized == fourth_property_serialized) | 
					
						
							|  |  |  |                 return MUST(String::formatted("{} {} {}", first_property_serialized, second_property_serialized, third_property_serialized)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return MUST(String::formatted("{} {} {} {}", first_property_serialized, second_property_serialized, third_property_serialized, fourth_property_serialized)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 00:31:24 +12:00
										 |  |  |     auto default_to_string = [&]() { | 
					
						
							|  |  |  |         auto all_properties_same_value = true; | 
					
						
							|  |  |  |         auto first_property_value = m_properties.values.first(); | 
					
						
							|  |  |  |         for (auto i = 1u; i < m_properties.values.size(); ++i) { | 
					
						
							|  |  |  |             if (m_properties.values[i] != first_property_value) { | 
					
						
							|  |  |  |                 all_properties_same_value = false; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (all_properties_same_value) | 
					
						
							|  |  |  |             return first_property_value->to_string(mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         StringBuilder builder; | 
					
						
							|  |  |  |         auto first = true; | 
					
						
							|  |  |  |         for (size_t i = 0; i < m_properties.values.size(); ++i) { | 
					
						
							|  |  |  |             auto value = m_properties.values[i]; | 
					
						
							|  |  |  |             auto value_string = value->to_string(mode); | 
					
						
							|  |  |  |             auto initial_value_string = property_initial_value(m_properties.sub_properties[i])->to_string(mode); | 
					
						
							|  |  |  |             if (value_string == initial_value_string) | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             if (first) | 
					
						
							|  |  |  |                 first = false; | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 builder.append(' '); | 
					
						
							|  |  |  |             builder.append(value->to_string(mode)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (builder.is_empty()) | 
					
						
							|  |  |  |             return m_properties.values.first()->to_string(mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return MUST(builder.to_string()); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-29 13:46:37 +00:00
										 |  |  |     // Then special cases
 | 
					
						
							| 
									
										
										
										
											2023-09-19 16:26:23 +01:00
										 |  |  |     switch (m_properties.shorthand_property) { | 
					
						
							| 
									
										
										
										
											2025-06-10 01:31:14 +12:00
										 |  |  |     case PropertyID::All: { | 
					
						
							|  |  |  |         // NOTE: 'all' can only be serialized in the case all sub-properties share the same CSS-wide keyword, this is
 | 
					
						
							|  |  |  |         //       handled above, thus, if we get to here that mustn't be the case and we should return the empty string.
 | 
					
						
							|  |  |  |         return ""_string; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |     case PropertyID::Background: { | 
					
						
							|  |  |  |         auto color = longhand(PropertyID::BackgroundColor); | 
					
						
							|  |  |  |         auto image = longhand(PropertyID::BackgroundImage); | 
					
						
							|  |  |  |         auto position = longhand(PropertyID::BackgroundPosition); | 
					
						
							| 
									
										
										
										
											2024-11-29 22:44:14 +11:00
										 |  |  |         auto position_x = position->as_shorthand().longhand(PropertyID::BackgroundPositionX); | 
					
						
							|  |  |  |         auto position_y = position->as_shorthand().longhand(PropertyID::BackgroundPositionY); | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |         auto size = longhand(PropertyID::BackgroundSize); | 
					
						
							|  |  |  |         auto repeat = longhand(PropertyID::BackgroundRepeat); | 
					
						
							|  |  |  |         auto attachment = longhand(PropertyID::BackgroundAttachment); | 
					
						
							|  |  |  |         auto origin = longhand(PropertyID::BackgroundOrigin); | 
					
						
							|  |  |  |         auto clip = longhand(PropertyID::BackgroundClip); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-03 20:10:12 +12:00
										 |  |  |         auto serialize_layer = [mode](Optional<String> color_value_string, String image_value_string, String position_x_value_string, String position_y_value_string, String size_value_string, String repeat_value_string, String attachment_value_string, String origin_value_string, String clip_value_string) { | 
					
						
							|  |  |  |             StringBuilder builder; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Vector<PropertyID> property_ids = { PropertyID::BackgroundColor, PropertyID::BackgroundImage, PropertyID::BackgroundPositionX, PropertyID::BackgroundPositionY, PropertyID::BackgroundSize, PropertyID::BackgroundRepeat, PropertyID::BackgroundAttachment, PropertyID::BackgroundOrigin, PropertyID::BackgroundClip }; | 
					
						
							|  |  |  |             Vector<Optional<String>> property_value_strings = { move(color_value_string), move(image_value_string), move(position_x_value_string), move(position_y_value_string), move(size_value_string), move(repeat_value_string), move(attachment_value_string), move(origin_value_string), move(clip_value_string) }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (size_t i = 0; i < property_ids.size(); i++) { | 
					
						
							|  |  |  |                 if (!property_value_strings[i].has_value()) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 auto intial_property_string_value = property_initial_value(property_ids[i])->to_string(mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (property_value_strings[i].value() != intial_property_string_value) { | 
					
						
							|  |  |  |                     if (!builder.is_empty()) | 
					
						
							|  |  |  |                         builder.append(" "sv); | 
					
						
							|  |  |  |                     builder.append(property_value_strings[i].value()); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (builder.is_empty()) | 
					
						
							|  |  |  |                 return "none"_string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return builder.to_string_without_validation(); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |         auto get_layer_count = [](auto style_value) -> size_t { | 
					
						
							|  |  |  |             return style_value->is_value_list() ? style_value->as_value_list().size() : 1; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-29 22:44:14 +11:00
										 |  |  |         auto layer_count = max(get_layer_count(image), max(get_layer_count(position_x), max(get_layer_count(position_y), max(get_layer_count(size), max(get_layer_count(repeat), max(get_layer_count(attachment), max(get_layer_count(origin), get_layer_count(clip)))))))); | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (layer_count == 1) { | 
					
						
							| 
									
										
										
										
											2025-06-03 20:10:12 +12:00
										 |  |  |             return serialize_layer(color->to_string(mode), image->to_string(mode), position_x->to_string(mode), position_y->to_string(mode), size->to_string(mode), repeat->to_string(mode), attachment->to_string(mode), origin->to_string(mode), clip->to_string(mode)); | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |         auto get_layer_value_string = [mode](ValueComparingRefPtr<StyleValue const> const& style_value, size_t index) { | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |             if (style_value->is_value_list()) | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  |                 return style_value->as_value_list().value_at(index, true)->to_string(mode); | 
					
						
							|  |  |  |             return style_value->to_string(mode); | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         StringBuilder builder; | 
					
						
							|  |  |  |         for (size_t i = 0; i < layer_count; i++) { | 
					
						
							|  |  |  |             if (i) | 
					
						
							|  |  |  |                 builder.append(", "sv); | 
					
						
							| 
									
										
										
										
											2025-06-03 20:10:12 +12:00
										 |  |  | 
 | 
					
						
							|  |  |  |             Optional<String> maybe_color_value_string; | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |             if (i == layer_count - 1) | 
					
						
							| 
									
										
										
										
											2025-06-03 20:10:12 +12:00
										 |  |  |                 maybe_color_value_string = color->to_string(mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             builder.append(serialize_layer(maybe_color_value_string, get_layer_value_string(image, i), get_layer_value_string(position_x, i), get_layer_value_string(position_y, i), get_layer_value_string(size, i), get_layer_value_string(repeat, i), get_layer_value_string(attachment, i), get_layer_value_string(origin, i), get_layer_value_string(clip, i))); | 
					
						
							| 
									
										
										
										
											2024-11-29 22:44:14 +11:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return MUST(builder.to_string()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case Web::CSS::PropertyID::BackgroundPosition: { | 
					
						
							|  |  |  |         auto x_edges = longhand(PropertyID::BackgroundPositionX); | 
					
						
							|  |  |  |         auto y_edges = longhand(PropertyID::BackgroundPositionY); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto get_layer_count = [](auto style_value) -> size_t { | 
					
						
							|  |  |  |             return style_value->is_value_list() ? style_value->as_value_list().size() : 1; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // FIXME: The spec is unclear about how differing layer counts should be handled
 | 
					
						
							|  |  |  |         auto layer_count = max(get_layer_count(x_edges), get_layer_count(y_edges)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (layer_count == 1) { | 
					
						
							|  |  |  |             return MUST(String::formatted("{} {}", x_edges->to_string(mode), y_edges->to_string(mode))); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |         auto get_layer_value_string = [mode](ValueComparingRefPtr<StyleValue const> const& style_value, size_t index) { | 
					
						
							| 
									
										
										
										
											2024-11-29 22:44:14 +11:00
										 |  |  |             if (style_value->is_value_list()) | 
					
						
							|  |  |  |                 return style_value->as_value_list().value_at(index, true)->to_string(mode); | 
					
						
							|  |  |  |             return style_value->to_string(mode); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         StringBuilder builder; | 
					
						
							|  |  |  |         for (size_t i = 0; i < layer_count; i++) { | 
					
						
							|  |  |  |             if (i) | 
					
						
							|  |  |  |                 builder.append(", "sv); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             builder.appendff("{} {}", get_layer_value_string(x_edges, i), get_layer_value_string(y_edges, i)); | 
					
						
							| 
									
										
										
										
											2023-09-19 15:27:13 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return MUST(builder.to_string()); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-07-10 22:45:44 +12:00
										 |  |  |     case PropertyID::Border: { | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |         auto all_longhands_same_value = [](ValueComparingRefPtr<StyleValue const> const& shorthand) -> bool { | 
					
						
							| 
									
										
										
										
											2025-07-10 22:45:44 +12:00
										 |  |  |             VERIFY(shorthand); | 
					
						
							| 
									
										
										
										
											2025-07-16 09:46:04 +12:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // FIXME: This can be removed once we parse border-width, border-style and border-color directly to ShorthandStyleValue
 | 
					
						
							|  |  |  |             if (!shorthand->is_shorthand()) | 
					
						
							|  |  |  |                 return !shorthand->is_value_list(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-10 22:45:44 +12:00
										 |  |  |             VERIFY(shorthand->is_shorthand()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto longhands = shorthand->as_shorthand().values(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return all_of(longhands, [&](auto const& longhand) { return longhand == longhands[0]; }); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // `border` only has a reasonable value if all four sides are the same.
 | 
					
						
							|  |  |  |         if (!all_longhands_same_value(longhand(PropertyID::BorderWidth)) || !all_longhands_same_value(longhand(PropertyID::BorderStyle)) || !all_longhands_same_value(longhand(PropertyID::BorderColor))) | 
					
						
							|  |  |  |             return ""_string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return default_to_string(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-06-11 12:35:32 +01:00
										 |  |  |     case PropertyID::BorderImage: { | 
					
						
							|  |  |  |         auto source = longhand(PropertyID::BorderImageSource); | 
					
						
							|  |  |  |         auto slice = longhand(PropertyID::BorderImageSlice); | 
					
						
							|  |  |  |         auto width = longhand(PropertyID::BorderImageWidth); | 
					
						
							|  |  |  |         auto outset = longhand(PropertyID::BorderImageOutset); | 
					
						
							|  |  |  |         auto repeat = longhand(PropertyID::BorderImageRepeat); | 
					
						
							|  |  |  |         return MUST(String::formatted("{} {} / {} / {} {}", | 
					
						
							|  |  |  |             source->to_string(mode), | 
					
						
							|  |  |  |             slice->to_string(mode), | 
					
						
							|  |  |  |             width->to_string(mode), | 
					
						
							|  |  |  |             outset->to_string(mode), | 
					
						
							|  |  |  |             repeat->to_string(mode))); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-19 16:01:44 +01:00
										 |  |  |     case PropertyID::BorderRadius: { | 
					
						
							| 
									
										
										
										
											2025-03-18 12:49:16 +00:00
										 |  |  |         auto top_left = longhand(PropertyID::BorderTopLeftRadius); | 
					
						
							|  |  |  |         auto top_right = longhand(PropertyID::BorderTopRightRadius); | 
					
						
							|  |  |  |         auto bottom_right = longhand(PropertyID::BorderBottomRightRadius); | 
					
						
							|  |  |  |         auto bottom_left = longhand(PropertyID::BorderBottomLeftRadius); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto horizontal_radius = [&](auto& style_value) -> String { | 
					
						
							|  |  |  |             if (style_value->is_border_radius()) | 
					
						
							| 
									
										
										
										
											2025-08-03 15:37:42 +12:00
										 |  |  |                 return style_value->as_border_radius().horizontal_radius().to_string(mode); | 
					
						
							| 
									
										
										
										
											2025-03-18 12:49:16 +00:00
										 |  |  |             return style_value->to_string(mode); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto top_left_horizontal_string = horizontal_radius(top_left); | 
					
						
							|  |  |  |         auto top_right_horizontal_string = horizontal_radius(top_right); | 
					
						
							|  |  |  |         auto bottom_right_horizontal_string = horizontal_radius(bottom_right); | 
					
						
							|  |  |  |         auto bottom_left_horizontal_string = horizontal_radius(bottom_left); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto vertical_radius = [&](auto& style_value) -> String { | 
					
						
							|  |  |  |             if (style_value->is_border_radius()) | 
					
						
							| 
									
										
										
										
											2025-08-03 15:37:42 +12:00
										 |  |  |                 return style_value->as_border_radius().vertical_radius().to_string(mode); | 
					
						
							| 
									
										
										
										
											2025-03-18 12:49:16 +00:00
										 |  |  |             return style_value->to_string(mode); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto top_left_vertical_string = vertical_radius(top_left); | 
					
						
							|  |  |  |         auto top_right_vertical_string = vertical_radius(top_right); | 
					
						
							|  |  |  |         auto bottom_right_vertical_string = vertical_radius(bottom_right); | 
					
						
							|  |  |  |         auto bottom_left_vertical_string = vertical_radius(bottom_left); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto serialize_radius = [](auto top_left, auto const& top_right, auto const& bottom_right, auto const& bottom_left) -> String { | 
					
						
							|  |  |  |             if (first_is_equal_to_all_of(top_left, top_right, bottom_right, bottom_left)) | 
					
						
							|  |  |  |                 return top_left; | 
					
						
							|  |  |  |             if (top_left == bottom_right && top_right == bottom_left) | 
					
						
							|  |  |  |                 return MUST(String::formatted("{} {}", top_left, top_right)); | 
					
						
							|  |  |  |             if (top_right == bottom_left) | 
					
						
							|  |  |  |                 return MUST(String::formatted("{} {} {}", top_left, top_right, bottom_right)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return MUST(String::formatted("{} {} {} {}", top_left, top_right, bottom_right, bottom_left)); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto first_radius_serialization = serialize_radius(move(top_left_horizontal_string), top_right_horizontal_string, bottom_right_horizontal_string, bottom_left_horizontal_string); | 
					
						
							|  |  |  |         auto second_radius_serialization = serialize_radius(move(top_left_vertical_string), top_right_vertical_string, bottom_right_vertical_string, bottom_left_vertical_string); | 
					
						
							|  |  |  |         if (first_radius_serialization == second_radius_serialization) | 
					
						
							|  |  |  |             return first_radius_serialization; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return MUST(String::formatted("{} / {}", first_radius_serialization, second_radius_serialization)); | 
					
						
							| 
									
										
										
										
											2023-09-19 16:01:44 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-21 11:09:08 -04:00
										 |  |  |     case PropertyID::Columns: { | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  |         auto column_width = longhand(PropertyID::ColumnWidth)->to_string(mode); | 
					
						
							|  |  |  |         auto column_count = longhand(PropertyID::ColumnCount)->to_string(mode); | 
					
						
							| 
									
										
										
										
											2024-08-21 11:09:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (column_width == column_count) | 
					
						
							|  |  |  |             return column_width; | 
					
						
							|  |  |  |         if (column_width.equals_ignoring_ascii_case("auto"sv)) | 
					
						
							|  |  |  |             return column_count; | 
					
						
							|  |  |  |         if (column_count.equals_ignoring_ascii_case("auto"sv)) | 
					
						
							|  |  |  |             return column_width; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return MUST(String::formatted("{} {}", column_width, column_count)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-19 16:26:23 +01:00
										 |  |  |     case PropertyID::Flex: | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  |         return MUST(String::formatted("{} {} {}", longhand(PropertyID::FlexGrow)->to_string(mode), longhand(PropertyID::FlexShrink)->to_string(mode), longhand(PropertyID::FlexBasis)->to_string(mode))); | 
					
						
							| 
									
										
										
										
											2025-02-05 16:29:59 +00:00
										 |  |  |     case PropertyID::Font: { | 
					
						
							|  |  |  |         auto font_style = longhand(PropertyID::FontStyle); | 
					
						
							|  |  |  |         auto font_variant = longhand(PropertyID::FontVariant); | 
					
						
							|  |  |  |         auto font_weight = longhand(PropertyID::FontWeight); | 
					
						
							|  |  |  |         auto font_width = longhand(PropertyID::FontWidth); | 
					
						
							|  |  |  |         auto font_size = longhand(PropertyID::FontSize); | 
					
						
							|  |  |  |         auto line_height = longhand(PropertyID::LineHeight); | 
					
						
							|  |  |  |         auto font_family = longhand(PropertyID::FontFamily); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-12 13:11:42 +00:00
										 |  |  |         // Some longhands prevent serialization if they are not allowed in the shorthand.
 | 
					
						
							|  |  |  |         // <font-variant-css2> = normal | small-caps
 | 
					
						
							|  |  |  |         auto font_variant_string = font_variant->to_string(mode); | 
					
						
							|  |  |  |         if (!first_is_one_of(font_variant_string, "normal"sv, "small-caps"sv) && !CSS::is_css_wide_keyword(font_variant_string)) { | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // <font-width-css3> = normal | ultra-condensed | extra-condensed | condensed | semi-condensed | semi-expanded | expanded | extra-expanded | ultra-expanded
 | 
					
						
							|  |  |  |         switch (font_width->to_keyword()) { | 
					
						
							|  |  |  |         case Keyword::Initial: | 
					
						
							|  |  |  |         case Keyword::Normal: | 
					
						
							|  |  |  |         case Keyword::UltraCondensed: | 
					
						
							|  |  |  |         case Keyword::ExtraCondensed: | 
					
						
							|  |  |  |         case Keyword::Condensed: | 
					
						
							|  |  |  |         case Keyword::SemiCondensed: | 
					
						
							|  |  |  |         case Keyword::SemiExpanded: | 
					
						
							|  |  |  |         case Keyword::Expanded: | 
					
						
							|  |  |  |         case Keyword::ExtraExpanded: | 
					
						
							|  |  |  |         case Keyword::UltraExpanded: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             if (!font_width->is_css_wide_keyword()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-05 16:29:59 +00:00
										 |  |  |         StringBuilder builder; | 
					
						
							|  |  |  |         auto append = [&](auto const& string) { | 
					
						
							|  |  |  |             if (!builder.is_empty()) | 
					
						
							|  |  |  |                 builder.append(' '); | 
					
						
							|  |  |  |             builder.append(string); | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2025-05-02 13:55:58 +01:00
										 |  |  |         auto font_style_string = font_style->to_string(mode); | 
					
						
							|  |  |  |         if (font_style_string != "normal"sv) | 
					
						
							|  |  |  |             append(font_style_string); | 
					
						
							| 
									
										
										
										
											2025-02-12 13:11:42 +00:00
										 |  |  |         if (font_variant_string != "normal"sv && font_variant_string != "initial"sv) | 
					
						
							|  |  |  |             append(font_variant_string); | 
					
						
							| 
									
										
										
										
											2025-05-23 16:19:35 +01:00
										 |  |  |         auto font_weight_string = font_weight->to_string(mode); | 
					
						
							|  |  |  |         if (font_weight_string != "normal"sv && font_weight_string != "initial"sv && font_weight_string != "400"sv) | 
					
						
							|  |  |  |             append(font_weight_string); | 
					
						
							| 
									
										
										
										
											2025-02-05 16:29:59 +00:00
										 |  |  |         if (font_width->to_keyword() != Keyword::Normal && font_width->to_keyword() != Keyword::Initial) | 
					
						
							|  |  |  |             append(font_width->to_string(mode)); | 
					
						
							|  |  |  |         append(font_size->to_string(mode)); | 
					
						
							|  |  |  |         if (line_height->to_keyword() != Keyword::Normal && line_height->to_keyword() != Keyword::Initial) | 
					
						
							|  |  |  |             append(MUST(String::formatted("/ {}", line_height->to_string(mode)))); | 
					
						
							|  |  |  |         append(font_family->to_string(mode)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return builder.to_string_without_validation(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-12-05 01:19:03 +01:00
										 |  |  |     case PropertyID::FontVariant: { | 
					
						
							| 
									
										
										
										
											2025-02-06 17:21:34 +00:00
										 |  |  |         auto ligatures = longhand(PropertyID::FontVariantLigatures); | 
					
						
							|  |  |  |         auto caps = longhand(PropertyID::FontVariantCaps); | 
					
						
							|  |  |  |         auto alternates = longhand(PropertyID::FontVariantAlternates); | 
					
						
							|  |  |  |         auto numeric = longhand(PropertyID::FontVariantNumeric); | 
					
						
							|  |  |  |         auto east_asian = longhand(PropertyID::FontVariantEastAsian); | 
					
						
							|  |  |  |         auto position = longhand(PropertyID::FontVariantPosition); | 
					
						
							|  |  |  |         auto emoji = longhand(PropertyID::FontVariantEmoji); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-10 17:04:57 +12:00
										 |  |  |         // If ligatures is `none` and any other value isn't `normal`, that's invalid.
 | 
					
						
							|  |  |  |         if (ligatures->to_keyword() == Keyword::None && !first_is_equal_to_all_of(Keyword::Normal, caps->to_keyword(), alternates->to_keyword(), numeric->to_keyword(), east_asian->to_keyword(), position->to_keyword(), emoji->to_keyword())) | 
					
						
							|  |  |  |             return ""_string; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-06 17:21:34 +00:00
										 |  |  |         Vector<String> values; | 
					
						
							|  |  |  |         if (ligatures->to_keyword() != Keyword::Normal) | 
					
						
							|  |  |  |             values.append(ligatures->to_string(mode)); | 
					
						
							|  |  |  |         if (caps->to_keyword() != Keyword::Normal) | 
					
						
							|  |  |  |             values.append(caps->to_string(mode)); | 
					
						
							|  |  |  |         if (alternates->to_keyword() != Keyword::Normal) | 
					
						
							|  |  |  |             values.append(alternates->to_string(mode)); | 
					
						
							|  |  |  |         if (numeric->to_keyword() != Keyword::Normal) | 
					
						
							|  |  |  |             values.append(numeric->to_string(mode)); | 
					
						
							|  |  |  |         if (east_asian->to_keyword() != Keyword::Normal) | 
					
						
							|  |  |  |             values.append(east_asian->to_string(mode)); | 
					
						
							|  |  |  |         if (position->to_keyword() != Keyword::Normal) | 
					
						
							|  |  |  |             values.append(position->to_string(mode)); | 
					
						
							|  |  |  |         if (emoji->to_keyword() != Keyword::Normal) | 
					
						
							|  |  |  |             values.append(emoji->to_string(mode)); | 
					
						
							| 
									
										
										
										
											2024-12-05 01:19:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (values.is_empty()) | 
					
						
							| 
									
										
										
										
											2025-02-06 17:21:34 +00:00
										 |  |  |             return "normal"_string; | 
					
						
							|  |  |  |         return MUST(String::join(' ', values)); | 
					
						
							| 
									
										
										
										
											2024-12-05 01:19:03 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-20 14:47:37 +01:00
										 |  |  |     case PropertyID::GridArea: { | 
					
						
							|  |  |  |         auto& row_start = longhand(PropertyID::GridRowStart)->as_grid_track_placement(); | 
					
						
							|  |  |  |         auto& column_start = longhand(PropertyID::GridColumnStart)->as_grid_track_placement(); | 
					
						
							|  |  |  |         auto& row_end = longhand(PropertyID::GridRowEnd)->as_grid_track_placement(); | 
					
						
							|  |  |  |         auto& column_end = longhand(PropertyID::GridColumnEnd)->as_grid_track_placement(); | 
					
						
							|  |  |  |         StringBuilder builder; | 
					
						
							|  |  |  |         if (!row_start.grid_track_placement().is_auto()) | 
					
						
							| 
									
										
										
										
											2025-08-04 22:04:18 +12:00
										 |  |  |             builder.appendff("{}", row_start.grid_track_placement().to_string(mode)); | 
					
						
							| 
									
										
										
										
											2023-09-20 14:47:37 +01:00
										 |  |  |         if (!column_start.grid_track_placement().is_auto()) | 
					
						
							| 
									
										
										
										
											2025-08-04 22:04:18 +12:00
										 |  |  |             builder.appendff(" / {}", column_start.grid_track_placement().to_string(mode)); | 
					
						
							| 
									
										
										
										
											2023-09-20 14:47:37 +01:00
										 |  |  |         if (!row_end.grid_track_placement().is_auto()) | 
					
						
							| 
									
										
										
										
											2025-08-04 22:04:18 +12:00
										 |  |  |             builder.appendff(" / {}", row_end.grid_track_placement().to_string(mode)); | 
					
						
							| 
									
										
										
										
											2023-09-20 14:47:37 +01:00
										 |  |  |         if (!column_end.grid_track_placement().is_auto()) | 
					
						
							| 
									
										
										
										
											2025-08-04 22:04:18 +12:00
										 |  |  |             builder.appendff(" / {}", column_end.grid_track_placement().to_string(mode)); | 
					
						
							| 
									
										
										
										
											2025-03-20 14:14:44 +00:00
										 |  |  |         if (builder.is_empty()) | 
					
						
							|  |  |  |             return "auto"_string; | 
					
						
							| 
									
										
										
										
											2023-09-20 14:47:37 +01:00
										 |  |  |         return MUST(builder.to_string()); | 
					
						
							| 
									
										
										
										
											2023-09-20 14:50:44 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |         // FIXME: Serialize Grid differently once we support it better!
 | 
					
						
							|  |  |  |     case PropertyID::Grid: | 
					
						
							|  |  |  |     case PropertyID::GridTemplate: { | 
					
						
							|  |  |  |         auto& areas = longhand(PropertyID::GridTemplateAreas)->as_grid_template_area(); | 
					
						
							|  |  |  |         auto& rows = longhand(PropertyID::GridTemplateRows)->as_grid_track_size_list(); | 
					
						
							|  |  |  |         auto& columns = longhand(PropertyID::GridTemplateColumns)->as_grid_track_size_list(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-12 20:39:03 +12:00
										 |  |  |         if (areas.grid_template_area().size() == 0 && rows.grid_track_size_list().track_list().size() == 0 && columns.grid_track_size_list().track_list().size() == 0) | 
					
						
							|  |  |  |             return "none"_string; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-20 14:50:44 +01:00
										 |  |  |         auto construct_rows_string = [&]() { | 
					
						
							|  |  |  |             StringBuilder builder; | 
					
						
							|  |  |  |             size_t idx = 0; | 
					
						
							|  |  |  |             for (auto const& row : rows.grid_track_size_list().track_list()) { | 
					
						
							|  |  |  |                 if (areas.grid_template_area().size() > idx) { | 
					
						
							|  |  |  |                     builder.append("\""sv); | 
					
						
							|  |  |  |                     for (size_t y = 0; y < areas.grid_template_area()[idx].size(); ++y) { | 
					
						
							|  |  |  |                         builder.append(areas.grid_template_area()[idx][y]); | 
					
						
							|  |  |  |                         if (y != areas.grid_template_area()[idx].size() - 1) | 
					
						
							|  |  |  |                             builder.append(" "sv); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     builder.append("\" "sv); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-08-03 15:37:42 +12:00
										 |  |  |                 builder.append(row.to_string(mode)); | 
					
						
							| 
									
										
										
										
											2023-09-20 14:50:44 +01:00
										 |  |  |                 if (idx < rows.grid_track_size_list().track_list().size() - 1) | 
					
						
							|  |  |  |                     builder.append(' '); | 
					
						
							|  |  |  |                 idx++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return MUST(builder.to_string()); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (columns.grid_track_size_list().track_list().size() == 0) | 
					
						
							|  |  |  |             return MUST(String::formatted("{}", construct_rows_string())); | 
					
						
							| 
									
										
										
										
											2025-08-03 15:37:42 +12:00
										 |  |  |         return MUST(String::formatted("{} / {}", construct_rows_string(), columns.grid_track_size_list().to_string(mode))); | 
					
						
							| 
									
										
										
										
											2023-09-20 14:47:37 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-20 16:19:37 +01:00
										 |  |  |     case PropertyID::GridColumn: { | 
					
						
							|  |  |  |         auto start = longhand(PropertyID::GridColumnStart); | 
					
						
							|  |  |  |         auto end = longhand(PropertyID::GridColumnEnd); | 
					
						
							| 
									
										
										
										
											2025-03-18 15:08:05 +00:00
										 |  |  |         if (end->as_grid_track_placement().grid_track_placement().is_auto() || start == end) | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  |             return start->to_string(mode); | 
					
						
							|  |  |  |         return MUST(String::formatted("{} / {}", start->to_string(mode), end->to_string(mode))); | 
					
						
							| 
									
										
										
										
											2023-09-20 16:19:37 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     case PropertyID::GridRow: { | 
					
						
							|  |  |  |         auto start = longhand(PropertyID::GridRowStart); | 
					
						
							|  |  |  |         auto end = longhand(PropertyID::GridRowEnd); | 
					
						
							| 
									
										
										
										
											2025-03-18 15:03:35 +00:00
										 |  |  |         if (end->as_grid_track_placement().grid_track_placement().is_auto() || start == end) | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  |             return start->to_string(mode); | 
					
						
							|  |  |  |         return MUST(String::formatted("{} / {}", start->to_string(mode), end->to_string(mode))); | 
					
						
							| 
									
										
										
										
											2023-09-20 16:19:37 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-07-18 20:36:15 +02:00
										 |  |  |     case PropertyID::Mask: { | 
					
						
							|  |  |  |         StringBuilder builder; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto serialize_layer = [mode, &builder](String image_value_string, String position_value_string, String size_value_string, String repeat_value_string, String origin_value_string, String clip_value_string, String composite_value_string, String mode_value_string) { | 
					
						
							|  |  |  |             PropertyID canonical_property_order[] = { | 
					
						
							|  |  |  |                 PropertyID::MaskImage, | 
					
						
							|  |  |  |                 PropertyID::MaskPosition, | 
					
						
							|  |  |  |                 // Intentionally skipping MaskSize here, it is handled together with MaskPosition.
 | 
					
						
							|  |  |  |                 PropertyID::MaskRepeat, | 
					
						
							|  |  |  |                 PropertyID::MaskOrigin, | 
					
						
							|  |  |  |                 PropertyID::MaskClip, | 
					
						
							|  |  |  |                 PropertyID::MaskComposite, | 
					
						
							|  |  |  |                 PropertyID::MaskMode, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto property_value = [&](PropertyID property) -> String const& { | 
					
						
							|  |  |  |                 switch (property) { | 
					
						
							|  |  |  |                 case PropertyID::MaskImage: | 
					
						
							|  |  |  |                     return image_value_string; | 
					
						
							|  |  |  |                 case PropertyID::MaskPosition: | 
					
						
							|  |  |  |                     return position_value_string; | 
					
						
							|  |  |  |                 case PropertyID::MaskSize: | 
					
						
							|  |  |  |                     return size_value_string; | 
					
						
							|  |  |  |                 case PropertyID::MaskRepeat: | 
					
						
							|  |  |  |                     return repeat_value_string; | 
					
						
							|  |  |  |                 case PropertyID::MaskOrigin: | 
					
						
							|  |  |  |                     return origin_value_string; | 
					
						
							|  |  |  |                 case PropertyID::MaskClip: | 
					
						
							|  |  |  |                     return clip_value_string; | 
					
						
							|  |  |  |                 case PropertyID::MaskComposite: | 
					
						
							|  |  |  |                     return composite_value_string; | 
					
						
							|  |  |  |                 case PropertyID::MaskMode: | 
					
						
							|  |  |  |                     return mode_value_string; | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto is_initial_value = [mode, property_value](PropertyID property) -> bool { | 
					
						
							|  |  |  |                 return property_value(property) == property_initial_value(property)->to_string(mode); | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto can_skip_serializing_initial_value = [is_initial_value, property_value](PropertyID property) -> bool { | 
					
						
							|  |  |  |                 switch (property) { | 
					
						
							|  |  |  |                 case PropertyID::MaskPosition: | 
					
						
							|  |  |  |                     return is_initial_value(PropertyID::MaskSize); | 
					
						
							|  |  |  |                 case PropertyID::MaskOrigin: | 
					
						
							|  |  |  |                     return is_initial_value(PropertyID::MaskClip) || property_value(PropertyID::MaskClip) == string_from_keyword(Keyword::NoClip); | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     return true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             bool layer_is_empty = true; | 
					
						
							|  |  |  |             for (size_t i = 0; i < array_size(canonical_property_order); i++) { | 
					
						
							|  |  |  |                 auto property = canonical_property_order[i]; | 
					
						
							|  |  |  |                 auto const& value = property_value(property); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (is_initial_value(property) && can_skip_serializing_initial_value(property)) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 if (property == PropertyID::MaskClip && value == property_value(PropertyID::MaskOrigin)) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (!layer_is_empty) | 
					
						
							|  |  |  |                     builder.append(" "sv); | 
					
						
							|  |  |  |                 builder.append(value); | 
					
						
							|  |  |  |                 if (property == PropertyID::MaskPosition && !is_initial_value(PropertyID::MaskSize)) { | 
					
						
							|  |  |  |                     builder.append(" / "sv); | 
					
						
							|  |  |  |                     builder.append(property_value(PropertyID::MaskSize)); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 layer_is_empty = false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (layer_is_empty) | 
					
						
							|  |  |  |                 builder.append("none"sv); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto get_layer_count = [](auto const& style_value) -> size_t { | 
					
						
							|  |  |  |             return style_value->is_value_list() ? style_value->as_value_list().size() : 1; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto mask_image = longhand(PropertyID::MaskImage); | 
					
						
							|  |  |  |         auto mask_position = longhand(PropertyID::MaskPosition); | 
					
						
							|  |  |  |         auto mask_size = longhand(PropertyID::MaskSize); | 
					
						
							|  |  |  |         auto mask_repeat = longhand(PropertyID::MaskRepeat); | 
					
						
							|  |  |  |         auto mask_origin = longhand(PropertyID::MaskOrigin); | 
					
						
							|  |  |  |         auto mask_clip = longhand(PropertyID::MaskClip); | 
					
						
							|  |  |  |         auto mask_composite = longhand(PropertyID::MaskComposite); | 
					
						
							|  |  |  |         auto mask_mode = longhand(PropertyID::MaskMode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto layer_count = max(get_layer_count(mask_image), max(get_layer_count(mask_position), max(get_layer_count(mask_size), max(get_layer_count(mask_repeat), max(get_layer_count(mask_origin), max(get_layer_count(mask_clip), max(get_layer_count(mask_composite), get_layer_count(mask_mode)))))))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (layer_count == 1) { | 
					
						
							|  |  |  |             serialize_layer(mask_image->to_string(mode), mask_position->to_string(mode), mask_size->to_string(mode), mask_repeat->to_string(mode), mask_origin->to_string(mode), mask_clip->to_string(mode), mask_composite->to_string(mode), mask_mode->to_string(mode)); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |             auto get_layer_value_string = [mode](ValueComparingRefPtr<StyleValue const> const& style_value, size_t index) { | 
					
						
							| 
									
										
										
										
											2025-07-18 20:36:15 +02:00
										 |  |  |                 if (style_value->is_value_list()) | 
					
						
							|  |  |  |                     return style_value->as_value_list().value_at(index, true)->to_string(mode); | 
					
						
							|  |  |  |                 return style_value->to_string(mode); | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (size_t i = 0; i < layer_count; i++) { | 
					
						
							|  |  |  |                 if (i) | 
					
						
							|  |  |  |                     builder.append(", "sv); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 serialize_layer(get_layer_value_string(mask_image, i), get_layer_value_string(mask_position, i), get_layer_value_string(mask_size, i), get_layer_value_string(mask_repeat, i), get_layer_value_string(mask_origin, i), get_layer_value_string(mask_clip, i), get_layer_value_string(mask_composite, i), get_layer_value_string(mask_mode, i)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return builder.to_string_without_validation(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-07-10 22:23:07 +12:00
										 |  |  |     case PropertyID::PlaceContent: | 
					
						
							|  |  |  |     case PropertyID::PlaceItems: | 
					
						
							|  |  |  |     case PropertyID::PlaceSelf: | 
					
						
							|  |  |  |         return positional_value_list_shorthand_to_string(m_properties.values); | 
					
						
							| 
									
										
										
										
											2024-11-29 14:49:31 +00:00
										 |  |  |     case PropertyID::TextDecoration: { | 
					
						
							|  |  |  |         // The rule here seems to be, only print what's different from the default value,
 | 
					
						
							|  |  |  |         // but if they're all default, print the line.
 | 
					
						
							|  |  |  |         StringBuilder builder; | 
					
						
							|  |  |  |         auto append_if_non_default = [&](PropertyID property_id) { | 
					
						
							|  |  |  |             auto value = longhand(property_id); | 
					
						
							| 
									
										
										
										
											2024-12-05 10:58:21 +00:00
										 |  |  |             if (!value->equals(property_initial_value(property_id))) { | 
					
						
							| 
									
										
										
										
											2024-11-29 14:49:31 +00:00
										 |  |  |                 if (!builder.is_empty()) | 
					
						
							|  |  |  |                     builder.append(' '); | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  |                 builder.append(value->to_string(mode)); | 
					
						
							| 
									
										
										
										
											2024-11-29 14:49:31 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         append_if_non_default(PropertyID::TextDecorationLine); | 
					
						
							|  |  |  |         append_if_non_default(PropertyID::TextDecorationThickness); | 
					
						
							|  |  |  |         append_if_non_default(PropertyID::TextDecorationStyle); | 
					
						
							|  |  |  |         append_if_non_default(PropertyID::TextDecorationColor); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (builder.is_empty()) | 
					
						
							| 
									
										
										
										
											2024-12-07 00:59:49 +01:00
										 |  |  |             return longhand(PropertyID::TextDecorationLine)->to_string(mode); | 
					
						
							| 
									
										
										
										
											2024-11-29 14:49:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return builder.to_string_without_validation(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-05-22 00:31:24 +12:00
										 |  |  |     case PropertyID::WhiteSpace: { | 
					
						
							|  |  |  |         auto white_space_collapse_property = longhand(PropertyID::WhiteSpaceCollapse); | 
					
						
							|  |  |  |         auto text_wrap_mode_property = longhand(PropertyID::TextWrapMode); | 
					
						
							|  |  |  |         auto white_space_trim_property = longhand(PropertyID::WhiteSpaceTrim); | 
					
						
							| 
									
										
										
										
											2025-03-20 10:58:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |         RefPtr<StyleValue const> value; | 
					
						
							| 
									
										
										
										
											2025-05-22 00:31:24 +12:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (white_space_trim_property->is_keyword() && white_space_trim_property->as_keyword().keyword() == Keyword::None) { | 
					
						
							|  |  |  |             auto white_space_collapse_keyword = white_space_collapse_property->as_keyword().keyword(); | 
					
						
							|  |  |  |             auto text_wrap_mode_keyword = text_wrap_mode_property->as_keyword().keyword(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (white_space_collapse_keyword == Keyword::Collapse && text_wrap_mode_keyword == Keyword::Wrap) | 
					
						
							|  |  |  |                 return "normal"_string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (white_space_collapse_keyword == Keyword::Preserve && text_wrap_mode_keyword == Keyword::Nowrap) | 
					
						
							|  |  |  |                 return "pre"_string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (white_space_collapse_keyword == Keyword::Preserve && text_wrap_mode_keyword == Keyword::Wrap) | 
					
						
							|  |  |  |                 return "pre-wrap"_string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (white_space_collapse_keyword == Keyword::PreserveBreaks && text_wrap_mode_keyword == Keyword::Wrap) | 
					
						
							|  |  |  |                 return "pre-line"_string; | 
					
						
							| 
									
										
										
										
											2023-09-19 16:26:23 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-04-06 23:34:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 00:31:24 +12:00
										 |  |  |         return default_to_string(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2025-07-10 22:23:07 +12:00
										 |  |  |         if (property_is_positional_value_list_shorthand(m_properties.shorthand_property)) | 
					
						
							|  |  |  |             return positional_value_list_shorthand_to_string(m_properties.values); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 00:31:24 +12:00
										 |  |  |         return default_to_string(); | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-10 16:35:59 +01:00
										 |  |  | void ShorthandStyleValue::set_style_sheet(GC::Ptr<CSSStyleSheet> style_sheet) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Base::set_style_sheet(style_sheet); | 
					
						
							|  |  |  |     for (auto& value : m_properties.values) | 
					
						
							| 
									
										
										
										
											2025-08-08 10:11:51 +01:00
										 |  |  |         const_cast<StyleValue&>(*value).set_style_sheet(style_sheet); | 
					
						
							| 
									
										
										
										
											2025-04-10 16:35:59 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 23:16:43 +03:30
										 |  |  | } |