| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |  * Copyright (c) 2018-2022, Andreas Kling <kling@serenityos.org> | 
					
						
							| 
									
										
										
										
											2021-02-21 13:45:26 +02:00
										 |  |  |  |  * Copyright (c) 2021, the SerenityOS developers. | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |  * Copyright (c) 2021-2022, Sam Atkins <atkinssj@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
										 |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  | #include <AK/Debug.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-10 16:41:30 +02:00
										 |  |  |  | #include <AK/QuickSort.h>
 | 
					
						
							| 
									
										
										
										
											2021-12-03 20:00:31 +00:00
										 |  |  |  | #include <AK/TemporaryChange.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | #include <LibGfx/Font.h>
 | 
					
						
							|  |  |  |  | #include <LibGfx/FontDatabase.h>
 | 
					
						
							| 
									
										
										
										
											2022-02-19 21:21:20 +01:00
										 |  |  |  | #include <LibGfx/FontStyleMapping.h>
 | 
					
						
							| 
									
										
										
										
											2021-03-07 15:00:02 +01:00
										 |  |  |  | #include <LibWeb/CSS/CSSStyleRule.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-14 19:57:02 +01:00
										 |  |  |  | #include <LibWeb/CSS/Parser/Parser.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  |  | #include <LibWeb/CSS/SelectorEngine.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-24 13:49:57 +02:00
										 |  |  |  | #include <LibWeb/CSS/StyleComputer.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  |  | #include <LibWeb/CSS/StyleSheet.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/DOM/Element.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | #include <LibWeb/FontCache.h>
 | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2019-06-27 17:47:59 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 20:01:35 +02:00
										 |  |  |  | namespace Web::CSS { | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 13:49:57 +02:00
										 |  |  |  | StyleComputer::StyleComputer(DOM::Document& document) | 
					
						
							| 
									
										
										
										
											2019-06-27 17:47:59 +02:00
										 |  |  |  |     : m_document(document) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 13:49:57 +02:00
										 |  |  |  | StyleComputer::~StyleComputer() | 
					
						
							| 
									
										
										
										
											2019-06-27 17:47:59 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-05 09:01:12 +02:00
										 |  |  |  | static StyleSheet& default_stylesheet() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     static StyleSheet* sheet; | 
					
						
							|  |  |  |  |     if (!sheet) { | 
					
						
							| 
									
										
										
										
											2021-07-14 16:56:11 +01:00
										 |  |  |  |         extern char const default_stylesheet_source[]; | 
					
						
							| 
									
										
										
										
											2019-10-05 09:01:12 +02:00
										 |  |  |  |         String css = default_stylesheet_source; | 
					
						
							| 
									
										
										
										
											2021-07-30 19:31:46 +01:00
										 |  |  |  |         sheet = parse_css(CSS::ParsingContext(), css).leak_ref(); | 
					
						
							| 
									
										
										
										
											2019-10-05 09:01:12 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     return *sheet; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-24 10:33:33 +02:00
										 |  |  |  | static StyleSheet& quirks_mode_stylesheet() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     static StyleSheet* sheet; | 
					
						
							|  |  |  |  |     if (!sheet) { | 
					
						
							| 
									
										
										
										
											2021-07-14 16:56:11 +01:00
										 |  |  |  |         extern char const quirks_mode_stylesheet_source[]; | 
					
						
							| 
									
										
										
										
											2020-09-24 10:33:33 +02:00
										 |  |  |  |         String css = quirks_mode_stylesheet_source; | 
					
						
							| 
									
										
										
										
											2021-07-30 19:31:46 +01:00
										 |  |  |  |         sheet = parse_css(CSS::ParsingContext(), css).leak_ref(); | 
					
						
							| 
									
										
										
										
											2020-09-24 10:33:33 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     return *sheet; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-05 09:01:12 +02:00
										 |  |  |  | template<typename Callback> | 
					
						
							| 
									
										
										
										
											2021-09-24 13:49:57 +02:00
										 |  |  |  | void StyleComputer::for_each_stylesheet(CascadeOrigin cascade_origin, Callback callback) const | 
					
						
							| 
									
										
										
										
											2019-10-05 09:01:12 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-11 12:08:27 +01:00
										 |  |  |  |     if (cascade_origin == CascadeOrigin::UserAgent) { | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |         callback(default_stylesheet()); | 
					
						
							|  |  |  |  |         if (document().in_quirks_mode()) | 
					
						
							|  |  |  |  |             callback(quirks_mode_stylesheet()); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-11 12:08:27 +01:00
										 |  |  |  |     if (cascade_origin == CascadeOrigin::Author) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |         for (auto const& sheet : document().style_sheets().sheets()) { | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |             callback(sheet); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-10-05 09:01:12 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | Vector<MatchingRule> StyleComputer::collect_matching_rules(DOM::Element const& element, CascadeOrigin cascade_origin, Optional<CSS::Selector::PseudoElement> pseudo_element) const | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |     if (cascade_origin == CascadeOrigin::Author) { | 
					
						
							|  |  |  |  |         Vector<MatchingRule> rules_to_run; | 
					
						
							| 
									
										
										
										
											2022-02-25 17:50:06 +00:00
										 |  |  |  |         if (pseudo_element.has_value()) { | 
					
						
							|  |  |  |  |             if (auto it = m_rule_cache->rules_by_pseudo_element.find(pseudo_element.value()); it != m_rule_cache->rules_by_pseudo_element.end()) | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |                 rules_to_run.extend(it->value); | 
					
						
							| 
									
										
										
										
											2022-02-25 17:50:06 +00:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |             for (auto const& class_name : element.class_names()) { | 
					
						
							|  |  |  |  |                 if (auto it = m_rule_cache->rules_by_class.find(class_name); it != m_rule_cache->rules_by_class.end()) | 
					
						
							|  |  |  |  |                     rules_to_run.extend(it->value); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (auto id = element.get_attribute(HTML::AttributeNames::id); !id.is_null()) { | 
					
						
							|  |  |  |  |                 if (auto it = m_rule_cache->rules_by_id.find(id); it != m_rule_cache->rules_by_id.end()) | 
					
						
							|  |  |  |  |                     rules_to_run.extend(it->value); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (auto it = m_rule_cache->rules_by_tag_name.find(element.local_name()); it != m_rule_cache->rules_by_tag_name.end()) | 
					
						
							| 
									
										
										
										
											2022-02-10 18:50:58 +01:00
										 |  |  |  |                 rules_to_run.extend(it->value); | 
					
						
							| 
									
										
										
										
											2022-02-25 17:50:06 +00:00
										 |  |  |  |             rules_to_run.extend(m_rule_cache->other_rules); | 
					
						
							| 
									
										
										
										
											2022-02-10 18:50:58 +01:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         Vector<MatchingRule> matching_rules; | 
					
						
							|  |  |  |  |         for (auto const& rule_to_run : rules_to_run) { | 
					
						
							|  |  |  |  |             auto const& selector = rule_to_run.rule->selectors()[rule_to_run.selector_index]; | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |             if (SelectorEngine::matches(selector, element, pseudo_element)) | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |                 matching_rules.append(rule_to_run); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         return matching_rules; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-05 09:01:12 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |     Vector<MatchingRule> matching_rules; | 
					
						
							| 
									
										
										
										
											2020-06-13 00:44:26 +02:00
										 |  |  |  |     size_t style_sheet_index = 0; | 
					
						
							| 
									
										
										
										
											2022-02-09 20:06:16 +01:00
										 |  |  |  |     for_each_stylesheet(cascade_origin, [&](auto& sheet) { | 
					
						
							| 
									
										
										
										
											2020-06-13 00:44:26 +02:00
										 |  |  |  |         size_t rule_index = 0; | 
					
						
							| 
									
										
										
										
											2021-09-30 22:57:35 +02:00
										 |  |  |  |         static_cast<CSSStyleSheet const&>(sheet).for_each_effective_style_rule([&](auto const& rule) { | 
					
						
							| 
									
										
										
										
											2020-06-13 00:44:26 +02:00
										 |  |  |  |             size_t selector_index = 0; | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  |  |             for (auto& selector : rule.selectors()) { | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |                 if (SelectorEngine::matches(selector, element, pseudo_element)) { | 
					
						
							| 
									
										
										
										
											2021-05-24 23:01:24 +02:00
										 |  |  |  |                     matching_rules.append({ rule, style_sheet_index, rule_index, selector_index, selector.specificity() }); | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  |  |                     break; | 
					
						
							|  |  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-06-13 00:44:26 +02:00
										 |  |  |  |                 ++selector_index; | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-06-13 00:44:26 +02:00
										 |  |  |  |             ++rule_index; | 
					
						
							| 
									
										
										
										
											2021-02-21 13:45:26 +02:00
										 |  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2020-06-13 00:44:26 +02:00
										 |  |  |  |         ++style_sheet_index; | 
					
						
							| 
									
										
										
										
											2021-03-07 17:53:20 +01:00
										 |  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-09-25 12:42:10 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  |  |     return matching_rules; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  | static void sort_matching_rules(Vector<MatchingRule>& matching_rules) | 
					
						
							| 
									
										
										
										
											2020-12-14 21:31:10 +00:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     quick_sort(matching_rules, [&](MatchingRule& a, MatchingRule& b) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |         auto const& a_selector = a.rule->selectors()[a.selector_index]; | 
					
						
							|  |  |  |  |         auto const& b_selector = b.rule->selectors()[b.selector_index]; | 
					
						
							| 
									
										
										
										
											2020-12-14 21:31:10 +00:00
										 |  |  |  |         auto a_specificity = a_selector.specificity(); | 
					
						
							|  |  |  |  |         auto b_specificity = b_selector.specificity(); | 
					
						
							|  |  |  |  |         if (a_selector.specificity() == b_selector.specificity()) { | 
					
						
							|  |  |  |  |             if (a.style_sheet_index == b.style_sheet_index) | 
					
						
							|  |  |  |  |                 return a.rule_index < b.rule_index; | 
					
						
							|  |  |  |  |             return a.style_sheet_index < b.style_sheet_index; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         return a_specificity < b_specificity; | 
					
						
							|  |  |  |  |     }); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 16:14:31 +02:00
										 |  |  |  | enum class Edge { | 
					
						
							|  |  |  |  |     Top, | 
					
						
							|  |  |  |  |     Right, | 
					
						
							|  |  |  |  |     Bottom, | 
					
						
							|  |  |  |  |     Left, | 
					
						
							|  |  |  |  |     All, | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static bool contains(Edge a, Edge b) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return a == b || b == Edge::All; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-04 17:10:12 +00:00
										 |  |  |  | static void set_property_expanding_shorthands(StyleProperties& style, CSS::PropertyID property_id, StyleValue const& value, DOM::Document& document) | 
					
						
							| 
									
										
										
										
											2019-11-18 11:49:19 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |     auto assign_edge_values = [&style](PropertyID top_property, PropertyID right_property, PropertyID bottom_property, PropertyID left_property, auto const& values) { | 
					
						
							|  |  |  |  |         if (values.size() == 4) { | 
					
						
							|  |  |  |  |             style.set_property(top_property, values[0]); | 
					
						
							|  |  |  |  |             style.set_property(right_property, values[1]); | 
					
						
							|  |  |  |  |             style.set_property(bottom_property, values[2]); | 
					
						
							|  |  |  |  |             style.set_property(left_property, values[3]); | 
					
						
							|  |  |  |  |         } else if (values.size() == 3) { | 
					
						
							|  |  |  |  |             style.set_property(top_property, values[0]); | 
					
						
							|  |  |  |  |             style.set_property(right_property, values[1]); | 
					
						
							|  |  |  |  |             style.set_property(bottom_property, values[2]); | 
					
						
							|  |  |  |  |             style.set_property(left_property, values[1]); | 
					
						
							|  |  |  |  |         } else if (values.size() == 2) { | 
					
						
							|  |  |  |  |             style.set_property(top_property, values[0]); | 
					
						
							|  |  |  |  |             style.set_property(right_property, values[1]); | 
					
						
							|  |  |  |  |             style.set_property(bottom_property, values[0]); | 
					
						
							|  |  |  |  |             style.set_property(left_property, values[1]); | 
					
						
							|  |  |  |  |         } else if (values.size() == 1) { | 
					
						
							|  |  |  |  |             style.set_property(top_property, values[0]); | 
					
						
							|  |  |  |  |             style.set_property(right_property, values[0]); | 
					
						
							|  |  |  |  |             style.set_property(bottom_property, values[0]); | 
					
						
							|  |  |  |  |             style.set_property(left_property, values[0]); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 13:36:27 +01:00
										 |  |  |  |     if (property_id == CSS::PropertyID::TextDecoration) { | 
					
						
							| 
									
										
										
										
											2021-08-04 12:34:14 +01:00
										 |  |  |  |         if (value.is_text_decoration()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& text_decoration = value.as_text_decoration(); | 
					
						
							| 
									
										
										
										
											2021-08-04 12:34:14 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::TextDecorationLine, text_decoration.line()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::TextDecorationStyle, text_decoration.style()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::TextDecorationColor, text_decoration.color()); | 
					
						
							| 
									
										
										
										
											2021-07-22 14:56:44 +01:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::TextDecorationLine, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::TextDecorationStyle, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::TextDecorationColor, value); | 
					
						
							| 
									
										
										
										
											2020-12-15 13:36:27 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 15:20:31 +01:00
										 |  |  |  |     if (property_id == CSS::PropertyID::Overflow) { | 
					
						
							| 
									
										
										
										
											2021-08-09 14:54:40 +01:00
										 |  |  |  |         if (value.is_overflow()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& overflow = value.as_overflow(); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:54:40 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::OverflowX, overflow.overflow_x()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::OverflowY, overflow.overflow_y()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::OverflowX, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::OverflowY, value); | 
					
						
							| 
									
										
										
										
											2021-02-22 15:20:31 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:39 +01:00
										 |  |  |  |     if (property_id == CSS::PropertyID::Border) { | 
					
						
							| 
									
										
										
										
											2020-06-10 17:47:04 +02:00
										 |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BorderTop, value, document); | 
					
						
							|  |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BorderRight, value, document); | 
					
						
							|  |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BorderBottom, value, document); | 
					
						
							|  |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BorderLeft, value, document); | 
					
						
							| 
									
										
										
										
											2021-08-05 21:11:38 +01:00
										 |  |  |  |         // FIXME: Also reset border-image, in line with the spec: https://www.w3.org/TR/css-backgrounds-3/#border-shorthands
 | 
					
						
							| 
									
										
										
										
											2020-12-19 19:30:49 +01:00
										 |  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2020-06-10 16:14:31 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 22:31:52 +02:00
										 |  |  |  |     if (property_id == CSS::PropertyID::BorderRadius) { | 
					
						
							| 
									
										
										
										
											2021-08-06 16:55:08 +01:00
										 |  |  |  |         if (value.is_value_list()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& values_list = value.as_value_list(); | 
					
						
							| 
									
										
										
										
											2021-08-06 16:55:08 +01:00
										 |  |  |  |             assign_edge_values(PropertyID::BorderTopLeftRadius, PropertyID::BorderTopRightRadius, PropertyID::BorderBottomRightRadius, PropertyID::BorderBottomLeftRadius, values_list.values()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderTopLeftRadius, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderTopRightRadius, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderBottomRightRadius, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderBottomLeftRadius, value); | 
					
						
							| 
									
										
										
										
											2021-05-14 22:31:52 +02:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 16:14:31 +02:00
										 |  |  |  |     if (property_id == CSS::PropertyID::BorderTop | 
					
						
							|  |  |  |  |         || property_id == CSS::PropertyID::BorderRight | 
					
						
							|  |  |  |  |         || property_id == CSS::PropertyID::BorderBottom | 
					
						
							|  |  |  |  |         || property_id == CSS::PropertyID::BorderLeft) { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         Edge edge = Edge::All; | 
					
						
							|  |  |  |  |         switch (property_id) { | 
					
						
							|  |  |  |  |         case CSS::PropertyID::BorderTop: | 
					
						
							|  |  |  |  |             edge = Edge::Top; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::PropertyID::BorderRight: | 
					
						
							|  |  |  |  |             edge = Edge::Right; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::PropertyID::BorderBottom: | 
					
						
							|  |  |  |  |             edge = Edge::Bottom; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::PropertyID::BorderLeft: | 
					
						
							|  |  |  |  |             edge = Edge::Left; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-05 21:11:38 +01:00
										 |  |  |  |         if (value.is_border()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& border = value.as_border(); | 
					
						
							| 
									
										
										
										
											2021-08-05 21:11:38 +01:00
										 |  |  |  |             if (contains(Edge::Top, edge)) { | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderTopWidth, border.border_width()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderTopStyle, border.border_style()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderTopColor, border.border_color()); | 
					
						
							| 
									
										
										
										
											2021-07-14 19:57:02 +01:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-08-05 21:11:38 +01:00
										 |  |  |  |             if (contains(Edge::Right, edge)) { | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderRightWidth, border.border_width()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderRightStyle, border.border_style()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderRightColor, border.border_color()); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (contains(Edge::Bottom, edge)) { | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderBottomWidth, border.border_width()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderBottomStyle, border.border_style()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderBottomColor, border.border_color()); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (contains(Edge::Left, edge)) { | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderLeftWidth, border.border_width()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderLeftStyle, border.border_style()); | 
					
						
							|  |  |  |  |                 style.set_property(PropertyID::BorderLeftColor, border.border_color()); | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:39 +01:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-27 20:51:15 +01:00
										 |  |  |  |     if (property_id == CSS::PropertyID::BorderStyle) { | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |         if (value.is_value_list()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& values_list = value.as_value_list(); | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |             assign_edge_values(PropertyID::BorderTopStyle, PropertyID::BorderRightStyle, PropertyID::BorderBottomStyle, PropertyID::BorderLeftStyle, values_list.values()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderTopStyle, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderRightStyle, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderBottomStyle, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderLeftStyle, value); | 
					
						
							| 
									
										
										
										
											2019-11-27 20:51:15 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (property_id == CSS::PropertyID::BorderWidth) { | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |         if (value.is_value_list()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& values_list = value.as_value_list(); | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |             assign_edge_values(PropertyID::BorderTopWidth, PropertyID::BorderRightWidth, PropertyID::BorderBottomWidth, PropertyID::BorderLeftWidth, values_list.values()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderTopWidth, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderRightWidth, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderBottomWidth, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderLeftWidth, value); | 
					
						
							| 
									
										
										
										
											2019-11-27 20:51:15 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (property_id == CSS::PropertyID::BorderColor) { | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |         if (value.is_value_list()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& values_list = value.as_value_list(); | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |             assign_edge_values(PropertyID::BorderTopColor, PropertyID::BorderRightColor, PropertyID::BorderBottomColor, PropertyID::BorderLeftColor, values_list.values()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderTopColor, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderRightColor, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderBottomColor, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::BorderLeftColor, value); | 
					
						
							| 
									
										
										
										
											2019-11-27 20:51:15 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-30 11:58:05 +02:00
										 |  |  |  |     if (property_id == CSS::PropertyID::Background) { | 
					
						
							| 
									
										
										
										
											2021-11-11 11:52:33 +00:00
										 |  |  |  |         if (value.is_background()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& background = value.as_background(); | 
					
						
							| 
									
										
										
										
											2021-08-03 14:56:08 +01:00
										 |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundColor, background.color(), document); | 
					
						
							|  |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundImage, background.image(), document); | 
					
						
							| 
									
										
										
										
											2021-11-03 16:30:55 +00:00
										 |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundPosition, background.position(), document); | 
					
						
							| 
									
										
										
										
											2021-11-05 16:15:02 +00:00
										 |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundSize, background.size(), document); | 
					
						
							| 
									
										
										
										
											2021-11-04 15:56:10 +00:00
										 |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundRepeat, background.repeat(), document); | 
					
						
							| 
									
										
										
										
											2021-11-03 17:46:45 +00:00
										 |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundAttachment, background.attachment(), document); | 
					
						
							| 
									
										
										
										
											2021-11-03 20:28:34 +00:00
										 |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundOrigin, background.origin(), document); | 
					
						
							|  |  |  |  |             set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundClip, background.clip(), document); | 
					
						
							| 
									
										
										
										
											2021-07-21 17:47:50 +01:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-07-14 19:57:02 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundColor, value, document); | 
					
						
							|  |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundImage, value, document); | 
					
						
							| 
									
										
										
										
											2021-11-03 16:30:55 +00:00
										 |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundPosition, value, document); | 
					
						
							| 
									
										
										
										
											2021-11-05 16:15:02 +00:00
										 |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundSize, value, document); | 
					
						
							| 
									
										
										
										
											2021-11-04 15:56:10 +00:00
										 |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundRepeat, value, document); | 
					
						
							| 
									
										
										
										
											2021-11-03 17:46:45 +00:00
										 |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundAttachment, value, document); | 
					
						
							| 
									
										
										
										
											2021-11-03 20:28:34 +00:00
										 |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundOrigin, value, document); | 
					
						
							|  |  |  |  |         set_property_expanding_shorthands(style, CSS::PropertyID::BackgroundClip, value, document); | 
					
						
							| 
									
										
										
										
											2021-11-03 17:46:45 +00:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-18 11:49:19 +01:00
										 |  |  |  |     if (property_id == CSS::PropertyID::Margin) { | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |         if (value.is_value_list()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& values_list = value.as_value_list(); | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |             assign_edge_values(PropertyID::MarginTop, PropertyID::MarginRight, PropertyID::MarginBottom, PropertyID::MarginLeft, values_list.values()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::MarginTop, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::MarginRight, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::MarginBottom, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::MarginLeft, value); | 
					
						
							| 
									
										
										
										
											2019-11-18 11:49:19 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-18 12:25:22 +01:00
										 |  |  |  |     if (property_id == CSS::PropertyID::Padding) { | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |         if (value.is_value_list()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& values_list = value.as_value_list(); | 
					
						
							| 
									
										
										
										
											2021-08-06 14:34:59 +01:00
										 |  |  |  |             assign_edge_values(PropertyID::PaddingTop, PropertyID::PaddingRight, PropertyID::PaddingBottom, PropertyID::PaddingLeft, values_list.values()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::PaddingTop, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::PaddingRight, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::PaddingBottom, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::PaddingLeft, value); | 
					
						
							| 
									
										
										
										
											2019-11-18 12:25:22 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-30 12:04:15 +02:00
										 |  |  |  |     if (property_id == CSS::PropertyID::ListStyle) { | 
					
						
							| 
									
										
										
										
											2021-08-03 15:56:51 +01:00
										 |  |  |  |         if (value.is_list_style()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& list_style = value.as_list_style(); | 
					
						
							| 
									
										
										
										
											2021-08-03 15:56:51 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::ListStylePosition, list_style.position()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::ListStyleImage, list_style.image()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::ListStyleType, list_style.style_type()); | 
					
						
							| 
									
										
										
										
											2021-07-14 19:57:02 +01:00
										 |  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2020-05-30 12:04:15 +02:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::ListStylePosition, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::ListStyleImage, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::ListStyleType, value); | 
					
						
							| 
									
										
										
										
											2020-05-30 12:04:15 +02:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-13 20:41:57 +03:00
										 |  |  |  |     if (property_id == CSS::PropertyID::Font) { | 
					
						
							| 
									
										
										
										
											2021-08-03 11:37:24 +01:00
										 |  |  |  |         if (value.is_font()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& font_shorthand = value.as_font(); | 
					
						
							| 
									
										
										
										
											2021-08-03 11:37:24 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::FontSize, font_shorthand.font_size()); | 
					
						
							| 
									
										
										
										
											2021-08-14 21:05:15 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::FontFamily, font_shorthand.font_families()); | 
					
						
							| 
									
										
										
										
											2021-08-03 11:37:24 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::FontStyle, font_shorthand.font_style()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::FontWeight, font_shorthand.font_weight()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::LineHeight, font_shorthand.line_height()); | 
					
						
							|  |  |  |  |             // FIXME: Implement font-stretch and font-variant
 | 
					
						
							| 
									
										
										
										
											2021-07-21 15:39:40 +01:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FontSize, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FontFamily, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FontStyle, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FontWeight, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::LineHeight, value); | 
					
						
							|  |  |  |  |         // FIXME: Implement font-stretch and font-variant
 | 
					
						
							| 
									
										
										
										
											2021-07-21 15:39:40 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 12:05:43 +01:00
										 |  |  |  |     if (property_id == CSS::PropertyID::Flex) { | 
					
						
							| 
									
										
										
										
											2021-08-04 17:48:08 +01:00
										 |  |  |  |         if (value.is_flex()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& flex = value.as_flex(); | 
					
						
							| 
									
										
										
										
											2021-08-04 17:48:08 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::FlexGrow, flex.grow()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::FlexShrink, flex.shrink()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::FlexBasis, flex.basis()); | 
					
						
							| 
									
										
										
										
											2021-07-20 12:05:43 +01:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FlexGrow, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FlexShrink, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FlexBasis, value); | 
					
						
							| 
									
										
										
										
											2021-07-20 12:05:43 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-30 12:25:10 +02:00
										 |  |  |  |     if (property_id == CSS::PropertyID::FlexFlow) { | 
					
						
							| 
									
										
										
										
											2021-08-05 17:19:29 +01:00
										 |  |  |  |         if (value.is_flex_flow()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& flex_flow = value.as_flex_flow(); | 
					
						
							| 
									
										
										
										
											2021-08-05 17:19:29 +01:00
										 |  |  |  |             style.set_property(CSS::PropertyID::FlexDirection, flex_flow.flex_direction()); | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::FlexWrap, flex_flow.flex_wrap()); | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 15:23:04 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FlexDirection, value); | 
					
						
							|  |  |  |  |         style.set_property(CSS::PropertyID::FlexWrap, value); | 
					
						
							| 
									
										
										
										
											2021-07-14 19:57:02 +01:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-18 11:49:19 +01:00
										 |  |  |  |     style.set_property(property_id, value); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  | bool StyleComputer::expand_unresolved_values(DOM::Element& element, StringView property_name, HashMap<String, NonnullRefPtr<PropertyDependencyNode>>& dependencies, Vector<StyleComponentValueRule> const& source, Vector<StyleComponentValueRule>& dest, size_t source_start_index, HashMap<String, StyleProperty const*> const& custom_properties) const | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     // FIXME: Do this better!
 | 
					
						
							|  |  |  |  |     // We build a copy of the tree of StyleComponentValueRules, with all var()s replaced with their contents.
 | 
					
						
							|  |  |  |  |     // This is a very naive solution, and we could do better if the CSS Parser could accept tokens one at a time.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 13:10:21 +00:00
										 |  |  |  |     // Arbitrary large value chosen to avoid the billion-laughs attack.
 | 
					
						
							|  |  |  |  |     // https://www.w3.org/TR/css-variables-1/#long-variables
 | 
					
						
							|  |  |  |  |     const size_t MAX_VALUE_COUNT = 16384; | 
					
						
							|  |  |  |  |     if (source.size() + dest.size() > MAX_VALUE_COUNT) { | 
					
						
							|  |  |  |  |         dbgln("Stopped expanding CSS variables: maximum length reached."); | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |     auto get_custom_property = [&custom_properties](auto& name) -> RefPtr<StyleValue> { | 
					
						
							|  |  |  |  |         auto it = custom_properties.find(name); | 
					
						
							|  |  |  |  |         if (it != custom_properties.end()) | 
					
						
							|  |  |  |  |             return it->value->value; | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |         return nullptr; | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 20:00:31 +00:00
										 |  |  |  |     auto get_dependency_node = [&](auto name) -> NonnullRefPtr<PropertyDependencyNode> { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |         if (auto existing = dependencies.get(name); existing.has_value()) | 
					
						
							| 
									
										
										
										
											2021-12-03 20:00:31 +00:00
										 |  |  |  |             return *existing.value(); | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |         auto new_node = PropertyDependencyNode::create(name); | 
					
						
							|  |  |  |  |         dependencies.set(name, new_node); | 
					
						
							|  |  |  |  |         return new_node; | 
					
						
							| 
									
										
										
										
											2021-12-03 20:00:31 +00:00
										 |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 12:52:14 +00:00
										 |  |  |  |     for (size_t source_index = source_start_index; source_index < source.size(); source_index++) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |         auto const& value = source[source_index]; | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |         if (value.is_function()) { | 
					
						
							|  |  |  |  |             if (value.function().name().equals_ignoring_case("var"sv)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |                 auto const& var_contents = value.function().values(); | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |                 if (var_contents.is_empty()) | 
					
						
							|  |  |  |  |                     return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |                 auto const& custom_property_name_token = var_contents.first(); | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |                 if (!custom_property_name_token.is(Token::Type::Ident)) | 
					
						
							|  |  |  |  |                     return false; | 
					
						
							|  |  |  |  |                 auto custom_property_name = custom_property_name_token.token().ident(); | 
					
						
							|  |  |  |  |                 if (!custom_property_name.starts_with("--")) | 
					
						
							|  |  |  |  |                     return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 20:00:31 +00:00
										 |  |  |  |                 // Detect dependency cycles. https://www.w3.org/TR/css-variables-1/#cycles
 | 
					
						
							|  |  |  |  |                 // We do not do this by the spec, since we are not keeping a graph of var dependencies around,
 | 
					
						
							|  |  |  |  |                 // but rebuilding it every time.
 | 
					
						
							|  |  |  |  |                 if (custom_property_name == property_name) | 
					
						
							|  |  |  |  |                     return false; | 
					
						
							|  |  |  |  |                 auto parent = get_dependency_node(property_name); | 
					
						
							|  |  |  |  |                 auto child = get_dependency_node(custom_property_name); | 
					
						
							|  |  |  |  |                 parent->add_child(child); | 
					
						
							|  |  |  |  |                 if (parent->has_cycles()) | 
					
						
							|  |  |  |  |                     return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |                 if (auto custom_property_value = get_custom_property(custom_property_name)) { | 
					
						
							|  |  |  |  |                     VERIFY(custom_property_value->is_unresolved()); | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |                     if (!expand_unresolved_values(element, custom_property_name, dependencies, custom_property_value->as_unresolved().values(), dest, 0, custom_properties)) | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |                         return false; | 
					
						
							|  |  |  |  |                     continue; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 12:52:14 +00:00
										 |  |  |  |                 // Use the provided fallback value, if any.
 | 
					
						
							|  |  |  |  |                 if (var_contents.size() > 2 && var_contents[1].is(Token::Type::Comma)) { | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |                     if (!expand_unresolved_values(element, property_name, dependencies, var_contents, dest, 2, custom_properties)) | 
					
						
							| 
									
										
										
										
											2021-12-03 12:52:14 +00:00
										 |  |  |  |                         return false; | 
					
						
							|  |  |  |  |                     continue; | 
					
						
							|  |  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& source_function = value.function(); | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |             Vector<StyleComponentValueRule> function_values; | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |             if (!expand_unresolved_values(element, property_name, dependencies, source_function.values(), function_values, 0, custom_properties)) | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |                 return false; | 
					
						
							|  |  |  |  |             NonnullRefPtr<StyleFunctionRule> function = adopt_ref(*new StyleFunctionRule(source_function.name(), move(function_values))); | 
					
						
							|  |  |  |  |             dest.empend(function); | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         if (value.is_block()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |             auto const& source_block = value.block(); | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |             Vector<StyleComponentValueRule> block_values; | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |             if (!expand_unresolved_values(element, property_name, dependencies, source_block.values(), block_values, 0, custom_properties)) | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |                 return false; | 
					
						
							|  |  |  |  |             NonnullRefPtr<StyleBlockRule> block = adopt_ref(*new StyleBlockRule(source_block.token(), move(block_values))); | 
					
						
							|  |  |  |  |             dest.empend(block); | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         dest.empend(value.token()); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  | RefPtr<StyleValue> StyleComputer::resolve_unresolved_style_value(DOM::Element& element, PropertyID property_id, UnresolvedStyleValue const& unresolved, HashMap<String, StyleProperty const*> const& custom_properties) const | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     // Unresolved always contains a var(), unless it is a custom property's value, in which case we shouldn't be trying
 | 
					
						
							|  |  |  |  |     // to produce a different StyleValue from it.
 | 
					
						
							|  |  |  |  |     VERIFY(unresolved.contains_var()); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Vector<StyleComponentValueRule> expanded_values; | 
					
						
							| 
									
										
										
										
											2021-12-03 20:00:31 +00:00
										 |  |  |  |     HashMap<String, NonnullRefPtr<PropertyDependencyNode>> dependencies; | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |     if (!expand_unresolved_values(element, string_from_property_id(property_id), dependencies, unresolved.values(), expanded_values, 0, custom_properties)) | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |         return {}; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (auto parsed_value = Parser::parse_css_value({}, ParsingContext { document() }, property_id, expanded_values)) | 
					
						
							|  |  |  |  |         return parsed_value.release_nonnull(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return {}; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-12 14:53:59 +00:00
										 |  |  |  | void StyleComputer::cascade_declarations(StyleProperties& style, DOM::Element& element, Vector<MatchingRule> const& matching_rules, CascadeOrigin cascade_origin, Important important, HashMap<String, StyleProperty const*> const& custom_properties) const | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |     for (auto const& match : matching_rules) { | 
					
						
							|  |  |  |  |         for (auto const& property : verify_cast<PropertyOwningCSSStyleDeclaration>(match.rule->declaration()).properties()) { | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |             if (important != property.important) | 
					
						
							|  |  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2021-05-24 23:02:58 +02:00
										 |  |  |  |             auto property_value = property.value; | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |             if (property.value->is_unresolved()) { | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |                 if (auto resolved = resolve_unresolved_style_value(element, property.property_id, property.value->as_unresolved(), custom_properties)) | 
					
						
							| 
									
										
										
										
											2021-12-03 12:32:12 +00:00
										 |  |  |  |                     property_value = resolved.release_nonnull(); | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |             set_property_expanding_shorthands(style, property.property_id, property_value, m_document); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (cascade_origin == CascadeOrigin::Author) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |         if (auto const* inline_style = verify_cast<ElementInlineCSSStyleDeclaration>(element.inline_style())) { | 
					
						
							|  |  |  |  |             for (auto const& property : inline_style->properties()) { | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |                 if (important != property.important) | 
					
						
							| 
									
										
										
										
											2021-09-20 20:11:33 +02:00
										 |  |  |  |                     continue; | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |                 set_property_expanding_shorthands(style, property.property_id, property.value, m_document); | 
					
						
							| 
									
										
										
										
											2021-09-20 20:11:33 +02:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-06-28 21:17:34 +02:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  | static HashMap<String, StyleProperty const*> cascade_custom_properties(DOM::Element& element, Vector<MatchingRule> const& matching_rules) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     HashMap<String, StyleProperty const*> custom_properties; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (auto* parent_element = element.parent_element()) { | 
					
						
							|  |  |  |  |         for (auto const& it : parent_element->custom_properties()) | 
					
						
							|  |  |  |  |             custom_properties.set(it.key, &it.value); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (auto const& matching_rule : matching_rules) { | 
					
						
							|  |  |  |  |         for (auto const& it : verify_cast<PropertyOwningCSSStyleDeclaration>(matching_rule.rule->declaration()).custom_properties()) { | 
					
						
							|  |  |  |  |             custom_properties.set(it.key, &it.value); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     element.custom_properties().clear(); | 
					
						
							|  |  |  |  |     for (auto& it : custom_properties) | 
					
						
							|  |  |  |  |         element.add_custom_property(it.key, *it.value); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return custom_properties; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-15 19:46:52 +01:00
										 |  |  |  | // https://www.w3.org/TR/css-cascade/#cascading
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | void StyleComputer::compute_cascaded_values(StyleProperties& style, DOM::Element& element, Optional<CSS::Selector::PseudoElement> pseudo_element) const | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     // First, we collect all the CSS rules whose selectors match `element`:
 | 
					
						
							|  |  |  |  |     MatchingRuleSet matching_rule_set; | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     matching_rule_set.user_agent_rules = collect_matching_rules(element, CascadeOrigin::UserAgent, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |     sort_matching_rules(matching_rule_set.user_agent_rules); | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     matching_rule_set.author_rules = collect_matching_rules(element, CascadeOrigin::Author, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |     sort_matching_rules(matching_rule_set.author_rules); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 17:04:31 +01:00
										 |  |  |  |     // Then we resolve all the CSS custom properties ("variables") for this element:
 | 
					
						
							|  |  |  |  |     auto custom_properties = cascade_custom_properties(element, matching_rule_set.author_rules); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |     // Then we apply the declarations from the matched rules in cascade order:
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Normal user agent declarations
 | 
					
						
							| 
									
										
										
										
											2022-02-12 14:53:59 +00:00
										 |  |  |  |     cascade_declarations(style, element, matching_rule_set.user_agent_rules, CascadeOrigin::UserAgent, Important::No, custom_properties); | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: Normal user declarations
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Normal author declarations
 | 
					
						
							| 
									
										
										
										
											2022-02-12 14:53:59 +00:00
										 |  |  |  |     cascade_declarations(style, element, matching_rule_set.author_rules, CascadeOrigin::Author, Important::No, custom_properties); | 
					
						
							| 
									
										
										
										
											2019-09-30 20:25:33 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |     // Author presentational hints (NOTE: The spec doesn't say exactly how to prioritize these.)
 | 
					
						
							|  |  |  |  |     element.apply_presentational_hints(style); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: Animation declarations [css-animations-1]
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Important author declarations
 | 
					
						
							| 
									
										
										
										
											2022-02-12 14:53:59 +00:00
										 |  |  |  |     cascade_declarations(style, element, matching_rule_set.author_rules, CascadeOrigin::Author, Important::Yes, custom_properties); | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: Important user declarations
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Important user agent declarations
 | 
					
						
							| 
									
										
										
										
											2022-02-12 14:53:59 +00:00
										 |  |  |  |     cascade_declarations(style, element, matching_rule_set.user_agent_rules, CascadeOrigin::UserAgent, Important::Yes, custom_properties); | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: Transition declarations [css-transitions-1]
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | static DOM::Element const* get_parent_element(DOM::Element const* element, Optional<CSS::Selector::PseudoElement> pseudo_element) | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     // Pseudo-elements treat their originating element as their parent.
 | 
					
						
							|  |  |  |  |     DOM::Element const* parent_element = nullptr; | 
					
						
							|  |  |  |  |     if (pseudo_element.has_value()) { | 
					
						
							|  |  |  |  |         parent_element = element; | 
					
						
							|  |  |  |  |     } else if (element) { | 
					
						
							|  |  |  |  |         parent_element = element->parent_element(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return parent_element; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static NonnullRefPtr<StyleValue> get_inherit_value(CSS::PropertyID property_id, DOM::Element const* element, Optional<CSS::Selector::PseudoElement> pseudo_element) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     auto* parent_element = get_parent_element(element, pseudo_element); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!parent_element || !parent_element->specified_css_values()) | 
					
						
							| 
									
										
										
										
											2021-11-10 13:54:33 +00:00
										 |  |  |  |         return property_initial_value(property_id); | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     return parent_element->specified_css_values()->property(property_id).release_value(); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | void StyleComputer::compute_defaulted_property_value(StyleProperties& style, DOM::Element const* element, CSS::PropertyID property_id, Optional<CSS::Selector::PseudoElement> pseudo_element) const | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     // FIXME: If we don't know the correct initial value for a property, we fall back to InitialStyleValue.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-18 20:21:49 +01:00
										 |  |  |  |     auto& value_slot = style.m_property_values[to_underlying(property_id)]; | 
					
						
							|  |  |  |  |     if (!value_slot) { | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         if (is_inherited_property(property_id)) | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |             style.m_property_values[to_underlying(property_id)] = get_inherit_value(property_id, element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         else | 
					
						
							| 
									
										
										
										
											2022-02-18 20:21:49 +01:00
										 |  |  |  |             style.m_property_values[to_underlying(property_id)] = property_initial_value(property_id); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-18 20:21:49 +01:00
										 |  |  |  |     if (value_slot->is_initial()) { | 
					
						
							|  |  |  |  |         value_slot = property_initial_value(property_id); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-18 20:21:49 +01:00
										 |  |  |  |     if (value_slot->is_inherit()) { | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |         value_slot = get_inherit_value(property_id, element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-15 19:46:52 +01:00
										 |  |  |  | // https://www.w3.org/TR/css-cascade/#defaulting
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | void StyleComputer::compute_defaulted_values(StyleProperties& style, DOM::Element const* element, Optional<CSS::Selector::PseudoElement> pseudo_element) const | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |     // Walk the list of all known CSS properties and:
 | 
					
						
							|  |  |  |  |     // - Add them to `style` if they are missing.
 | 
					
						
							|  |  |  |  |     // - Resolve `inherit` and `initial` as needed.
 | 
					
						
							| 
									
										
										
										
											2021-09-21 15:51:51 +02:00
										 |  |  |  |     for (auto i = to_underlying(CSS::first_longhand_property_id); i <= to_underlying(CSS::last_longhand_property_id); ++i) { | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |         auto property_id = (CSS::PropertyID)i; | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |         compute_defaulted_property_value(style, element, property_id, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | void StyleComputer::compute_font(StyleProperties& style, DOM::Element const* element, Optional<CSS::Selector::PseudoElement> pseudo_element) const | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     // To compute the font, first ensure that we've defaulted the relevant CSS font properties.
 | 
					
						
							|  |  |  |  |     // FIXME: This should be more sophisticated.
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     compute_defaulted_property_value(style, element, CSS::PropertyID::FontFamily, pseudo_element); | 
					
						
							|  |  |  |  |     compute_defaulted_property_value(style, element, CSS::PropertyID::FontSize, pseudo_element); | 
					
						
							|  |  |  |  |     compute_defaulted_property_value(style, element, CSS::PropertyID::FontStyle, pseudo_element); | 
					
						
							|  |  |  |  |     compute_defaulted_property_value(style, element, CSS::PropertyID::FontWeight, pseudo_element); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     auto* parent_element = get_parent_element(element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     auto viewport_rect = document().browsing_context()->viewport_rect(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     auto font_size = style.property(CSS::PropertyID::FontSize).value(); | 
					
						
							| 
									
										
										
										
											2022-02-19 21:21:20 +01:00
										 |  |  |  |     auto font_style = style.property(CSS::PropertyID::FontStyle).value(); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     auto font_weight = style.property(CSS::PropertyID::FontWeight).value(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     int weight = Gfx::FontWeight::Regular; | 
					
						
							|  |  |  |  |     if (font_weight->is_identifier()) { | 
					
						
							|  |  |  |  |         switch (static_cast<IdentifierStyleValue const&>(*font_weight).id()) { | 
					
						
							|  |  |  |  |         case CSS::ValueID::Normal: | 
					
						
							|  |  |  |  |             weight = Gfx::FontWeight::Regular; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::ValueID::Bold: | 
					
						
							|  |  |  |  |             weight = Gfx::FontWeight::Bold; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::ValueID::Lighter: | 
					
						
							|  |  |  |  |             // FIXME: This should be relative to the parent.
 | 
					
						
							|  |  |  |  |             weight = Gfx::FontWeight::Regular; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::ValueID::Bolder: | 
					
						
							|  |  |  |  |             // FIXME: This should be relative to the parent.
 | 
					
						
							|  |  |  |  |             weight = Gfx::FontWeight::Bold; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-10-19 16:43:56 +01:00
										 |  |  |  |     } else if (font_weight->has_integer()) { | 
					
						
							|  |  |  |  |         int font_weight_integer = font_weight->to_integer(); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         if (font_weight_integer <= Gfx::FontWeight::Regular) | 
					
						
							|  |  |  |  |             weight = Gfx::FontWeight::Regular; | 
					
						
							|  |  |  |  |         else if (font_weight_integer <= Gfx::FontWeight::Bold) | 
					
						
							|  |  |  |  |             weight = Gfx::FontWeight::Bold; | 
					
						
							|  |  |  |  |         else | 
					
						
							|  |  |  |  |             weight = Gfx::FontWeight::Black; | 
					
						
							| 
									
										
										
										
											2022-01-27 15:54:38 +00:00
										 |  |  |  |     } else if (font_weight->is_calculated()) { | 
					
						
							|  |  |  |  |         auto maybe_weight = font_weight->as_calculated().resolve_integer(); | 
					
						
							|  |  |  |  |         if (maybe_weight.has_value()) | 
					
						
							|  |  |  |  |             weight = maybe_weight.value(); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     bool bold = weight > Gfx::FontWeight::Regular; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     int size = 10; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (font_size->is_identifier()) { | 
					
						
							|  |  |  |  |         switch (static_cast<const IdentifierStyleValue&>(*font_size).id()) { | 
					
						
							|  |  |  |  |         case CSS::ValueID::XxSmall: | 
					
						
							|  |  |  |  |         case CSS::ValueID::XSmall: | 
					
						
							|  |  |  |  |         case CSS::ValueID::Small: | 
					
						
							|  |  |  |  |         case CSS::ValueID::Medium: | 
					
						
							|  |  |  |  |             // FIXME: Should be based on "user's default font size"
 | 
					
						
							|  |  |  |  |             size = 10; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::ValueID::Large: | 
					
						
							|  |  |  |  |         case CSS::ValueID::XLarge: | 
					
						
							|  |  |  |  |         case CSS::ValueID::XxLarge: | 
					
						
							|  |  |  |  |         case CSS::ValueID::XxxLarge: | 
					
						
							|  |  |  |  |             // FIXME: Should be based on "user's default font size"
 | 
					
						
							|  |  |  |  |             size = 12; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::ValueID::Smaller: | 
					
						
							|  |  |  |  |         case CSS::ValueID::Larger: | 
					
						
							|  |  |  |  |             // FIXME: Should be based on parent element
 | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-09-23 19:51:45 +02:00
										 |  |  |  |         // FIXME: Get the root element font.
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         float root_font_size = 10; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         Gfx::FontMetrics font_metrics; | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |         if (parent_element && parent_element->specified_css_values()) | 
					
						
							|  |  |  |  |             font_metrics = parent_element->specified_css_values()->computed_font().metrics('M'); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         else | 
					
						
							|  |  |  |  |             font_metrics = Gfx::FontDatabase::default_font().metrics('M'); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         Optional<Length> maybe_length; | 
					
						
							| 
									
										
										
										
											2022-01-18 17:01:15 +00:00
										 |  |  |  |         if (font_size->is_percentage()) { | 
					
						
							|  |  |  |  |             // Percentages refer to parent element's font size
 | 
					
						
							|  |  |  |  |             auto percentage = font_size->as_percentage().percentage(); | 
					
						
							|  |  |  |  |             auto parent_font_size = size; | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |             if (parent_element && parent_element->layout_node() && parent_element->specified_css_values()) { | 
					
						
							|  |  |  |  |                 auto value = parent_element->specified_css_values()->property(CSS::PropertyID::FontSize).value(); | 
					
						
							| 
									
										
										
										
											2022-01-18 17:01:15 +00:00
										 |  |  |  |                 if (value->is_length()) { | 
					
						
							|  |  |  |  |                     auto length = static_cast<LengthStyleValue const&>(*value).to_length(); | 
					
						
							|  |  |  |  |                     if (length.is_absolute() || length.is_relative()) | 
					
						
							| 
									
										
										
										
											2022-02-21 16:24:12 +01:00
										 |  |  |  |                         parent_font_size = length.to_px(viewport_rect, font_metrics, size, root_font_size); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-01-18 17:01:15 +00:00
										 |  |  |  |             maybe_length = Length::make_px(percentage.as_fraction() * parent_font_size); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         } else if (font_size->is_length()) { | 
					
						
							|  |  |  |  |             maybe_length = font_size->to_length(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         } else if (font_size->is_calculated()) { | 
					
						
							| 
									
										
										
										
											2022-01-25 12:11:23 +00:00
										 |  |  |  |             maybe_length = Length::make_calculated(font_size->as_calculated()); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         if (maybe_length.has_value()) { | 
					
						
							| 
									
										
										
										
											2021-09-24 15:47:42 +02:00
										 |  |  |  |             // FIXME: Support font-size: calc(...)
 | 
					
						
							| 
									
										
										
										
											2022-01-27 15:54:38 +00:00
										 |  |  |  |             //        Theoretically we can do this now, but to resolve it we need a layout_node which we might not have. :^(
 | 
					
						
							| 
									
										
										
										
											2021-09-24 15:47:42 +02:00
										 |  |  |  |             if (!maybe_length->is_calculated()) { | 
					
						
							| 
									
										
										
										
											2022-02-21 16:24:12 +01:00
										 |  |  |  |                 auto px = maybe_length.value().to_px(viewport_rect, font_metrics, size, root_font_size); | 
					
						
							| 
									
										
										
										
											2021-09-24 15:47:42 +02:00
										 |  |  |  |                 if (px != 0) | 
					
						
							|  |  |  |  |                     size = px; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-19 21:21:20 +01:00
										 |  |  |  |     int slope = Gfx::name_to_slope("Normal"); | 
					
						
							|  |  |  |  |     // FIXME: Implement oblique <angle>
 | 
					
						
							|  |  |  |  |     if (font_style->is_identifier()) { | 
					
						
							|  |  |  |  |         switch (static_cast<IdentifierStyleValue const&>(*font_style).id()) { | 
					
						
							|  |  |  |  |         case CSS::ValueID::Italic: | 
					
						
							|  |  |  |  |             slope = Gfx::name_to_slope("Italic"); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::ValueID::Oblique: | 
					
						
							|  |  |  |  |             slope = Gfx::name_to_slope("Oblique"); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case CSS::ValueID::Normal: | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     // FIXME: Implement the full font-matching algorithm: https://www.w3.org/TR/css-fonts-4/#font-matching-algorithm
 | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     // Note: This is modified by the find_font() lambda
 | 
					
						
							|  |  |  |  |     FontSelector font_selector; | 
					
						
							|  |  |  |  |     bool monospace = false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     auto find_font = [&](String const& family) -> RefPtr<Gfx::Font> { | 
					
						
							| 
									
										
										
										
											2022-02-19 21:21:20 +01:00
										 |  |  |  |         font_selector = { family, size, weight, slope }; | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (auto found_font = FontCache::the().get(font_selector)) | 
					
						
							|  |  |  |  |             return found_font; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-25 13:40:26 +01:00
										 |  |  |  |         if (auto found_font = Gfx::FontDatabase::the().get(family, size, weight, slope, Gfx::Font::AllowInexactSizeMatch::Yes)) | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |             return found_font; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return {}; | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: Replace hard-coded font names with a relevant call to FontDatabase.
 | 
					
						
							|  |  |  |  |     // Currently, we cannot request the default font's name, or request it at a specific size and weight.
 | 
					
						
							|  |  |  |  |     // So, hard-coded font names it is.
 | 
					
						
							|  |  |  |  |     auto find_generic_font = [&](ValueID font_id) -> RefPtr<Gfx::Font> { | 
					
						
							|  |  |  |  |         switch (font_id) { | 
					
						
							|  |  |  |  |         case ValueID::Monospace: | 
					
						
							|  |  |  |  |         case ValueID::UiMonospace: | 
					
						
							|  |  |  |  |             monospace = true; | 
					
						
							|  |  |  |  |             return find_font("Csilla"); | 
					
						
							|  |  |  |  |         case ValueID::Serif: | 
					
						
							|  |  |  |  |         case ValueID::SansSerif: | 
					
						
							|  |  |  |  |         case ValueID::Cursive: | 
					
						
							|  |  |  |  |         case ValueID::Fantasy: | 
					
						
							|  |  |  |  |         case ValueID::UiSerif: | 
					
						
							|  |  |  |  |         case ValueID::UiSansSerif: | 
					
						
							|  |  |  |  |         case ValueID::UiRounded: | 
					
						
							|  |  |  |  |             return find_font("Katica"); | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             return {}; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     RefPtr<Gfx::Font> found_font; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     auto family_value = style.property(PropertyID::FontFamily).value(); | 
					
						
							|  |  |  |  |     if (family_value->is_value_list()) { | 
					
						
							| 
									
										
										
										
											2022-02-09 20:11:16 +01:00
										 |  |  |  |         auto const& family_list = static_cast<StyleValueList const&>(*family_value).values(); | 
					
						
							|  |  |  |  |         for (auto const& family : family_list) { | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |             if (family.is_identifier()) { | 
					
						
							|  |  |  |  |                 found_font = find_generic_font(family.to_identifier()); | 
					
						
							|  |  |  |  |             } else if (family.is_string()) { | 
					
						
							|  |  |  |  |                 found_font = find_font(family.to_string()); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (found_font) | 
					
						
							|  |  |  |  |                 break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } else if (family_value->is_identifier()) { | 
					
						
							|  |  |  |  |         found_font = find_generic_font(family_value->to_identifier()); | 
					
						
							|  |  |  |  |     } else if (family_value->is_string()) { | 
					
						
							|  |  |  |  |         found_font = find_font(family_value->to_string()); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!found_font) { | 
					
						
							|  |  |  |  |         found_font = StyleProperties::font_fallback(monospace, bold); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     FontCache::the().set(font_selector, *found_font); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-21 16:24:12 +01:00
										 |  |  |  |     style.set_property(CSS::PropertyID::FontSize, LengthStyleValue::create(CSS::Length::make_px(size))); | 
					
						
							|  |  |  |  |     style.set_property(CSS::PropertyID::FontWeight, NumericStyleValue::create_integer(weight)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     style.set_computed_font(found_font.release_nonnull()); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | void StyleComputer::absolutize_values(StyleProperties& style, DOM::Element const*, Optional<CSS::Selector::PseudoElement>) const | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto viewport_rect = document().browsing_context()->viewport_rect(); | 
					
						
							|  |  |  |  |     auto font_metrics = style.computed_font().metrics('M'); | 
					
						
							|  |  |  |  |     // FIXME: Get the root element font.
 | 
					
						
							|  |  |  |  |     float root_font_size = 10; | 
					
						
							| 
									
										
										
										
											2022-02-21 16:24:12 +01:00
										 |  |  |  |     float font_size = style.property(CSS::PropertyID::FontSize).value()->to_length().to_px(viewport_rect, font_metrics, root_font_size, root_font_size); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-26 01:35:25 +01:00
										 |  |  |  |     for (size_t i = 0; i < style.m_property_values.size(); ++i) { | 
					
						
							|  |  |  |  |         auto& value_slot = style.m_property_values[i]; | 
					
						
							| 
									
										
										
										
											2022-02-18 20:21:49 +01:00
										 |  |  |  |         if (!value_slot) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2022-02-26 01:35:25 +01:00
										 |  |  |  |         value_slot = value_slot->absolutized(viewport_rect, font_metrics, font_size, root_font_size); | 
					
						
							| 
									
										
										
										
											2019-09-30 20:25:33 +02:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-09-30 20:25:33 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  | enum class BoxTypeTransformation { | 
					
						
							|  |  |  |  |     None, | 
					
						
							|  |  |  |  |     Blockify, | 
					
						
							|  |  |  |  |     Inlinify, | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-28 11:34:15 +01:00
										 |  |  |  | static BoxTypeTransformation required_box_type_transformation(StyleProperties const& style, DOM::Element const& element, Optional<CSS::Selector::PseudoElement> const&) | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto display = style.display(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Absolute positioning or floating an element blockifies the box’s display type. [CSS2]
 | 
					
						
							|  |  |  |  |     if (style.position() == CSS::Position::Absolute || style.position() == CSS::Position::Fixed || style.float_() != CSS::Float::None) | 
					
						
							|  |  |  |  |         return BoxTypeTransformation::Blockify; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: Containment in a ruby container inlinifies the box’s display type, as described in [CSS-RUBY-1].
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-28 11:34:15 +01:00
										 |  |  |  |     // A parent with a grid or flex display value blockifies the box’s display type. [CSS-GRID-1] [CSS-FLEXBOX-1]
 | 
					
						
							|  |  |  |  |     if (element.parent() && element.parent()->layout_node()) { | 
					
						
							|  |  |  |  |         auto const& parent_display = element.parent()->layout_node()->computed_values().display(); | 
					
						
							|  |  |  |  |         if (parent_display.is_grid_inside() || parent_display.is_flex_inside()) | 
					
						
							|  |  |  |  |             return BoxTypeTransformation::Blockify; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return BoxTypeTransformation::None; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-24 14:41:48 +01:00
										 |  |  |  | // https://drafts.csswg.org/css-display/#transformations
 | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  | void StyleComputer::transform_box_type_if_needed(StyleProperties& style, DOM::Element const& element, Optional<CSS::Selector::PseudoElement> pseudo_element) const | 
					
						
							| 
									
										
										
										
											2022-01-24 14:41:48 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     // 2.7. Automatic Box Type Transformations
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Some layout effects require blockification or inlinification of the box type,
 | 
					
						
							|  |  |  |  |     // which sets the box’s computed outer display type to block or inline (respectively).
 | 
					
						
							|  |  |  |  |     // (This has no effect on display types that generate no box at all, such as none or contents.)
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: If a block box (block flow) is inlinified, its inner display type is set to flow-root so that it remains a block container.
 | 
					
						
							|  |  |  |  |     //
 | 
					
						
							|  |  |  |  |     // FIXME: If an inline box (inline flow) is inlinified, it recursively inlinifies all of its in-flow children,
 | 
					
						
							|  |  |  |  |     //        so that no block-level descendants break up the inline formatting context in which it participates.
 | 
					
						
							|  |  |  |  |     //
 | 
					
						
							|  |  |  |  |     // FIXME: For legacy reasons, if an inline block box (inline flow-root) is blockified, it becomes a block box (losing its flow-root nature).
 | 
					
						
							|  |  |  |  |     //        For consistency, a run-in flow-root box also blockifies to a block box.
 | 
					
						
							|  |  |  |  |     //
 | 
					
						
							|  |  |  |  |     // FIXME: If a layout-internal box is blockified, its inner display type converts to flow so that it becomes a block container.
 | 
					
						
							|  |  |  |  |     //        Inlinification has no effect on layout-internal boxes. (However, placement in such an inline context will typically cause them
 | 
					
						
							|  |  |  |  |     //        to be wrapped in an appropriately-typed anonymous inline-level box.)
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     auto display = style.display(); | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  |     if (display.is_none() || display.is_contents()) | 
					
						
							|  |  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2022-01-24 14:41:48 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  |     switch (required_box_type_transformation(style, element, pseudo_element)) { | 
					
						
							|  |  |  |  |     case BoxTypeTransformation::None: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     case BoxTypeTransformation::Blockify: | 
					
						
							| 
									
										
										
										
											2022-02-28 11:35:37 +01:00
										 |  |  |  |         if (!display.is_block_outside()) | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::Display, IdentifierStyleValue::create(CSS::ValueID::Block)); | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  |     case BoxTypeTransformation::Inlinify: | 
					
						
							| 
									
										
										
										
											2022-02-28 11:35:37 +01:00
										 |  |  |  |         if (!display.is_inline_outside()) | 
					
						
							|  |  |  |  |             style.set_property(CSS::PropertyID::Display, IdentifierStyleValue::create(CSS::ValueID::Inline)); | 
					
						
							| 
									
										
										
										
											2022-02-28 11:27:57 +01:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-24 14:41:48 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 13:49:57 +02:00
										 |  |  |  | NonnullRefPtr<StyleProperties> StyleComputer::create_document_style() const | 
					
						
							| 
									
										
										
										
											2021-09-23 19:48:41 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto style = StyleProperties::create(); | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     compute_font(style, nullptr, {}); | 
					
						
							|  |  |  |  |     compute_defaulted_values(style, nullptr, {}); | 
					
						
							|  |  |  |  |     absolutize_values(style, nullptr, {}); | 
					
						
							| 
									
										
										
										
											2022-02-19 17:34:05 +01:00
										 |  |  |  |     if (auto* browsing_context = m_document.browsing_context()) { | 
					
						
							|  |  |  |  |         auto viewport_rect = browsing_context->viewport_rect(); | 
					
						
							|  |  |  |  |         style->set_property(CSS::PropertyID::Width, CSS::LengthStyleValue::create(CSS::Length::make_px(viewport_rect.width()))); | 
					
						
							|  |  |  |  |         style->set_property(CSS::PropertyID::Height, CSS::LengthStyleValue::create(CSS::Length::make_px(viewport_rect.height()))); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-23 19:48:41 +02:00
										 |  |  |  |     return style; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  | NonnullRefPtr<StyleProperties> StyleComputer::compute_style(DOM::Element& element, Optional<CSS::Selector::PseudoElement> pseudo_element) const | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |     build_rule_cache_if_needed(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 11:38:18 +02:00
										 |  |  |  |     auto style = StyleProperties::create(); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  |     // 1. Perform the cascade. This produces the "specified style"
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     compute_cascaded_values(style, element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 2. Compute the font, since that may be needed for font-relative CSS units
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     compute_font(style, &element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 3. Absolutize values, turning font/viewport relative lengths into absolute lengths
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     absolutize_values(style, &element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 4. Default the values, applying inheritance and 'initial' as needed
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     compute_defaulted_values(style, &element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2021-09-23 13:13:51 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-24 14:41:48 +01:00
										 |  |  |  |     // 5. Run automatic box type transformations
 | 
					
						
							| 
									
										
										
										
											2022-02-24 15:54:12 +00:00
										 |  |  |  |     transform_box_type_if_needed(style, element, pseudo_element); | 
					
						
							| 
									
										
										
										
											2022-01-24 14:41:48 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-07 09:23:53 +02:00
										 |  |  |  |     return style; | 
					
						
							| 
									
										
										
										
											2019-06-27 17:47:59 +02:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-03 20:00:31 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | PropertyDependencyNode::PropertyDependencyNode(String name) | 
					
						
							|  |  |  |  |     : m_name(move(name)) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void PropertyDependencyNode::add_child(NonnullRefPtr<PropertyDependencyNode> new_child) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     for (auto const& child : m_children) { | 
					
						
							|  |  |  |  |         if (child.m_name == new_child->m_name) | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // We detect self-reference already.
 | 
					
						
							|  |  |  |  |     VERIFY(new_child->m_name != m_name); | 
					
						
							|  |  |  |  |     m_children.append(move(new_child)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool PropertyDependencyNode::has_cycles() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (m_marked) | 
					
						
							|  |  |  |  |         return true; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TemporaryChange change { m_marked, true }; | 
					
						
							|  |  |  |  |     for (auto& child : m_children) { | 
					
						
							|  |  |  |  |         if (child.has_cycles()) | 
					
						
							|  |  |  |  |             return true; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return false; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | void StyleComputer::build_rule_cache_if_needed() const | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (m_rule_cache && m_rule_cache->generation == m_document.style_sheets().generation()) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     const_cast<StyleComputer&>(*this).build_rule_cache(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void StyleComputer::build_rule_cache() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     // FIXME: Make a rule cache for UA style as well.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     m_rule_cache = make<RuleCache>(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     size_t num_class_rules = 0; | 
					
						
							| 
									
										
										
										
											2022-02-10 18:50:58 +01:00
										 |  |  |  |     size_t num_id_rules = 0; | 
					
						
							| 
									
										
										
										
											2022-02-10 18:59:19 +01:00
										 |  |  |  |     size_t num_tag_name_rules = 0; | 
					
						
							| 
									
										
										
										
											2022-02-25 17:50:06 +00:00
										 |  |  |  |     size_t num_pseudo_element_rules = 0; | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     Vector<MatchingRule> matching_rules; | 
					
						
							|  |  |  |  |     size_t style_sheet_index = 0; | 
					
						
							|  |  |  |  |     for_each_stylesheet(CascadeOrigin::Author, [&](auto& sheet) { | 
					
						
							|  |  |  |  |         size_t rule_index = 0; | 
					
						
							|  |  |  |  |         static_cast<CSSStyleSheet const&>(sheet).for_each_effective_style_rule([&](auto const& rule) { | 
					
						
							|  |  |  |  |             size_t selector_index = 0; | 
					
						
							|  |  |  |  |             for (CSS::Selector const& selector : rule.selectors()) { | 
					
						
							|  |  |  |  |                 MatchingRule matching_rule { rule, style_sheet_index, rule_index, selector_index, selector.specificity() }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 bool added_to_bucket = false; | 
					
						
							|  |  |  |  |                 for (auto const& simple_selector : selector.compound_selectors().last().simple_selectors) { | 
					
						
							| 
									
										
										
										
											2022-02-25 17:50:06 +00:00
										 |  |  |  |                     if (simple_selector.type == CSS::Selector::SimpleSelector::Type::PseudoElement) { | 
					
						
							|  |  |  |  |                         m_rule_cache->rules_by_pseudo_element.ensure(simple_selector.pseudo_element).append(move(matching_rule)); | 
					
						
							|  |  |  |  |                         ++num_pseudo_element_rules; | 
					
						
							| 
									
										
										
										
											2022-02-10 18:50:58 +01:00
										 |  |  |  |                         added_to_bucket = true; | 
					
						
							|  |  |  |  |                         break; | 
					
						
							|  |  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2022-02-25 17:50:06 +00:00
										 |  |  |  |                 } | 
					
						
							|  |  |  |  |                 if (!added_to_bucket) { | 
					
						
							|  |  |  |  |                     for (auto const& simple_selector : selector.compound_selectors().last().simple_selectors) { | 
					
						
							|  |  |  |  |                         if (simple_selector.type == CSS::Selector::SimpleSelector::Type::Id) { | 
					
						
							|  |  |  |  |                             m_rule_cache->rules_by_id.ensure(simple_selector.value).append(move(matching_rule)); | 
					
						
							|  |  |  |  |                             ++num_id_rules; | 
					
						
							|  |  |  |  |                             added_to_bucket = true; | 
					
						
							|  |  |  |  |                             break; | 
					
						
							|  |  |  |  |                         } | 
					
						
							|  |  |  |  |                         if (simple_selector.type == CSS::Selector::SimpleSelector::Type::Class) { | 
					
						
							|  |  |  |  |                             m_rule_cache->rules_by_class.ensure(simple_selector.value).append(move(matching_rule)); | 
					
						
							|  |  |  |  |                             ++num_class_rules; | 
					
						
							|  |  |  |  |                             added_to_bucket = true; | 
					
						
							|  |  |  |  |                             break; | 
					
						
							|  |  |  |  |                         } | 
					
						
							|  |  |  |  |                         if (simple_selector.type == CSS::Selector::SimpleSelector::Type::TagName) { | 
					
						
							|  |  |  |  |                             m_rule_cache->rules_by_tag_name.ensure(simple_selector.value).append(move(matching_rule)); | 
					
						
							|  |  |  |  |                             ++num_tag_name_rules; | 
					
						
							|  |  |  |  |                             added_to_bucket = true; | 
					
						
							|  |  |  |  |                             break; | 
					
						
							|  |  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2022-02-10 18:59:19 +01:00
										 |  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |                 } | 
					
						
							|  |  |  |  |                 if (!added_to_bucket) | 
					
						
							|  |  |  |  |                     m_rule_cache->other_rules.append(move(matching_rule)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 ++selector_index; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             ++rule_index; | 
					
						
							|  |  |  |  |         }); | 
					
						
							|  |  |  |  |         ++style_sheet_index; | 
					
						
							|  |  |  |  |     }); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if constexpr (LIBWEB_CSS_DEBUG) { | 
					
						
							|  |  |  |  |         dbgln("Built rule cache!"); | 
					
						
							| 
									
										
										
										
											2022-02-25 17:50:06 +00:00
										 |  |  |  |         dbgln("           ID: {}", num_id_rules); | 
					
						
							|  |  |  |  |         dbgln("        Class: {}", num_class_rules); | 
					
						
							|  |  |  |  |         dbgln("      TagName: {}", num_tag_name_rules); | 
					
						
							|  |  |  |  |         dbgln("PseudoElement: {}", num_pseudo_element_rules); | 
					
						
							|  |  |  |  |         dbgln("        Other: {}", m_rule_cache->other_rules.size()); | 
					
						
							|  |  |  |  |         dbgln("        Total: {}", num_class_rules + num_id_rules + num_tag_name_rules + m_rule_cache->other_rules.size()); | 
					
						
							| 
									
										
										
										
											2022-02-10 17:49:50 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     m_rule_cache->generation = m_document.style_sheets().generation(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void StyleComputer::invalidate_rule_cache() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     m_rule_cache = nullptr; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  |  | } |