mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 01:19:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			4752 lines
		
	
	
	
		
			199 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			4752 lines
		
	
	
	
		
			199 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2025, Andreas Kling <andreas@ladybird.org>
 | |
|  * Copyright (c) 2020-2021, the SerenityOS developers.
 | |
|  * Copyright (c) 2021-2025, Sam Atkins <sam@ladybird.org>
 | |
|  * Copyright (c) 2021, Tobias Christiansen <tobyase@serenityos.org>
 | |
|  * Copyright (c) 2022, MacDue <macdue@dueutil.tech>
 | |
|  * Copyright (c) 2024, Shannon Booth <shannon@serenityos.org>
 | |
|  * Copyright (c) 2024, Tommy van der Vorst <tommy@pixelspark.nl>
 | |
|  * Copyright (c) 2024, Matthew Olsson <mattco@serenityos.org>
 | |
|  * Copyright (c) 2024, Glenn Skrzypczak <glenn.skrzypczak@gmail.com>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/Debug.h>
 | |
| #include <AK/QuickSort.h>
 | |
| #include <LibWeb/CSS/CSSStyleValue.h>
 | |
| #include <LibWeb/CSS/CharacterTypes.h>
 | |
| #include <LibWeb/CSS/Parser/Parser.h>
 | |
| #include <LibWeb/CSS/StyleValues/AngleStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/BackgroundRepeatStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/BackgroundSizeStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/BorderRadiusStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/CSSColorValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/CSSKeywordValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/ColorSchemeStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/ContentStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/CounterDefinitionsStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/CursorStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/CustomIdentStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/DisplayStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/EasingStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/EdgeStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/FilterValueListStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/FitContentStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/FlexStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/FontStyleStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/FrequencyStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/GridAutoFlowStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/GridTemplateAreaStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/GridTrackPlacementStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/GridTrackSizeListStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/IntegerStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/LengthStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/MathDepthStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/NumberStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/OpenTypeTaggedStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/PercentageStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/PositionStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/ResolutionStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/ScrollbarGutterStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/ShadowStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/ShorthandStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/StringStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/StyleValueList.h>
 | |
| #include <LibWeb/CSS/StyleValues/TimeStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/TransformationStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/TransitionStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/URLStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/UnresolvedStyleValue.h>
 | |
| #include <LibWeb/Dump.h>
 | |
| #include <LibWeb/Infra/Strings.h>
 | |
| 
 | |
| namespace Web::CSS::Parser {
 | |
| 
 | |
| static void remove_property(Vector<PropertyID>& properties, PropertyID property_to_remove)
 | |
| {
 | |
|     properties.remove_first_matching([&](auto it) { return it == property_to_remove; });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_all_as_single_keyword_value(TokenStream<ComponentValue>& tokens, Keyword keyword)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     tokens.discard_whitespace();
 | |
|     auto keyword_value = parse_keyword_value(tokens);
 | |
|     tokens.discard_whitespace();
 | |
| 
 | |
|     if (tokens.has_next_token() || !keyword_value || keyword_value->to_keyword() != keyword)
 | |
|         return {};
 | |
| 
 | |
|     transaction.commit();
 | |
|     return keyword_value;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_simple_comma_separated_value_list(PropertyID property_id, TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     return parse_comma_separated_value_list(tokens, [this, property_id](auto& tokens) -> RefPtr<CSSStyleValue const> {
 | |
|         if (auto value = parse_css_value_for_property(property_id, tokens))
 | |
|             return value;
 | |
|         tokens.reconsume_current_input_token();
 | |
|         return nullptr;
 | |
|     });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_css_value_for_property(PropertyID property_id, TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     return parse_css_value_for_properties({ &property_id, 1 }, tokens)
 | |
|         .map([](auto& it) { return it.style_value; })
 | |
|         .value_or(nullptr);
 | |
| }
 | |
| 
 | |
| Optional<Parser::PropertyAndValue> Parser::parse_css_value_for_properties(ReadonlySpan<PropertyID> property_ids, TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto any_property_accepts_type = [](ReadonlySpan<PropertyID> property_ids, ValueType value_type) -> Optional<PropertyID> {
 | |
|         for (auto const& property : property_ids) {
 | |
|             if (property_accepts_type(property, value_type))
 | |
|                 return property;
 | |
|         }
 | |
|         return {};
 | |
|     };
 | |
|     auto any_property_accepts_keyword = [](ReadonlySpan<PropertyID> property_ids, Keyword keyword) -> Optional<PropertyID> {
 | |
|         for (auto const& property : property_ids) {
 | |
|             if (property_accepts_keyword(property, keyword))
 | |
|                 return property;
 | |
|         }
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     auto& peek_token = tokens.next_token();
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::EasingFunction); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_easing_function = parse_easing_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_easing_function };
 | |
|     }
 | |
| 
 | |
|     if (peek_token.is(Token::Type::Ident)) {
 | |
|         // NOTE: We do not try to parse "CSS-wide keywords" here. https://www.w3.org/TR/css-values-4/#common-keywords
 | |
|         //       These are only valid on their own, and so should be parsed directly in `parse_css_value()`.
 | |
|         auto keyword = keyword_from_string(peek_token.token().ident());
 | |
|         if (keyword.has_value()) {
 | |
|             if (auto property = any_property_accepts_keyword(property_ids, keyword.value()); property.has_value()) {
 | |
|                 tokens.discard_a_token();
 | |
|                 return PropertyAndValue { *property, CSSKeywordValue::create(keyword.value()) };
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Custom idents
 | |
|         if (auto property = any_property_accepts_type(property_ids, ValueType::CustomIdent); property.has_value()) {
 | |
|             auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|             if (auto custom_ident = parse_custom_ident_value(tokens, property_custom_ident_blacklist(*property)))
 | |
|                 return PropertyAndValue { *property, custom_ident };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Color); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_color = parse_color_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_color };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Counter); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_counter = parse_counter_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_counter };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Image); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_image = parse_image_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_image };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Position); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_position = parse_position_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_position };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::BackgroundPosition); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_position = parse_position_value(tokens, PositionParsingMode::BackgroundPosition))
 | |
|             return PropertyAndValue { *property, maybe_position };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::BasicShape); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_basic_shape = parse_basic_shape_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_basic_shape };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Ratio); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_ratio = parse_ratio_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_ratio };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::OpenTypeTag); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_rect = parse_opentype_tag_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_rect };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Rect); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto maybe_rect = parse_rect_value(tokens))
 | |
|             return PropertyAndValue { *property, maybe_rect };
 | |
|     }
 | |
| 
 | |
|     if (peek_token.is(Token::Type::String)) {
 | |
|         if (auto property = any_property_accepts_type(property_ids, ValueType::String); property.has_value()) {
 | |
|             auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|             return PropertyAndValue { *property, StringStyleValue::create(tokens.consume_a_token().token().string()) };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Url); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto url = parse_url_value(tokens))
 | |
|             return PropertyAndValue { *property, url };
 | |
|     }
 | |
| 
 | |
|     // <integer>/<number> come before <length>, so that 0 is not interpreted as a <length> in case both are allowed.
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Integer); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto value = parse_integer_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_integer() && property_accepts_integer(*property, value->as_integer().integer()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Number); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto value = parse_number_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_number() && property_accepts_number(*property, value->as_number().number()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Angle); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (property_accepts_type(*property, ValueType::Percentage)) {
 | |
|             if (auto value = parse_angle_percentage_value(tokens)) {
 | |
|                 if (value->is_calculated())
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_angle() && property_accepts_angle(*property, value->as_angle().angle()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_percentage() && property_accepts_percentage(*property, value->as_percentage().percentage()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|             }
 | |
|         }
 | |
|         if (auto value = parse_angle_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_angle() && property_accepts_angle(*property, value->as_angle().angle()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Flex); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto value = parse_flex_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_flex() && property_accepts_flex(*property, value->as_flex().flex()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Frequency); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (property_accepts_type(*property, ValueType::Percentage)) {
 | |
|             if (auto value = parse_frequency_percentage_value(tokens)) {
 | |
|                 if (value->is_calculated())
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_frequency() && property_accepts_frequency(*property, value->as_frequency().frequency()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_percentage() && property_accepts_percentage(*property, value->as_percentage().percentage()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|             }
 | |
|         }
 | |
|         if (auto value = parse_frequency_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_frequency() && property_accepts_frequency(*property, value->as_frequency().frequency()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::FitContent); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto value = parse_fit_content_value(tokens))
 | |
|             return PropertyAndValue { *property, value };
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Length); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (property_accepts_type(*property, ValueType::Percentage)) {
 | |
|             if (auto value = parse_length_percentage_value(tokens)) {
 | |
|                 if (value->is_calculated())
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_length() && property_accepts_length(*property, value->as_length().length()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_percentage() && property_accepts_percentage(*property, value->as_percentage().percentage()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|             }
 | |
|         }
 | |
|         if (auto value = parse_length_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_length() && property_accepts_length(*property, value->as_length().length()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Resolution); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto value = parse_resolution_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_resolution() && property_accepts_resolution(*property, value->as_resolution().resolution()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Time); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (property_accepts_type(*property, ValueType::Percentage)) {
 | |
|             if (auto value = parse_time_percentage_value(tokens)) {
 | |
|                 if (value->is_calculated())
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_time() && property_accepts_time(*property, value->as_time().time()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|                 if (value->is_percentage() && property_accepts_percentage(*property, value->as_percentage().percentage()))
 | |
|                     return PropertyAndValue { *property, value };
 | |
|             }
 | |
|         }
 | |
|         if (auto value = parse_time_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_time() && property_accepts_time(*property, value->as_time().time()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // <percentage> is checked after the <foo-percentage> types.
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Percentage); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto value = parse_percentage_value(tokens)) {
 | |
|             if (value->is_calculated())
 | |
|                 return PropertyAndValue { *property, value };
 | |
|             if (value->is_percentage() && property_accepts_percentage(*property, value->as_percentage().percentage()))
 | |
|                 return PropertyAndValue { *property, value };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (auto property = any_property_accepts_type(property_ids, ValueType::Paint); property.has_value()) {
 | |
|         auto context_guard = push_temporary_value_parsing_context(*property);
 | |
|         if (auto value = parse_paint_value(tokens))
 | |
|             return PropertyAndValue { *property, value.release_nonnull() };
 | |
|     }
 | |
| 
 | |
|     return OptionalNone {};
 | |
| }
 | |
| 
 | |
| Parser::ParseErrorOr<NonnullRefPtr<CSSStyleValue const>> Parser::parse_css_value(PropertyID property_id, TokenStream<ComponentValue>& unprocessed_tokens, Optional<String> original_source_text)
 | |
| {
 | |
|     auto context_guard = push_temporary_value_parsing_context(property_id);
 | |
| 
 | |
|     // FIXME: Stop removing whitespace here. It's less helpful than it seems.
 | |
|     Vector<ComponentValue> component_values;
 | |
|     bool contains_arbitrary_substitution_function = false;
 | |
|     bool const property_accepts_custom_ident = property_accepts_type(property_id, ValueType::CustomIdent);
 | |
| 
 | |
|     while (unprocessed_tokens.has_next_token()) {
 | |
|         auto const& token = unprocessed_tokens.consume_a_token();
 | |
| 
 | |
|         if (token.is(Token::Type::Semicolon)) {
 | |
|             unprocessed_tokens.reconsume_current_input_token();
 | |
|             return ParseError::SyntaxError;
 | |
|         }
 | |
| 
 | |
|         if (property_id != PropertyID::Custom) {
 | |
|             if (token.is(Token::Type::Whitespace))
 | |
|                 continue;
 | |
| 
 | |
|             if (!property_accepts_custom_ident && token.is(Token::Type::Ident) && has_ignored_vendor_prefix(token.token().ident()))
 | |
|                 return ParseError::IncludesIgnoredVendorPrefix;
 | |
|         }
 | |
| 
 | |
|         if (!contains_arbitrary_substitution_function) {
 | |
|             if (token.is_function() && token.function().contains_arbitrary_substitution_function())
 | |
|                 contains_arbitrary_substitution_function = true;
 | |
|             else if (token.is_block() && token.block().contains_arbitrary_substitution_function())
 | |
|                 contains_arbitrary_substitution_function = true;
 | |
|         }
 | |
| 
 | |
|         component_values.append(token);
 | |
|     }
 | |
| 
 | |
|     if (property_id == PropertyID::Custom || contains_arbitrary_substitution_function)
 | |
|         return UnresolvedStyleValue::create(move(component_values), contains_arbitrary_substitution_function, original_source_text);
 | |
| 
 | |
|     if (component_values.is_empty())
 | |
|         return ParseError::SyntaxError;
 | |
| 
 | |
|     auto tokens = TokenStream { component_values };
 | |
| 
 | |
|     if (component_values.size() == 1) {
 | |
|         if (auto parsed_value = parse_builtin_value(tokens))
 | |
|             return parsed_value.release_nonnull();
 | |
|     }
 | |
| 
 | |
|     // Special-case property handling
 | |
|     switch (property_id) {
 | |
|     case PropertyID::AspectRatio:
 | |
|         if (auto parsed_value = parse_aspect_ratio_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BackdropFilter:
 | |
|     case PropertyID::Filter:
 | |
|         if (auto parsed_value = parse_filter_value_list_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Background:
 | |
|         if (auto parsed_value = parse_background_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BackgroundAttachment:
 | |
|     case PropertyID::BackgroundBlendMode:
 | |
|     case PropertyID::BackgroundClip:
 | |
|     case PropertyID::BackgroundImage:
 | |
|     case PropertyID::BackgroundOrigin:
 | |
|         if (auto parsed_value = parse_simple_comma_separated_value_list(property_id, tokens))
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BackgroundPosition:
 | |
|         if (auto parsed_value = parse_comma_separated_value_list(tokens, [this](auto& tokens) { return parse_position_value(tokens, PositionParsingMode::BackgroundPosition); }))
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BackgroundPositionX:
 | |
|     case PropertyID::BackgroundPositionY:
 | |
|         if (auto parsed_value = parse_comma_separated_value_list(tokens, [this, property_id](auto& tokens) { return parse_single_background_position_x_or_y_value(tokens, property_id); }))
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BackgroundRepeat:
 | |
|         if (auto parsed_value = parse_comma_separated_value_list(tokens, [this](auto& tokens) { return parse_single_background_repeat_value(tokens); }))
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BackgroundSize:
 | |
|         if (auto parsed_value = parse_comma_separated_value_list(tokens, [this](auto& tokens) { return parse_single_background_size_value(tokens); }))
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Border:
 | |
|     case PropertyID::BorderBottom:
 | |
|     case PropertyID::BorderLeft:
 | |
|     case PropertyID::BorderRight:
 | |
|     case PropertyID::BorderTop:
 | |
|         if (auto parsed_value = parse_border_value(property_id, tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BorderTopLeftRadius:
 | |
|     case PropertyID::BorderTopRightRadius:
 | |
|     case PropertyID::BorderBottomRightRadius:
 | |
|     case PropertyID::BorderBottomLeftRadius:
 | |
|         if (auto parsed_value = parse_border_radius_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BorderRadius:
 | |
|         if (auto parsed_value = parse_border_radius_shorthand_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::BoxShadow:
 | |
|         if (auto parsed_value = parse_shadow_value(tokens, AllowInsetKeyword::Yes); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::ColorScheme:
 | |
|         if (auto parsed_value = parse_color_scheme_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Columns:
 | |
|         if (auto parsed_value = parse_columns_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Content:
 | |
|         if (auto parsed_value = parse_content_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::CounterIncrement:
 | |
|         if (auto parsed_value = parse_counter_increment_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::CounterReset:
 | |
|         if (auto parsed_value = parse_counter_reset_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::CounterSet:
 | |
|         if (auto parsed_value = parse_counter_set_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Cursor:
 | |
|         if (auto parsed_value = parse_cursor_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Display:
 | |
|         if (auto parsed_value = parse_display_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Flex:
 | |
|         if (auto parsed_value = parse_flex_shorthand_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FlexFlow:
 | |
|         if (auto parsed_value = parse_flex_flow_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Font:
 | |
|         if (auto parsed_value = parse_font_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontFamily:
 | |
|         if (auto parsed_value = parse_font_family_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontFeatureSettings:
 | |
|         if (auto parsed_value = parse_font_feature_settings_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontLanguageOverride:
 | |
|         if (auto parsed_value = parse_font_language_override_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontStyle:
 | |
|         if (auto parsed_value = parse_font_style_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontVariationSettings:
 | |
|         if (auto parsed_value = parse_font_variation_settings_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontVariant:
 | |
|         if (auto parsed_value = parse_font_variant(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontVariantAlternates:
 | |
|         if (auto parsed_value = parse_font_variant_alternates_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontVariantEastAsian:
 | |
|         if (auto parsed_value = parse_font_variant_east_asian_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontVariantLigatures:
 | |
|         if (auto parsed_value = parse_font_variant_ligatures_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::FontVariantNumeric:
 | |
|         if (auto parsed_value = parse_font_variant_numeric_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridArea:
 | |
|         if (auto parsed_value = parse_grid_area_shorthand_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridAutoFlow:
 | |
|         if (auto parsed_value = parse_grid_auto_flow_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridColumn:
 | |
|         if (auto parsed_value = parse_grid_track_placement_shorthand_value(property_id, tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridColumnEnd:
 | |
|         if (auto parsed_value = parse_grid_track_placement(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridColumnStart:
 | |
|         if (auto parsed_value = parse_grid_track_placement(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridRow:
 | |
|         if (auto parsed_value = parse_grid_track_placement_shorthand_value(property_id, tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridRowEnd:
 | |
|         if (auto parsed_value = parse_grid_track_placement(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridRowStart:
 | |
|         if (auto parsed_value = parse_grid_track_placement(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Grid:
 | |
|         if (auto parsed_value = parse_grid_shorthand_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridTemplate:
 | |
|         if (auto parsed_value = parse_grid_track_size_list_shorthand_value(property_id, tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridTemplateAreas:
 | |
|         if (auto parsed_value = parse_grid_template_areas_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridTemplateColumns:
 | |
|         if (auto parsed_value = parse_grid_track_size_list(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridTemplateRows:
 | |
|         if (auto parsed_value = parse_grid_track_size_list(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridAutoColumns:
 | |
|         if (auto parsed_value = parse_grid_auto_track_sizes(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::GridAutoRows:
 | |
|         if (auto parsed_value = parse_grid_auto_track_sizes(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::ListStyle:
 | |
|         if (auto parsed_value = parse_list_style_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::MathDepth:
 | |
|         if (auto parsed_value = parse_math_depth_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Overflow:
 | |
|         if (auto parsed_value = parse_overflow_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::PlaceContent:
 | |
|         if (auto parsed_value = parse_place_content_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::PlaceItems:
 | |
|         if (auto parsed_value = parse_place_items_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::PlaceSelf:
 | |
|         if (auto parsed_value = parse_place_self_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Quotes:
 | |
|         if (auto parsed_value = parse_quotes_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Rotate:
 | |
|         if (auto parsed_value = parse_rotate_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::ScrollbarGutter:
 | |
|         if (auto parsed_value = parse_scrollbar_gutter_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::StrokeDasharray:
 | |
|         if (auto parsed_value = parse_stroke_dasharray_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TextDecoration:
 | |
|         if (auto parsed_value = parse_text_decoration_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TextDecorationLine:
 | |
|         if (auto parsed_value = parse_text_decoration_line_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TextShadow:
 | |
|         if (auto parsed_value = parse_shadow_value(tokens, AllowInsetKeyword::No); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TouchAction:
 | |
|         if (auto parsed_value = parse_touch_action_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Transform:
 | |
|         if (auto parsed_value = parse_transform_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TransformOrigin:
 | |
|         if (auto parsed_value = parse_transform_origin_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Transition:
 | |
|         if (auto parsed_value = parse_transition_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TransitionDelay:
 | |
|         if (auto parsed_value = parse_list_of_time_values(property_id, tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TransitionDuration:
 | |
|         if (auto parsed_value = parse_list_of_time_values(property_id, tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TransitionProperty:
 | |
|         if (auto parsed_value = parse_transition_property_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::TransitionTimingFunction:
 | |
|         if (auto parsed_value = parse_comma_separated_value_list(tokens, [this](auto& tokens) { return parse_easing_value(tokens); }))
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Translate:
 | |
|         if (auto parsed_value = parse_translate_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Scale:
 | |
|         if (auto parsed_value = parse_scale_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     case PropertyID::Contain:
 | |
|         if (auto parsed_value = parse_contain_value(tokens); parsed_value && !tokens.has_next_token())
 | |
|             return parsed_value.release_nonnull();
 | |
|         return ParseError::SyntaxError;
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     // If there's only 1 ComponentValue, we can only produce a single CSSStyleValue.
 | |
|     if (component_values.size() == 1) {
 | |
|         auto stream = TokenStream { component_values };
 | |
|         if (auto parsed_value = parse_css_value_for_property(property_id, stream))
 | |
|             return parsed_value.release_nonnull();
 | |
|     } else {
 | |
|         StyleValueVector parsed_values;
 | |
|         auto stream = TokenStream { component_values };
 | |
|         while (auto parsed_value = parse_css_value_for_property(property_id, stream)) {
 | |
|             parsed_values.append(parsed_value.release_nonnull());
 | |
|             if (!stream.has_next_token())
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         if (!stream.has_next_token()) {
 | |
|             // Some types (such as <ratio>) can be made from multiple ComponentValues, so if we only made 1 CSSStyleValue, return it directly.
 | |
|             if (parsed_values.size() == 1)
 | |
|                 return *parsed_values.take_first();
 | |
| 
 | |
|             if (!parsed_values.is_empty() && parsed_values.size() <= property_maximum_value_count(property_id))
 | |
|                 return StyleValueList::create(move(parsed_values), StyleValueList::Separator::Space);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // We have multiple values, but the property claims to accept only a single one, check if it's a shorthand property.
 | |
|     auto unassigned_properties = longhands_for_shorthand(property_id);
 | |
|     if (unassigned_properties.is_empty())
 | |
|         return ParseError::SyntaxError;
 | |
| 
 | |
|     auto stream = TokenStream { component_values };
 | |
| 
 | |
|     HashMap<UnderlyingType<PropertyID>, Vector<ValueComparingNonnullRefPtr<CSSStyleValue const>>> assigned_values;
 | |
| 
 | |
|     while (stream.has_next_token() && !unassigned_properties.is_empty()) {
 | |
|         auto property_and_value = parse_css_value_for_properties(unassigned_properties, stream);
 | |
|         if (property_and_value.has_value()) {
 | |
|             auto property = property_and_value->property;
 | |
|             auto value = property_and_value->style_value;
 | |
|             auto& values = assigned_values.ensure(to_underlying(property));
 | |
|             if (values.size() + 1 == property_maximum_value_count(property)) {
 | |
|                 // We're done with this property, move on to the next one.
 | |
|                 unassigned_properties.remove_first_matching([&](auto& unassigned_property) { return unassigned_property == property; });
 | |
|             }
 | |
| 
 | |
|             values.append(value.release_nonnull());
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         // No property matched, so we're done.
 | |
|         if constexpr (CSS_PARSER_DEBUG) {
 | |
|             dbgln("No property (from {} properties) matched {}", unassigned_properties.size(), stream.next_token().to_debug_string());
 | |
|             for (auto id : unassigned_properties)
 | |
|                 dbgln("    {}", string_from_property_id(id));
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     for (auto& property : unassigned_properties)
 | |
|         assigned_values.ensure(to_underlying(property)).append(property_initial_value(property));
 | |
| 
 | |
|     stream.discard_whitespace();
 | |
|     if (stream.has_next_token())
 | |
|         return ParseError::SyntaxError;
 | |
| 
 | |
|     Vector<PropertyID> longhand_properties;
 | |
|     longhand_properties.ensure_capacity(assigned_values.size());
 | |
|     for (auto& it : assigned_values)
 | |
|         longhand_properties.unchecked_append(static_cast<PropertyID>(it.key));
 | |
| 
 | |
|     StyleValueVector longhand_values;
 | |
|     longhand_values.ensure_capacity(assigned_values.size());
 | |
|     for (auto& it : assigned_values) {
 | |
|         if (it.value.size() == 1)
 | |
|             longhand_values.unchecked_append(it.value.take_first());
 | |
|         else
 | |
|             longhand_values.unchecked_append(StyleValueList::create(move(it.value), StyleValueList::Separator::Space));
 | |
|     }
 | |
| 
 | |
|     return { ShorthandStyleValue::create(property_id, move(longhand_properties), move(longhand_values)) };
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_color_scheme_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // normal | [ light | dark | <custom-ident> ]+ && only?
 | |
| 
 | |
|     // normal
 | |
|     {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         tokens.discard_whitespace();
 | |
|         if (tokens.consume_a_token().is_ident("normal"sv)) {
 | |
|             if (tokens.has_next_token())
 | |
|                 return {};
 | |
|             transaction.commit();
 | |
|             return ColorSchemeStyleValue::normal();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bool only = false;
 | |
|     Vector<String> schemes;
 | |
| 
 | |
|     // only? && (..)
 | |
|     {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         tokens.discard_whitespace();
 | |
|         if (tokens.consume_a_token().is_ident("only"sv)) {
 | |
|             only = true;
 | |
|             transaction.commit();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // [ light | dark | <custom-ident> ]+
 | |
|     tokens.discard_whitespace();
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|         // The 'normal', 'light', 'dark', and 'only' keywords are not valid <custom-ident>s in this property.
 | |
|         // Note: only 'normal' is blacklisted here because 'light' and 'dark' aren't parsed differently and 'only' is checked for afterwards
 | |
|         auto ident = parse_custom_ident_value(tokens, { { "normal"sv } });
 | |
|         if (!ident)
 | |
|             return {};
 | |
| 
 | |
|         if (ident->custom_ident() == "only"_fly_string)
 | |
|             break;
 | |
| 
 | |
|         schemes.append(ident->custom_ident().to_string());
 | |
|         tokens.discard_whitespace();
 | |
|         transaction.commit();
 | |
|     }
 | |
| 
 | |
|     // (..) && only?
 | |
|     if (!only) {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         tokens.discard_whitespace();
 | |
|         if (tokens.consume_a_token().is_ident("only"sv)) {
 | |
|             only = true;
 | |
|             transaction.commit();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     tokens.discard_whitespace();
 | |
|     if (tokens.has_next_token() || schemes.is_empty())
 | |
|         return {};
 | |
| 
 | |
|     return ColorSchemeStyleValue::create(schemes, only);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_counter_definitions_value(TokenStream<ComponentValue>& tokens, AllowReversed allow_reversed, i32 default_value_if_not_reversed)
 | |
| {
 | |
|     // If AllowReversed is Yes, parses:
 | |
|     //   [ <counter-name> <integer>? | <reversed-counter-name> <integer>? ]+
 | |
|     // Otherwise parses:
 | |
|     //   [ <counter-name> <integer>? ]+
 | |
| 
 | |
|     // FIXME: This disabled parsing of `reversed()` counters. Remove this line once they're supported.
 | |
|     allow_reversed = AllowReversed::No;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     tokens.discard_whitespace();
 | |
| 
 | |
|     Vector<CounterDefinition> counter_definitions;
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto per_item_transaction = tokens.begin_transaction();
 | |
|         CounterDefinition definition {};
 | |
| 
 | |
|         // <counter-name> | <reversed-counter-name>
 | |
|         auto& token = tokens.next_token();
 | |
| 
 | |
|         // A <counter-name> name cannot match the keyword none; such an identifier is invalid as a <counter-name>.
 | |
|         if (auto counter_name = parse_custom_ident_value(tokens, { { "none"sv } })) {
 | |
|             definition.name = counter_name->custom_ident();
 | |
|             definition.is_reversed = false;
 | |
|         } else if (allow_reversed == AllowReversed::Yes && token.is_function("reversed"sv)) {
 | |
|             TokenStream function_tokens { token.function().value };
 | |
|             tokens.discard_a_token();
 | |
|             function_tokens.discard_whitespace();
 | |
|             auto& name_token = function_tokens.consume_a_token();
 | |
|             if (!name_token.is(Token::Type::Ident))
 | |
|                 break;
 | |
|             function_tokens.discard_whitespace();
 | |
|             if (function_tokens.has_next_token())
 | |
|                 break;
 | |
| 
 | |
|             definition.name = name_token.token().ident();
 | |
|             definition.is_reversed = true;
 | |
|         } else {
 | |
|             break;
 | |
|         }
 | |
|         tokens.discard_whitespace();
 | |
| 
 | |
|         // <integer>?
 | |
|         definition.value = parse_integer_value(tokens);
 | |
|         if (!definition.value && !definition.is_reversed)
 | |
|             definition.value = IntegerStyleValue::create(default_value_if_not_reversed);
 | |
| 
 | |
|         counter_definitions.append(move(definition));
 | |
|         tokens.discard_whitespace();
 | |
|         per_item_transaction.commit();
 | |
|     }
 | |
| 
 | |
|     if (counter_definitions.is_empty())
 | |
|         return {};
 | |
| 
 | |
|     transaction.commit();
 | |
|     return CounterDefinitionsStyleValue::create(move(counter_definitions));
 | |
| }
 | |
| 
 | |
| // https://drafts.csswg.org/css-lists-3/#propdef-counter-increment
 | |
| RefPtr<CSSStyleValue const> Parser::parse_counter_increment_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // [ <counter-name> <integer>? ]+ | none
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     return parse_counter_definitions_value(tokens, AllowReversed::No, 1);
 | |
| }
 | |
| 
 | |
| // https://drafts.csswg.org/css-lists-3/#propdef-counter-reset
 | |
| RefPtr<CSSStyleValue const> Parser::parse_counter_reset_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // [ <counter-name> <integer>? | <reversed-counter-name> <integer>? ]+ | none
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     return parse_counter_definitions_value(tokens, AllowReversed::Yes, 0);
 | |
| }
 | |
| 
 | |
| // https://drafts.csswg.org/css-lists-3/#propdef-counter-set
 | |
| RefPtr<CSSStyleValue const> Parser::parse_counter_set_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // [ <counter-name> <integer>? ]+ | none
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     return parse_counter_definitions_value(tokens, AllowReversed::No, 0);
 | |
| }
 | |
| 
 | |
| // https://drafts.csswg.org/css-ui-3/#cursor
 | |
| RefPtr<CSSStyleValue const> Parser::parse_cursor_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // [ [<url> [<x> <y>]?,]* <built-in-cursor> ]
 | |
|     // So, any number of custom cursor definitions, and then a mandatory cursor name keyword, all comma-separated.
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     StyleValueVector cursors;
 | |
| 
 | |
|     auto parts = parse_a_comma_separated_list_of_component_values(tokens);
 | |
|     for (auto i = 0u; i < parts.size(); ++i) {
 | |
|         auto& part = parts[i];
 | |
|         TokenStream part_tokens { part };
 | |
| 
 | |
|         if (i == parts.size() - 1) {
 | |
|             // Cursor keyword
 | |
|             part_tokens.discard_whitespace();
 | |
|             auto keyword_value = parse_keyword_value(part_tokens);
 | |
|             if (!keyword_value || !keyword_to_cursor(keyword_value->to_keyword()).has_value())
 | |
|                 return {};
 | |
| 
 | |
|             part_tokens.discard_whitespace();
 | |
|             if (part_tokens.has_next_token())
 | |
|                 return {};
 | |
| 
 | |
|             cursors.append(keyword_value.release_nonnull());
 | |
|         } else {
 | |
|             // Custom cursor definition
 | |
|             // <url> [<x> <y>]?
 | |
|             // "Conforming user agents may, instead of <url>, support <image> which is a superset."
 | |
| 
 | |
|             part_tokens.discard_whitespace();
 | |
|             auto image_value = parse_image_value(part_tokens);
 | |
|             if (!image_value)
 | |
|                 return {};
 | |
| 
 | |
|             part_tokens.discard_whitespace();
 | |
| 
 | |
|             if (part_tokens.has_next_token()) {
 | |
|                 // x and y, which are both <number>
 | |
|                 auto x = parse_number(part_tokens);
 | |
|                 part_tokens.discard_whitespace();
 | |
|                 auto y = parse_number(part_tokens);
 | |
|                 part_tokens.discard_whitespace();
 | |
|                 if (!x.has_value() || !y.has_value() || part_tokens.has_next_token())
 | |
|                     return nullptr;
 | |
| 
 | |
|                 cursors.append(CursorStyleValue::create(image_value.release_nonnull(), x.release_value(), y.release_value()));
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             cursors.append(CursorStyleValue::create(image_value.release_nonnull(), {}, {}));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (cursors.is_empty())
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     if (cursors.size() == 1)
 | |
|         return *cursors.first();
 | |
| 
 | |
|     return StyleValueList::create(move(cursors), StyleValueList::Separator::Comma);
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/css-sizing-4/#aspect-ratio
 | |
| RefPtr<CSSStyleValue const> Parser::parse_aspect_ratio_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // `auto || <ratio>`
 | |
|     RefPtr<CSSStyleValue const> auto_value;
 | |
|     RefPtr<CSSStyleValue const> ratio_value;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto maybe_value = parse_css_value_for_property(PropertyID::AspectRatio, tokens);
 | |
|         if (!maybe_value)
 | |
|             return nullptr;
 | |
| 
 | |
|         if (maybe_value->is_ratio()) {
 | |
|             if (ratio_value)
 | |
|                 return nullptr;
 | |
|             ratio_value = maybe_value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (maybe_value->is_keyword() && maybe_value->as_keyword().keyword() == Keyword::Auto) {
 | |
|             if (auto_value)
 | |
|                 return nullptr;
 | |
|             auto_value = maybe_value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     if (auto_value && ratio_value) {
 | |
|         transaction.commit();
 | |
|         return StyleValueList::create(
 | |
|             StyleValueVector { auto_value.release_nonnull(), ratio_value.release_nonnull() },
 | |
|             StyleValueList::Separator::Space);
 | |
|     }
 | |
| 
 | |
|     if (ratio_value) {
 | |
|         transaction.commit();
 | |
|         return ratio_value.release_nonnull();
 | |
|     }
 | |
| 
 | |
|     if (auto_value) {
 | |
|         transaction.commit();
 | |
|         return auto_value.release_nonnull();
 | |
|     }
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_background_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto make_background_shorthand = [&](auto background_color, auto background_image, auto background_position, auto background_size, auto background_repeat, auto background_attachment, auto background_origin, auto background_clip) {
 | |
|         return ShorthandStyleValue::create(PropertyID::Background,
 | |
|             { PropertyID::BackgroundColor, PropertyID::BackgroundImage, PropertyID::BackgroundPosition, PropertyID::BackgroundSize, PropertyID::BackgroundRepeat, PropertyID::BackgroundAttachment, PropertyID::BackgroundOrigin, PropertyID::BackgroundClip },
 | |
|             { move(background_color), move(background_image), move(background_position), move(background_size), move(background_repeat), move(background_attachment), move(background_origin), move(background_clip) });
 | |
|     };
 | |
| 
 | |
|     StyleValueVector background_images;
 | |
|     StyleValueVector background_position_xs;
 | |
|     StyleValueVector background_position_ys;
 | |
|     StyleValueVector background_sizes;
 | |
|     StyleValueVector background_repeats;
 | |
|     StyleValueVector background_attachments;
 | |
|     StyleValueVector background_clips;
 | |
|     StyleValueVector background_origins;
 | |
|     RefPtr<CSSStyleValue const> background_color;
 | |
| 
 | |
|     auto initial_background_image = property_initial_value(PropertyID::BackgroundImage);
 | |
|     auto initial_background_position_x = property_initial_value(PropertyID::BackgroundPositionX);
 | |
|     auto initial_background_position_y = property_initial_value(PropertyID::BackgroundPositionY);
 | |
|     auto initial_background_size = property_initial_value(PropertyID::BackgroundSize);
 | |
|     auto initial_background_repeat = property_initial_value(PropertyID::BackgroundRepeat);
 | |
|     auto initial_background_attachment = property_initial_value(PropertyID::BackgroundAttachment);
 | |
|     auto initial_background_clip = property_initial_value(PropertyID::BackgroundClip);
 | |
|     auto initial_background_origin = property_initial_value(PropertyID::BackgroundOrigin);
 | |
|     auto initial_background_color = property_initial_value(PropertyID::BackgroundColor);
 | |
| 
 | |
|     // Per-layer values
 | |
|     RefPtr<CSSStyleValue const> background_image;
 | |
|     RefPtr<CSSStyleValue const> background_position_x;
 | |
|     RefPtr<CSSStyleValue const> background_position_y;
 | |
|     RefPtr<CSSStyleValue const> background_size;
 | |
|     RefPtr<CSSStyleValue const> background_repeat;
 | |
|     RefPtr<CSSStyleValue const> background_attachment;
 | |
|     RefPtr<CSSStyleValue const> background_clip;
 | |
|     RefPtr<CSSStyleValue const> background_origin;
 | |
| 
 | |
|     bool has_multiple_layers = false;
 | |
|     // BackgroundSize is always parsed as part of BackgroundPosition, so we don't include it here.
 | |
|     Vector<PropertyID> remaining_layer_properties {
 | |
|         PropertyID::BackgroundAttachment,
 | |
|         PropertyID::BackgroundClip,
 | |
|         PropertyID::BackgroundColor,
 | |
|         PropertyID::BackgroundImage,
 | |
|         PropertyID::BackgroundOrigin,
 | |
|         PropertyID::BackgroundPosition,
 | |
|         PropertyID::BackgroundRepeat,
 | |
|     };
 | |
| 
 | |
|     auto background_layer_is_valid = [&](bool allow_background_color) -> bool {
 | |
|         if (allow_background_color) {
 | |
|             if (background_color)
 | |
|                 return true;
 | |
|         } else {
 | |
|             if (background_color)
 | |
|                 return false;
 | |
|         }
 | |
|         return background_image || background_position_x || background_position_y || background_size || background_repeat || background_attachment || background_clip || background_origin;
 | |
|     };
 | |
| 
 | |
|     auto complete_background_layer = [&]() {
 | |
|         background_images.append(background_image ? background_image.release_nonnull() : initial_background_image);
 | |
|         background_position_xs.append(background_position_x ? background_position_x.release_nonnull() : initial_background_position_x);
 | |
|         background_position_ys.append(background_position_y ? background_position_y.release_nonnull() : initial_background_position_y);
 | |
|         background_sizes.append(background_size ? background_size.release_nonnull() : initial_background_size);
 | |
|         background_repeats.append(background_repeat ? background_repeat.release_nonnull() : initial_background_repeat);
 | |
|         background_attachments.append(background_attachment ? background_attachment.release_nonnull() : initial_background_attachment);
 | |
| 
 | |
|         if (!background_origin && !background_clip) {
 | |
|             background_origin = initial_background_origin;
 | |
|             background_clip = initial_background_clip;
 | |
|         } else if (!background_clip) {
 | |
|             background_clip = background_origin;
 | |
|         }
 | |
|         background_origins.append(background_origin.release_nonnull());
 | |
|         background_clips.append(background_clip.release_nonnull());
 | |
| 
 | |
|         background_image = nullptr;
 | |
|         background_position_x = nullptr;
 | |
|         background_position_y = nullptr;
 | |
|         background_size = nullptr;
 | |
|         background_repeat = nullptr;
 | |
|         background_attachment = nullptr;
 | |
|         background_clip = nullptr;
 | |
|         background_origin = nullptr;
 | |
| 
 | |
|         remaining_layer_properties.clear_with_capacity();
 | |
|         remaining_layer_properties.unchecked_append(PropertyID::BackgroundAttachment);
 | |
|         remaining_layer_properties.unchecked_append(PropertyID::BackgroundClip);
 | |
|         remaining_layer_properties.unchecked_append(PropertyID::BackgroundColor);
 | |
|         remaining_layer_properties.unchecked_append(PropertyID::BackgroundImage);
 | |
|         remaining_layer_properties.unchecked_append(PropertyID::BackgroundOrigin);
 | |
|         remaining_layer_properties.unchecked_append(PropertyID::BackgroundPosition);
 | |
|         remaining_layer_properties.unchecked_append(PropertyID::BackgroundRepeat);
 | |
|     };
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         if (tokens.next_token().is(Token::Type::Comma)) {
 | |
|             has_multiple_layers = true;
 | |
|             if (!background_layer_is_valid(false))
 | |
|                 return nullptr;
 | |
|             complete_background_layer();
 | |
|             tokens.discard_a_token();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         auto value_and_property = parse_css_value_for_properties(remaining_layer_properties, tokens);
 | |
|         if (!value_and_property.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = value_and_property->style_value;
 | |
|         remove_property(remaining_layer_properties, value_and_property->property);
 | |
| 
 | |
|         switch (value_and_property->property) {
 | |
|         case PropertyID::BackgroundAttachment:
 | |
|             VERIFY(!background_attachment);
 | |
|             background_attachment = value.release_nonnull();
 | |
|             continue;
 | |
|         case PropertyID::BackgroundColor:
 | |
|             VERIFY(!background_color);
 | |
|             background_color = value.release_nonnull();
 | |
|             continue;
 | |
|         case PropertyID::BackgroundImage:
 | |
|             VERIFY(!background_image);
 | |
|             background_image = value.release_nonnull();
 | |
|             continue;
 | |
|         case PropertyID::BackgroundClip:
 | |
|         case PropertyID::BackgroundOrigin: {
 | |
|             // background-origin and background-clip accept the same values. From the spec:
 | |
|             //   "If one <box> value is present then it sets both background-origin and background-clip to that value.
 | |
|             //    If two values are present, then the first sets background-origin and the second background-clip."
 | |
|             //        - https://www.w3.org/TR/css-backgrounds-3/#background
 | |
|             // So, we put the first one in background-origin, then if we get a second, we put it in background-clip.
 | |
|             // If we only get one, we copy the value before creating the ShorthandStyleValue.
 | |
|             if (!background_origin) {
 | |
|                 background_origin = value.release_nonnull();
 | |
|             } else if (!background_clip) {
 | |
|                 background_clip = value.release_nonnull();
 | |
|             } else {
 | |
|                 VERIFY_NOT_REACHED();
 | |
|             }
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::BackgroundPosition: {
 | |
|             VERIFY(!background_position_x && !background_position_y);
 | |
|             auto position = value.release_nonnull();
 | |
|             background_position_x = position->as_position().edge_x();
 | |
|             background_position_y = position->as_position().edge_y();
 | |
| 
 | |
|             // Attempt to parse `/ <background-size>`
 | |
|             auto background_size_transaction = tokens.begin_transaction();
 | |
|             auto& maybe_slash = tokens.consume_a_token();
 | |
|             if (maybe_slash.is_delim('/')) {
 | |
|                 if (auto maybe_background_size = parse_single_background_size_value(tokens)) {
 | |
|                     background_size_transaction.commit();
 | |
|                     background_size = maybe_background_size.release_nonnull();
 | |
|                     continue;
 | |
|                 }
 | |
|                 return nullptr;
 | |
|             }
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::BackgroundRepeat: {
 | |
|             VERIFY(!background_repeat);
 | |
|             tokens.reconsume_current_input_token();
 | |
|             if (auto maybe_repeat = parse_single_background_repeat_value(tokens)) {
 | |
|                 background_repeat = maybe_repeat.release_nonnull();
 | |
|                 continue;
 | |
|             }
 | |
|             return nullptr;
 | |
|         }
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     if (!background_layer_is_valid(true))
 | |
|         return nullptr;
 | |
| 
 | |
|     // We only need to create StyleValueLists if there are multiple layers.
 | |
|     // Otherwise, we can pass the single StyleValues directly.
 | |
|     if (has_multiple_layers) {
 | |
|         complete_background_layer();
 | |
| 
 | |
|         if (!background_color)
 | |
|             background_color = initial_background_color;
 | |
|         transaction.commit();
 | |
|         return make_background_shorthand(
 | |
|             background_color.release_nonnull(),
 | |
|             StyleValueList::create(move(background_images), StyleValueList::Separator::Comma),
 | |
|             ShorthandStyleValue::create(PropertyID::BackgroundPosition,
 | |
|                 { PropertyID::BackgroundPositionX, PropertyID::BackgroundPositionY },
 | |
|                 { StyleValueList::create(move(background_position_xs), StyleValueList::Separator::Comma),
 | |
|                     StyleValueList::create(move(background_position_ys), StyleValueList::Separator::Comma) }),
 | |
|             StyleValueList::create(move(background_sizes), StyleValueList::Separator::Comma),
 | |
|             StyleValueList::create(move(background_repeats), StyleValueList::Separator::Comma),
 | |
|             StyleValueList::create(move(background_attachments), StyleValueList::Separator::Comma),
 | |
|             StyleValueList::create(move(background_origins), StyleValueList::Separator::Comma),
 | |
|             StyleValueList::create(move(background_clips), StyleValueList::Separator::Comma));
 | |
|     }
 | |
| 
 | |
|     if (!background_color)
 | |
|         background_color = initial_background_color;
 | |
|     if (!background_image)
 | |
|         background_image = initial_background_image;
 | |
|     if (!background_position_x)
 | |
|         background_position_x = initial_background_position_x;
 | |
|     if (!background_position_y)
 | |
|         background_position_y = initial_background_position_y;
 | |
|     if (!background_size)
 | |
|         background_size = initial_background_size;
 | |
|     if (!background_repeat)
 | |
|         background_repeat = initial_background_repeat;
 | |
|     if (!background_attachment)
 | |
|         background_attachment = initial_background_attachment;
 | |
| 
 | |
|     if (!background_origin && !background_clip) {
 | |
|         background_origin = initial_background_origin;
 | |
|         background_clip = initial_background_clip;
 | |
|     } else if (!background_clip) {
 | |
|         background_clip = background_origin;
 | |
|     }
 | |
| 
 | |
|     transaction.commit();
 | |
|     return make_background_shorthand(
 | |
|         background_color.release_nonnull(),
 | |
|         background_image.release_nonnull(),
 | |
|         ShorthandStyleValue::create(PropertyID::BackgroundPosition,
 | |
|             { PropertyID::BackgroundPositionX, PropertyID::BackgroundPositionY },
 | |
|             { background_position_x.release_nonnull(), background_position_y.release_nonnull() }),
 | |
|         background_size.release_nonnull(),
 | |
|         background_repeat.release_nonnull(),
 | |
|         background_attachment.release_nonnull(),
 | |
|         background_origin.release_nonnull(),
 | |
|         background_clip.release_nonnull());
 | |
| }
 | |
| 
 | |
| static Optional<LengthPercentage> style_value_to_length_percentage(auto value)
 | |
| {
 | |
|     if (value->is_percentage())
 | |
|         return LengthPercentage { value->as_percentage().percentage() };
 | |
|     if (value->is_length())
 | |
|         return LengthPercentage { value->as_length().length() };
 | |
|     if (value->is_calculated())
 | |
|         return LengthPercentage { value->as_calculated() };
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_single_background_position_x_or_y_value(TokenStream<ComponentValue>& tokens, PropertyID property)
 | |
| {
 | |
|     Optional<PositionEdge> relative_edge {};
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     if (!tokens.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     auto value = parse_css_value_for_property(property, tokens);
 | |
|     if (!value)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (value->is_keyword()) {
 | |
|         auto keyword = value->to_keyword();
 | |
|         if (keyword == Keyword::Center) {
 | |
|             transaction.commit();
 | |
|             return EdgeStyleValue::create(PositionEdge::Center, {});
 | |
|         }
 | |
|         if (auto edge = keyword_to_position_edge(keyword); edge.has_value()) {
 | |
|             relative_edge = edge;
 | |
|         } else {
 | |
|             return nullptr;
 | |
|         }
 | |
|         if (tokens.has_next_token()) {
 | |
|             value = parse_css_value_for_property(property, tokens);
 | |
|             if (!value) {
 | |
|                 transaction.commit();
 | |
|                 return EdgeStyleValue::create(relative_edge, {});
 | |
|             }
 | |
|             if (value->is_keyword())
 | |
|                 return {};
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     auto offset = style_value_to_length_percentage(value);
 | |
|     if (offset.has_value()) {
 | |
|         transaction.commit();
 | |
|         return EdgeStyleValue::create(relative_edge, *offset);
 | |
|     }
 | |
| 
 | |
|     if (!relative_edge.has_value()) {
 | |
|         if (property == PropertyID::BackgroundPositionX) {
 | |
|             // [ center | [ [ left | right | x-start | x-end ]? <length-percentage>? ]! ]#
 | |
|             relative_edge = PositionEdge::Left;
 | |
|         } else if (property == PropertyID::BackgroundPositionY) {
 | |
|             // [ center | [ [ top | bottom | y-start | y-end ]? <length-percentage>? ]! ]#
 | |
|             relative_edge = PositionEdge::Top;
 | |
|         } else {
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // If no offset is provided create this element but with an offset of default value of zero
 | |
|     transaction.commit();
 | |
|     return EdgeStyleValue::create(relative_edge, {});
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_single_background_repeat_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto is_directional_repeat = [](CSSStyleValue const& value) -> bool {
 | |
|         auto keyword = value.to_keyword();
 | |
|         return keyword == Keyword::RepeatX || keyword == Keyword::RepeatY;
 | |
|     };
 | |
| 
 | |
|     auto as_repeat = [](Keyword keyword) -> Optional<Repeat> {
 | |
|         switch (keyword) {
 | |
|         case Keyword::NoRepeat:
 | |
|             return Repeat::NoRepeat;
 | |
|         case Keyword::Repeat:
 | |
|             return Repeat::Repeat;
 | |
|         case Keyword::Round:
 | |
|             return Repeat::Round;
 | |
|         case Keyword::Space:
 | |
|             return Repeat::Space;
 | |
|         default:
 | |
|             return {};
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     auto maybe_x_value = parse_css_value_for_property(PropertyID::BackgroundRepeat, tokens);
 | |
|     if (!maybe_x_value)
 | |
|         return nullptr;
 | |
|     auto x_value = maybe_x_value.release_nonnull();
 | |
| 
 | |
|     if (is_directional_repeat(*x_value)) {
 | |
|         auto keyword = x_value->to_keyword();
 | |
|         transaction.commit();
 | |
|         return BackgroundRepeatStyleValue::create(
 | |
|             keyword == Keyword::RepeatX ? Repeat::Repeat : Repeat::NoRepeat,
 | |
|             keyword == Keyword::RepeatX ? Repeat::NoRepeat : Repeat::Repeat);
 | |
|     }
 | |
| 
 | |
|     auto x_repeat = as_repeat(x_value->to_keyword());
 | |
|     if (!x_repeat.has_value())
 | |
|         return nullptr;
 | |
| 
 | |
|     // See if we have a second value for Y
 | |
|     auto maybe_y_value = parse_css_value_for_property(PropertyID::BackgroundRepeat, tokens);
 | |
|     if (!maybe_y_value) {
 | |
|         // We don't have a second value, so use x for both
 | |
|         transaction.commit();
 | |
|         return BackgroundRepeatStyleValue::create(x_repeat.value(), x_repeat.value());
 | |
|     }
 | |
|     auto y_value = maybe_y_value.release_nonnull();
 | |
|     if (is_directional_repeat(*y_value))
 | |
|         return nullptr;
 | |
| 
 | |
|     auto y_repeat = as_repeat(y_value->to_keyword());
 | |
|     if (!y_repeat.has_value())
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return BackgroundRepeatStyleValue::create(x_repeat.value(), y_repeat.value());
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_single_background_size_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto get_length_percentage = [](CSSStyleValue const& style_value) -> Optional<LengthPercentage> {
 | |
|         if (style_value.has_auto())
 | |
|             return LengthPercentage { Length::make_auto() };
 | |
|         if (style_value.is_percentage())
 | |
|             return LengthPercentage { style_value.as_percentage().percentage() };
 | |
|         if (style_value.is_length())
 | |
|             return LengthPercentage { style_value.as_length().length() };
 | |
|         if (style_value.is_calculated())
 | |
|             return LengthPercentage { style_value.as_calculated() };
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     auto maybe_x_value = parse_css_value_for_property(PropertyID::BackgroundSize, tokens);
 | |
|     if (!maybe_x_value)
 | |
|         return nullptr;
 | |
|     auto x_value = maybe_x_value.release_nonnull();
 | |
| 
 | |
|     if (x_value->to_keyword() == Keyword::Cover || x_value->to_keyword() == Keyword::Contain) {
 | |
|         transaction.commit();
 | |
|         return x_value;
 | |
|     }
 | |
| 
 | |
|     auto maybe_y_value = parse_css_value_for_property(PropertyID::BackgroundSize, tokens);
 | |
|     if (!maybe_y_value) {
 | |
|         auto y_value = LengthPercentage { Length::make_auto() };
 | |
|         auto x_size = get_length_percentage(*x_value);
 | |
|         if (!x_size.has_value())
 | |
|             return nullptr;
 | |
| 
 | |
|         transaction.commit();
 | |
|         return BackgroundSizeStyleValue::create(x_size.value(), y_value);
 | |
|     }
 | |
| 
 | |
|     auto y_value = maybe_y_value.release_nonnull();
 | |
|     auto x_size = get_length_percentage(*x_value);
 | |
|     auto y_size = get_length_percentage(*y_value);
 | |
| 
 | |
|     if (!x_size.has_value() || !y_size.has_value())
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return BackgroundSizeStyleValue::create(x_size.release_value(), y_size.release_value());
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_border_value(PropertyID property_id, TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     RefPtr<CSSStyleValue const> border_width;
 | |
|     RefPtr<CSSStyleValue const> border_color;
 | |
|     RefPtr<CSSStyleValue const> border_style;
 | |
| 
 | |
|     auto color_property = PropertyID::Invalid;
 | |
|     auto style_property = PropertyID::Invalid;
 | |
|     auto width_property = PropertyID::Invalid;
 | |
| 
 | |
|     switch (property_id) {
 | |
|     case PropertyID::Border:
 | |
|         color_property = PropertyID::BorderColor;
 | |
|         style_property = PropertyID::BorderStyle;
 | |
|         width_property = PropertyID::BorderWidth;
 | |
|         break;
 | |
|     case PropertyID::BorderBottom:
 | |
|         color_property = PropertyID::BorderBottomColor;
 | |
|         style_property = PropertyID::BorderBottomStyle;
 | |
|         width_property = PropertyID::BorderBottomWidth;
 | |
|         break;
 | |
|     case PropertyID::BorderLeft:
 | |
|         color_property = PropertyID::BorderLeftColor;
 | |
|         style_property = PropertyID::BorderLeftStyle;
 | |
|         width_property = PropertyID::BorderLeftWidth;
 | |
|         break;
 | |
|     case PropertyID::BorderRight:
 | |
|         color_property = PropertyID::BorderRightColor;
 | |
|         style_property = PropertyID::BorderRightStyle;
 | |
|         width_property = PropertyID::BorderRightWidth;
 | |
|         break;
 | |
|     case PropertyID::BorderTop:
 | |
|         color_property = PropertyID::BorderTopColor;
 | |
|         style_property = PropertyID::BorderTopStyle;
 | |
|         width_property = PropertyID::BorderTopWidth;
 | |
|         break;
 | |
|     default:
 | |
|         VERIFY_NOT_REACHED();
 | |
|     }
 | |
| 
 | |
|     auto remaining_longhands = Vector { width_property, color_property, style_property };
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto property_and_value = parse_css_value_for_properties(remaining_longhands, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = property_and_value->style_value;
 | |
|         remove_property(remaining_longhands, property_and_value->property);
 | |
| 
 | |
|         if (property_and_value->property == width_property) {
 | |
|             VERIFY(!border_width);
 | |
|             border_width = value.release_nonnull();
 | |
|         } else if (property_and_value->property == color_property) {
 | |
|             VERIFY(!border_color);
 | |
|             border_color = value.release_nonnull();
 | |
|         } else if (property_and_value->property == style_property) {
 | |
|             VERIFY(!border_style);
 | |
|             border_style = value.release_nonnull();
 | |
|         } else {
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!border_width)
 | |
|         border_width = property_initial_value(width_property);
 | |
|     if (!border_style)
 | |
|         border_style = property_initial_value(style_property);
 | |
|     if (!border_color)
 | |
|         border_color = property_initial_value(color_property);
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(property_id,
 | |
|         { width_property, style_property, color_property },
 | |
|         { border_width.release_nonnull(), border_style.release_nonnull(), border_color.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_border_radius_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     if (tokens.remaining_token_count() == 2) {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         auto horizontal = parse_length_percentage(tokens);
 | |
|         auto vertical = parse_length_percentage(tokens);
 | |
|         if (horizontal.has_value() && vertical.has_value()) {
 | |
|             transaction.commit();
 | |
|             return BorderRadiusStyleValue::create(horizontal.release_value(), vertical.release_value());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (tokens.remaining_token_count() == 1) {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         auto radius = parse_length_percentage(tokens);
 | |
|         if (radius.has_value()) {
 | |
|             transaction.commit();
 | |
|             return BorderRadiusStyleValue::create(radius.value(), radius.value());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_border_radius_shorthand_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto top_left = [&](Vector<LengthPercentage>& radii) { return radii[0]; };
 | |
|     auto top_right = [&](Vector<LengthPercentage>& radii) {
 | |
|         switch (radii.size()) {
 | |
|         case 4:
 | |
|         case 3:
 | |
|         case 2:
 | |
|             return radii[1];
 | |
|         case 1:
 | |
|             return radii[0];
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     };
 | |
|     auto bottom_right = [&](Vector<LengthPercentage>& radii) {
 | |
|         switch (radii.size()) {
 | |
|         case 4:
 | |
|         case 3:
 | |
|             return radii[2];
 | |
|         case 2:
 | |
|         case 1:
 | |
|             return radii[0];
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     };
 | |
|     auto bottom_left = [&](Vector<LengthPercentage>& radii) {
 | |
|         switch (radii.size()) {
 | |
|         case 4:
 | |
|             return radii[3];
 | |
|         case 3:
 | |
|         case 2:
 | |
|             return radii[1];
 | |
|         case 1:
 | |
|             return radii[0];
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     Vector<LengthPercentage> horizontal_radii;
 | |
|     Vector<LengthPercentage> vertical_radii;
 | |
|     bool reading_vertical = false;
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         if (tokens.next_token().is_delim('/')) {
 | |
|             if (reading_vertical || horizontal_radii.is_empty())
 | |
|                 return nullptr;
 | |
| 
 | |
|             reading_vertical = true;
 | |
|             tokens.discard_a_token(); // `/`
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         auto maybe_dimension = parse_length_percentage(tokens);
 | |
|         if (!maybe_dimension.has_value())
 | |
|             return nullptr;
 | |
|         if (maybe_dimension->is_length() && !property_accepts_length(PropertyID::BorderRadius, maybe_dimension->length()))
 | |
|             return nullptr;
 | |
|         if (maybe_dimension->is_percentage() && !property_accepts_percentage(PropertyID::BorderRadius, maybe_dimension->percentage()))
 | |
|             return nullptr;
 | |
|         if (reading_vertical) {
 | |
|             vertical_radii.append(maybe_dimension.release_value());
 | |
|         } else {
 | |
|             horizontal_radii.append(maybe_dimension.release_value());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (horizontal_radii.size() > 4 || vertical_radii.size() > 4
 | |
|         || horizontal_radii.is_empty()
 | |
|         || (reading_vertical && vertical_radii.is_empty()))
 | |
|         return nullptr;
 | |
| 
 | |
|     auto top_left_radius = BorderRadiusStyleValue::create(top_left(horizontal_radii),
 | |
|         vertical_radii.is_empty() ? top_left(horizontal_radii) : top_left(vertical_radii));
 | |
|     auto top_right_radius = BorderRadiusStyleValue::create(top_right(horizontal_radii),
 | |
|         vertical_radii.is_empty() ? top_right(horizontal_radii) : top_right(vertical_radii));
 | |
|     auto bottom_right_radius = BorderRadiusStyleValue::create(bottom_right(horizontal_radii),
 | |
|         vertical_radii.is_empty() ? bottom_right(horizontal_radii) : bottom_right(vertical_radii));
 | |
|     auto bottom_left_radius = BorderRadiusStyleValue::create(bottom_left(horizontal_radii),
 | |
|         vertical_radii.is_empty() ? bottom_left(horizontal_radii) : bottom_left(vertical_radii));
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::BorderRadius,
 | |
|         { PropertyID::BorderTopLeftRadius, PropertyID::BorderTopRightRadius, PropertyID::BorderBottomRightRadius, PropertyID::BorderBottomLeftRadius },
 | |
|         { move(top_left_radius), move(top_right_radius), move(bottom_right_radius), move(bottom_left_radius) });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_columns_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     if (tokens.remaining_token_count() > 2)
 | |
|         return nullptr;
 | |
| 
 | |
|     RefPtr<CSSStyleValue const> column_count;
 | |
|     RefPtr<CSSStyleValue const> column_width;
 | |
| 
 | |
|     Vector<PropertyID> remaining_longhands { PropertyID::ColumnCount, PropertyID::ColumnWidth };
 | |
|     int found_autos = 0;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto property_and_value = parse_css_value_for_properties(remaining_longhands, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = property_and_value->style_value;
 | |
| 
 | |
|         // since the values can be in either order, we want to skip over autos
 | |
|         if (value->has_auto()) {
 | |
|             found_autos++;
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         remove_property(remaining_longhands, property_and_value->property);
 | |
| 
 | |
|         switch (property_and_value->property) {
 | |
|         case PropertyID::ColumnCount: {
 | |
|             VERIFY(!column_count);
 | |
|             column_count = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::ColumnWidth: {
 | |
|             VERIFY(!column_width);
 | |
|             column_width = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (found_autos > 2)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (found_autos == 2) {
 | |
|         column_count = CSSKeywordValue::create(Keyword::Auto);
 | |
|         column_width = CSSKeywordValue::create(Keyword::Auto);
 | |
|     }
 | |
| 
 | |
|     if (found_autos == 1) {
 | |
|         if (!column_count)
 | |
|             column_count = CSSKeywordValue::create(Keyword::Auto);
 | |
|         if (!column_width)
 | |
|             column_width = CSSKeywordValue::create(Keyword::Auto);
 | |
|     }
 | |
| 
 | |
|     if (!column_count)
 | |
|         column_count = property_initial_value(PropertyID::ColumnCount);
 | |
|     if (!column_width)
 | |
|         column_width = property_initial_value(PropertyID::ColumnWidth);
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::Columns,
 | |
|         { PropertyID::ColumnCount, PropertyID::ColumnWidth },
 | |
|         { column_count.release_nonnull(), column_width.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_shadow_value(TokenStream<ComponentValue>& tokens, AllowInsetKeyword allow_inset_keyword)
 | |
| {
 | |
|     // "none"
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     return parse_comma_separated_value_list(tokens, [this, allow_inset_keyword](auto& tokens) {
 | |
|         return parse_single_shadow_value(tokens, allow_inset_keyword);
 | |
|     });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_single_shadow_value(TokenStream<ComponentValue>& tokens, AllowInsetKeyword allow_inset_keyword)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     RefPtr<CSSStyleValue const> color;
 | |
|     RefPtr<CSSStyleValue const> offset_x;
 | |
|     RefPtr<CSSStyleValue const> offset_y;
 | |
|     RefPtr<CSSStyleValue const> blur_radius;
 | |
|     RefPtr<CSSStyleValue const> spread_distance;
 | |
|     Optional<ShadowPlacement> placement;
 | |
| 
 | |
|     auto possibly_dynamic_length = [&](ComponentValue const& token) -> RefPtr<CSSStyleValue const> {
 | |
|         auto tokens = TokenStream<ComponentValue>::of_single_token(token);
 | |
|         auto maybe_length = parse_length(tokens);
 | |
|         if (!maybe_length.has_value())
 | |
|             return nullptr;
 | |
|         return maybe_length->as_style_value();
 | |
|     };
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         if (auto maybe_color = parse_color_value(tokens); maybe_color) {
 | |
|             if (color)
 | |
|                 return nullptr;
 | |
|             color = maybe_color.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         auto const& token = tokens.next_token();
 | |
|         if (auto maybe_offset_x = possibly_dynamic_length(token); maybe_offset_x) {
 | |
|             // horizontal offset
 | |
|             if (offset_x)
 | |
|                 return nullptr;
 | |
|             offset_x = maybe_offset_x;
 | |
|             tokens.discard_a_token();
 | |
| 
 | |
|             // vertical offset
 | |
|             if (!tokens.has_next_token())
 | |
|                 return nullptr;
 | |
|             auto maybe_offset_y = possibly_dynamic_length(tokens.next_token());
 | |
|             if (!maybe_offset_y)
 | |
|                 return nullptr;
 | |
|             offset_y = maybe_offset_y;
 | |
|             tokens.discard_a_token();
 | |
| 
 | |
|             // blur radius (optional)
 | |
|             if (!tokens.has_next_token())
 | |
|                 break;
 | |
|             auto maybe_blur_radius = possibly_dynamic_length(tokens.next_token());
 | |
|             if (!maybe_blur_radius)
 | |
|                 continue;
 | |
|             blur_radius = maybe_blur_radius;
 | |
|             if (blur_radius->is_length() && blur_radius->as_length().length().raw_value() < 0)
 | |
|                 return nullptr;
 | |
|             if (blur_radius->is_percentage() && blur_radius->as_percentage().value() < 0)
 | |
|                 return nullptr;
 | |
|             tokens.discard_a_token();
 | |
| 
 | |
|             // spread distance (optional)
 | |
|             if (!tokens.has_next_token())
 | |
|                 break;
 | |
|             auto maybe_spread_distance = possibly_dynamic_length(tokens.next_token());
 | |
|             if (!maybe_spread_distance)
 | |
|                 continue;
 | |
|             spread_distance = maybe_spread_distance;
 | |
|             tokens.discard_a_token();
 | |
| 
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (allow_inset_keyword == AllowInsetKeyword::Yes && token.is_ident("inset"sv)) {
 | |
|             if (placement.has_value())
 | |
|                 return nullptr;
 | |
|             placement = ShadowPlacement::Inner;
 | |
|             tokens.discard_a_token();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (token.is(Token::Type::Comma))
 | |
|             break;
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     // If color is absent, default to `currentColor`
 | |
|     if (!color)
 | |
|         color = CSSKeywordValue::create(Keyword::Currentcolor);
 | |
| 
 | |
|     // x/y offsets are required
 | |
|     if (!offset_x || !offset_y)
 | |
|         return nullptr;
 | |
| 
 | |
|     // Other lengths default to 0
 | |
|     if (!blur_radius)
 | |
|         blur_radius = LengthStyleValue::create(Length::make_px(0));
 | |
|     if (!spread_distance)
 | |
|         spread_distance = LengthStyleValue::create(Length::make_px(0));
 | |
| 
 | |
|     // Placement is outer by default
 | |
|     if (!placement.has_value())
 | |
|         placement = ShadowPlacement::Outer;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShadowStyleValue::create(color.release_nonnull(), offset_x.release_nonnull(), offset_y.release_nonnull(), blur_radius.release_nonnull(), spread_distance.release_nonnull(), placement.release_value());
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_rotate_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // Value:	none | <angle> | [ x | y | z | <number>{3} ] && <angle>
 | |
| 
 | |
|     if (tokens.remaining_token_count() == 1) {
 | |
|         // "none"
 | |
|         if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|             return none;
 | |
| 
 | |
|         // <angle>
 | |
|         if (auto angle = parse_angle_value(tokens))
 | |
|             return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::Rotate, { angle.release_nonnull() });
 | |
|     }
 | |
| 
 | |
|     auto parse_one_of_xyz = [&]() -> Optional<ComponentValue const&> {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         auto const& axis = tokens.consume_a_token();
 | |
| 
 | |
|         if (axis.is_ident("x"sv) || axis.is_ident("y"sv) || axis.is_ident("z"sv)) {
 | |
|             transaction.commit();
 | |
|             return axis;
 | |
|         }
 | |
| 
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     // [ x | y | z ] && <angle>
 | |
|     if (tokens.remaining_token_count() == 2) {
 | |
|         // Try parsing `x <angle>`
 | |
|         if (auto axis = parse_one_of_xyz(); axis.has_value()) {
 | |
|             if (auto angle = parse_angle_value(tokens); angle) {
 | |
|                 if (axis->is_ident("x"sv))
 | |
|                     return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::RotateX, { angle.release_nonnull() });
 | |
|                 if (axis->is_ident("y"sv))
 | |
|                     return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::RotateY, { angle.release_nonnull() });
 | |
|                 if (axis->is_ident("z"sv))
 | |
|                     return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::RotateZ, { angle.release_nonnull() });
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Try parsing `<angle> x`
 | |
|         if (auto angle = parse_angle_value(tokens); angle) {
 | |
|             if (auto axis = parse_one_of_xyz(); axis.has_value()) {
 | |
|                 if (axis->is_ident("x"sv))
 | |
|                     return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::RotateX, { angle.release_nonnull() });
 | |
|                 if (axis->is_ident("y"sv))
 | |
|                     return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::RotateY, { angle.release_nonnull() });
 | |
|                 if (axis->is_ident("z"sv))
 | |
|                     return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::RotateZ, { angle.release_nonnull() });
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     auto parse_three_numbers = [&]() -> Optional<StyleValueVector> {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         StyleValueVector numbers;
 | |
|         for (size_t i = 0; i < 3; ++i) {
 | |
|             if (auto number = parse_number_value(tokens); number) {
 | |
|                 numbers.append(number.release_nonnull());
 | |
|             } else {
 | |
|                 return {};
 | |
|             }
 | |
|         }
 | |
|         transaction.commit();
 | |
|         return numbers;
 | |
|     };
 | |
| 
 | |
|     // <number>{3} && <angle>
 | |
|     if (tokens.remaining_token_count() == 4) {
 | |
|         // Try parsing <number>{3} <angle>
 | |
|         if (auto maybe_numbers = parse_three_numbers(); maybe_numbers.has_value()) {
 | |
|             if (auto angle = parse_angle_value(tokens); angle) {
 | |
|                 auto numbers = maybe_numbers.release_value();
 | |
|                 return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::Rotate3d, { numbers[0], numbers[1], numbers[2], angle.release_nonnull() });
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Try parsing <angle> <number>{3}
 | |
|         if (auto angle = parse_angle_value(tokens); angle) {
 | |
|             if (auto maybe_numbers = parse_three_numbers(); maybe_numbers.has_value()) {
 | |
|                 auto numbers = maybe_numbers.release_value();
 | |
|                 return TransformationStyleValue::create(PropertyID::Rotate, TransformFunction::Rotate3d, { numbers[0], numbers[1], numbers[2], angle.release_nonnull() });
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_stroke_dasharray_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://svgwg.org/svg2-draft/painting.html#StrokeDashing
 | |
|     // Value: none | <dasharray>
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     // https://svgwg.org/svg2-draft/painting.html#DataTypeDasharray
 | |
|     // <dasharray> = [ [ <length-percentage> | <number> ]+ ]#
 | |
|     Vector<ValueComparingNonnullRefPtr<CSSStyleValue const>> dashes;
 | |
|     while (tokens.has_next_token()) {
 | |
|         tokens.discard_whitespace();
 | |
| 
 | |
|         // A <dasharray> is a list of comma and/or white space separated <number> or <length-percentage> values. A <number> value represents a value in user units.
 | |
|         auto value = parse_number_value(tokens);
 | |
|         if (value) {
 | |
|             dashes.append(value.release_nonnull());
 | |
|         } else {
 | |
|             auto value = parse_length_percentage_value(tokens);
 | |
|             if (!value)
 | |
|                 return {};
 | |
|             dashes.append(value.release_nonnull());
 | |
|         }
 | |
| 
 | |
|         tokens.discard_whitespace();
 | |
|         if (tokens.has_next_token() && tokens.next_token().is(Token::Type::Comma))
 | |
|             tokens.discard_a_token();
 | |
|     }
 | |
| 
 | |
|     return StyleValueList::create(move(dashes), StyleValueList::Separator::Comma);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_content_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // FIXME: `content` accepts several kinds of function() type, which we don't handle in property_accepts_value() yet.
 | |
| 
 | |
|     auto is_single_value_keyword = [](Keyword keyword) -> bool {
 | |
|         switch (keyword) {
 | |
|         case Keyword::None:
 | |
|         case Keyword::Normal:
 | |
|             return true;
 | |
|         default:
 | |
|             return false;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     if (tokens.remaining_token_count() == 1) {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         if (auto keyword = parse_keyword_value(tokens)) {
 | |
|             if (is_single_value_keyword(keyword->to_keyword())) {
 | |
|                 transaction.commit();
 | |
|                 return keyword;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     StyleValueVector content_values;
 | |
|     StyleValueVector alt_text_values;
 | |
|     bool in_alt_text = false;
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto& next = tokens.next_token();
 | |
|         if (next.is_delim('/')) {
 | |
|             if (in_alt_text || content_values.is_empty())
 | |
|                 return nullptr;
 | |
|             in_alt_text = true;
 | |
|             tokens.discard_a_token();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (auto style_value = parse_css_value_for_property(PropertyID::Content, tokens)) {
 | |
|             if (is_single_value_keyword(style_value->to_keyword()))
 | |
|                 return nullptr;
 | |
| 
 | |
|             if (in_alt_text) {
 | |
|                 alt_text_values.append(style_value.release_nonnull());
 | |
|             } else {
 | |
|                 content_values.append(style_value.release_nonnull());
 | |
|             }
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     if (content_values.is_empty())
 | |
|         return nullptr;
 | |
|     if (in_alt_text && alt_text_values.is_empty())
 | |
|         return nullptr;
 | |
| 
 | |
|     RefPtr<StyleValueList> alt_text;
 | |
|     if (!alt_text_values.is_empty())
 | |
|         alt_text = StyleValueList::create(move(alt_text_values), StyleValueList::Separator::Space);
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ContentStyleValue::create(StyleValueList::create(move(content_values), StyleValueList::Separator::Space), move(alt_text));
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/css-display-3/#the-display-properties
 | |
| RefPtr<CSSStyleValue const> Parser::parse_display_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto parse_single_component_display = [this](TokenStream<ComponentValue>& tokens) -> Optional<Display> {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         if (auto keyword_value = parse_keyword_value(tokens)) {
 | |
|             auto keyword = keyword_value->to_keyword();
 | |
|             if (keyword == Keyword::ListItem) {
 | |
|                 transaction.commit();
 | |
|                 return Display::from_short(Display::Short::ListItem);
 | |
|             }
 | |
| 
 | |
|             if (auto display_outside = keyword_to_display_outside(keyword); display_outside.has_value()) {
 | |
|                 transaction.commit();
 | |
|                 switch (display_outside.value()) {
 | |
|                 case DisplayOutside::Block:
 | |
|                     return Display::from_short(Display::Short::Block);
 | |
|                 case DisplayOutside::Inline:
 | |
|                     return Display::from_short(Display::Short::Inline);
 | |
|                 case DisplayOutside::RunIn:
 | |
|                     return Display::from_short(Display::Short::RunIn);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (auto display_inside = keyword_to_display_inside(keyword); display_inside.has_value()) {
 | |
|                 transaction.commit();
 | |
|                 switch (display_inside.value()) {
 | |
|                 case DisplayInside::Flow:
 | |
|                     return Display::from_short(Display::Short::Flow);
 | |
|                 case DisplayInside::FlowRoot:
 | |
|                     return Display::from_short(Display::Short::FlowRoot);
 | |
|                 case DisplayInside::Table:
 | |
|                     return Display::from_short(Display::Short::Table);
 | |
|                 case DisplayInside::Flex:
 | |
|                     return Display::from_short(Display::Short::Flex);
 | |
|                 case DisplayInside::Grid:
 | |
|                     return Display::from_short(Display::Short::Grid);
 | |
|                 case DisplayInside::Ruby:
 | |
|                     return Display::from_short(Display::Short::Ruby);
 | |
|                 case DisplayInside::Math:
 | |
|                     return Display::from_short(Display::Short::Math);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (auto display_internal = keyword_to_display_internal(keyword); display_internal.has_value()) {
 | |
|                 transaction.commit();
 | |
|                 return Display { display_internal.value() };
 | |
|             }
 | |
| 
 | |
|             if (auto display_box = keyword_to_display_box(keyword); display_box.has_value()) {
 | |
|                 transaction.commit();
 | |
|                 switch (display_box.value()) {
 | |
|                 case DisplayBox::Contents:
 | |
|                     return Display::from_short(Display::Short::Contents);
 | |
|                 case DisplayBox::None:
 | |
|                     return Display::from_short(Display::Short::None);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (auto display_legacy = keyword_to_display_legacy(keyword); display_legacy.has_value()) {
 | |
|                 transaction.commit();
 | |
|                 switch (display_legacy.value()) {
 | |
|                 case DisplayLegacy::InlineBlock:
 | |
|                     return Display::from_short(Display::Short::InlineBlock);
 | |
|                 case DisplayLegacy::InlineTable:
 | |
|                     return Display::from_short(Display::Short::InlineTable);
 | |
|                 case DisplayLegacy::InlineFlex:
 | |
|                     return Display::from_short(Display::Short::InlineFlex);
 | |
|                 case DisplayLegacy::InlineGrid:
 | |
|                     return Display::from_short(Display::Short::InlineGrid);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return OptionalNone {};
 | |
|     };
 | |
| 
 | |
|     auto parse_multi_component_display = [this](TokenStream<ComponentValue>& tokens) -> Optional<Display> {
 | |
|         auto list_item = Display::ListItem::No;
 | |
|         Optional<DisplayInside> inside;
 | |
|         Optional<DisplayOutside> outside;
 | |
| 
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         while (tokens.has_next_token()) {
 | |
|             if (auto value = parse_keyword_value(tokens)) {
 | |
|                 auto keyword = value->to_keyword();
 | |
|                 if (keyword == Keyword::ListItem) {
 | |
|                     if (list_item == Display::ListItem::Yes)
 | |
|                         return {};
 | |
|                     list_item = Display::ListItem::Yes;
 | |
|                     continue;
 | |
|                 }
 | |
|                 if (auto inside_value = keyword_to_display_inside(keyword); inside_value.has_value()) {
 | |
|                     if (inside.has_value())
 | |
|                         return {};
 | |
|                     inside = inside_value.value();
 | |
|                     continue;
 | |
|                 }
 | |
|                 if (auto outside_value = keyword_to_display_outside(keyword); outside_value.has_value()) {
 | |
|                     if (outside.has_value())
 | |
|                         return {};
 | |
|                     outside = outside_value.value();
 | |
|                     continue;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Not a display value, abort.
 | |
|             dbgln_if(CSS_PARSER_DEBUG, "Unrecognized display value: `{}`", tokens.next_token().to_string());
 | |
|             return {};
 | |
|         }
 | |
| 
 | |
|         // The spec does not allow any other inside values to be combined with list-item
 | |
|         // <display-outside>? && [ flow | flow-root ]? && list-item
 | |
|         if (list_item == Display::ListItem::Yes && inside.has_value() && inside != DisplayInside::Flow && inside != DisplayInside::FlowRoot)
 | |
|             return {};
 | |
| 
 | |
|         transaction.commit();
 | |
|         return Display { outside.value_or(DisplayOutside::Block), inside.value_or(DisplayInside::Flow), list_item };
 | |
|     };
 | |
| 
 | |
|     Optional<Display> display;
 | |
|     if (tokens.remaining_token_count() == 1)
 | |
|         display = parse_single_component_display(tokens);
 | |
|     else
 | |
|         display = parse_multi_component_display(tokens);
 | |
| 
 | |
|     if (display.has_value())
 | |
|         return DisplayStyleValue::create(display.value());
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_flex_shorthand_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto make_flex_shorthand = [&](NonnullRefPtr<CSSStyleValue const> flex_grow, NonnullRefPtr<CSSStyleValue const> flex_shrink, NonnullRefPtr<CSSStyleValue const> flex_basis) {
 | |
|         transaction.commit();
 | |
|         return ShorthandStyleValue::create(PropertyID::Flex,
 | |
|             { PropertyID::FlexGrow, PropertyID::FlexShrink, PropertyID::FlexBasis },
 | |
|             { move(flex_grow), move(flex_shrink), move(flex_basis) });
 | |
|     };
 | |
| 
 | |
|     if (tokens.remaining_token_count() == 1) {
 | |
|         // One-value syntax: <flex-grow> | <flex-basis> | none
 | |
|         auto properties = Array { PropertyID::FlexGrow, PropertyID::FlexBasis, PropertyID::Flex };
 | |
|         auto property_and_value = parse_css_value_for_properties(properties, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
| 
 | |
|         auto& value = property_and_value->style_value;
 | |
|         switch (property_and_value->property) {
 | |
|         case PropertyID::FlexGrow: {
 | |
|             // NOTE: The spec says that flex-basis should be 0 here, but other engines currently use 0%.
 | |
|             // https://github.com/w3c/csswg-drafts/issues/5742
 | |
|             auto flex_basis = PercentageStyleValue::create(Percentage(0));
 | |
|             auto one = NumberStyleValue::create(1);
 | |
|             return make_flex_shorthand(*value, one, flex_basis);
 | |
|         }
 | |
|         case PropertyID::FlexBasis: {
 | |
|             auto one = NumberStyleValue::create(1);
 | |
|             return make_flex_shorthand(one, one, *value);
 | |
|         }
 | |
|         case PropertyID::Flex: {
 | |
|             if (value->is_keyword() && value->to_keyword() == Keyword::None) {
 | |
|                 auto zero = NumberStyleValue::create(0);
 | |
|                 return make_flex_shorthand(zero, zero, CSSKeywordValue::create(Keyword::Auto));
 | |
|             }
 | |
|             break;
 | |
|         }
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     RefPtr<CSSStyleValue const> flex_grow;
 | |
|     RefPtr<CSSStyleValue const> flex_shrink;
 | |
|     RefPtr<CSSStyleValue const> flex_basis;
 | |
| 
 | |
|     // NOTE: FlexGrow has to be before FlexBasis. `0` is a valid FlexBasis, but only
 | |
|     //       if FlexGrow (along with optional FlexShrink) have already been specified.
 | |
|     auto remaining_longhands = Vector { PropertyID::FlexGrow, PropertyID::FlexBasis };
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto property_and_value = parse_css_value_for_properties(remaining_longhands, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = property_and_value->style_value;
 | |
|         remove_property(remaining_longhands, property_and_value->property);
 | |
| 
 | |
|         switch (property_and_value->property) {
 | |
|         case PropertyID::FlexGrow: {
 | |
|             VERIFY(!flex_grow);
 | |
|             flex_grow = value.release_nonnull();
 | |
| 
 | |
|             // Flex-shrink may optionally follow directly after.
 | |
|             auto maybe_flex_shrink = parse_css_value_for_property(PropertyID::FlexShrink, tokens);
 | |
|             if (maybe_flex_shrink)
 | |
|                 flex_shrink = maybe_flex_shrink.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::FlexBasis: {
 | |
|             VERIFY(!flex_basis);
 | |
|             flex_basis = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!flex_grow)
 | |
|         flex_grow = property_initial_value(PropertyID::FlexGrow);
 | |
|     if (!flex_shrink)
 | |
|         flex_shrink = property_initial_value(PropertyID::FlexShrink);
 | |
|     if (!flex_basis) {
 | |
|         // NOTE: The spec says that flex-basis should be 0 here, but other engines currently use 0%.
 | |
|         // https://github.com/w3c/csswg-drafts/issues/5742
 | |
|         flex_basis = PercentageStyleValue::create(Percentage(0));
 | |
|     }
 | |
| 
 | |
|     return make_flex_shorthand(flex_grow.release_nonnull(), flex_shrink.release_nonnull(), flex_basis.release_nonnull());
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_flex_flow_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     RefPtr<CSSStyleValue const> flex_direction;
 | |
|     RefPtr<CSSStyleValue const> flex_wrap;
 | |
| 
 | |
|     auto remaining_longhands = Vector { PropertyID::FlexDirection, PropertyID::FlexWrap };
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto property_and_value = parse_css_value_for_properties(remaining_longhands, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = property_and_value->style_value;
 | |
|         remove_property(remaining_longhands, property_and_value->property);
 | |
| 
 | |
|         switch (property_and_value->property) {
 | |
|         case PropertyID::FlexDirection:
 | |
|             VERIFY(!flex_direction);
 | |
|             flex_direction = value.release_nonnull();
 | |
|             continue;
 | |
|         case PropertyID::FlexWrap:
 | |
|             VERIFY(!flex_wrap);
 | |
|             flex_wrap = value.release_nonnull();
 | |
|             continue;
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!flex_direction)
 | |
|         flex_direction = property_initial_value(PropertyID::FlexDirection);
 | |
|     if (!flex_wrap)
 | |
|         flex_wrap = property_initial_value(PropertyID::FlexWrap);
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::FlexFlow,
 | |
|         { PropertyID::FlexDirection, PropertyID::FlexWrap },
 | |
|         { flex_direction.release_nonnull(), flex_wrap.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     RefPtr<CSSStyleValue const> font_width;
 | |
|     RefPtr<CSSStyleValue const> font_style;
 | |
|     RefPtr<CSSStyleValue const> font_weight;
 | |
|     RefPtr<CSSStyleValue const> font_size;
 | |
|     RefPtr<CSSStyleValue const> line_height;
 | |
|     RefPtr<CSSStyleValue const> font_families;
 | |
|     RefPtr<CSSStyleValue const> font_variant;
 | |
| 
 | |
|     // FIXME: Handle system fonts. (caption, icon, menu, message-box, small-caption, status-bar)
 | |
| 
 | |
|     // Several sub-properties can be "normal", and appear in any order: style, variant, weight, stretch
 | |
|     // So, we have to handle that separately.
 | |
|     int normal_count = 0;
 | |
| 
 | |
|     // FIXME: `font-variant` allows a lot of different values which aren't allowed in the `font` shorthand.
 | |
|     // FIXME: `font-width` allows <percentage> values, which aren't allowed in the `font` shorthand.
 | |
|     auto remaining_longhands = Vector { PropertyID::FontSize, PropertyID::FontStyle, PropertyID::FontVariant, PropertyID::FontWeight, PropertyID::FontWidth };
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto& peek_token = tokens.next_token();
 | |
|         if (peek_token.is_ident("normal"sv)) {
 | |
|             normal_count++;
 | |
|             tokens.discard_a_token();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         auto property_and_value = parse_css_value_for_properties(remaining_longhands, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = property_and_value->style_value;
 | |
|         remove_property(remaining_longhands, property_and_value->property);
 | |
| 
 | |
|         switch (property_and_value->property) {
 | |
|         case PropertyID::FontSize: {
 | |
|             VERIFY(!font_size);
 | |
|             font_size = value.release_nonnull();
 | |
| 
 | |
|             // Consume `/ line-height` if present
 | |
|             if (tokens.next_token().is_delim('/')) {
 | |
|                 tokens.discard_a_token();
 | |
|                 auto maybe_line_height = parse_css_value_for_property(PropertyID::LineHeight, tokens);
 | |
|                 if (!maybe_line_height)
 | |
|                     return nullptr;
 | |
|                 line_height = maybe_line_height.release_nonnull();
 | |
|             }
 | |
| 
 | |
|             // Consume font-families
 | |
|             auto maybe_font_families = parse_font_family_value(tokens);
 | |
|             // font-family comes last, so we must not have any tokens left over.
 | |
|             if (!maybe_font_families || tokens.has_next_token())
 | |
|                 return nullptr;
 | |
|             font_families = maybe_font_families.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::FontWidth: {
 | |
|             VERIFY(!font_width);
 | |
|             font_width = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::FontStyle: {
 | |
|             VERIFY(!font_style);
 | |
|             font_style = FontStyleStyleValue::create(*keyword_to_font_style(value.release_nonnull()->to_keyword()));
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::FontVariant: {
 | |
|             VERIFY(!font_variant);
 | |
|             font_variant = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::FontWeight: {
 | |
|             VERIFY(!font_weight);
 | |
|             font_weight = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     // Since normal is the default value for all the properties that can have it, we don't have to actually
 | |
|     // set anything to normal here. It'll be set when we create the ShorthandStyleValue below.
 | |
|     // We just need to make sure we were not given more normals than will fit.
 | |
|     int unset_value_count = (font_style ? 0 : 1) + (font_weight ? 0 : 1) + (font_variant ? 0 : 1) + (font_width ? 0 : 1);
 | |
|     if (unset_value_count < normal_count)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!font_size || !font_families)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!font_style)
 | |
|         font_style = property_initial_value(PropertyID::FontStyle);
 | |
|     if (!font_variant)
 | |
|         font_variant = property_initial_value(PropertyID::FontVariant);
 | |
|     if (!font_weight)
 | |
|         font_weight = property_initial_value(PropertyID::FontWeight);
 | |
|     if (!font_width)
 | |
|         font_width = property_initial_value(PropertyID::FontWidth);
 | |
|     if (!line_height)
 | |
|         line_height = property_initial_value(PropertyID::LineHeight);
 | |
| 
 | |
|     transaction.commit();
 | |
|     auto initial_value = CSSKeywordValue::create(Keyword::Initial);
 | |
|     return ShorthandStyleValue::create(PropertyID::Font,
 | |
|         {
 | |
|             // Set explicitly https://drafts.csswg.org/css-fonts/#set-explicitly
 | |
|             PropertyID::FontFamily,
 | |
|             PropertyID::FontSize,
 | |
|             PropertyID::FontWidth,
 | |
|             // FIXME: PropertyID::FontStretch
 | |
|             PropertyID::FontStyle,
 | |
|             PropertyID::FontVariant,
 | |
|             PropertyID::FontWeight,
 | |
|             PropertyID::LineHeight,
 | |
| 
 | |
|             // Reset implicitly https://drafts.csswg.org/css-fonts/#reset-implicitly
 | |
|             PropertyID::FontFeatureSettings,
 | |
|             // FIXME: PropertyID::FontKerning,
 | |
|             PropertyID::FontLanguageOverride,
 | |
|             // FIXME: PropertyID::FontOpticalSizing,
 | |
|             // FIXME: PropertyID::FontSizeAdjust,
 | |
|             PropertyID::FontVariantAlternates,
 | |
|             PropertyID::FontVariantCaps,
 | |
|             PropertyID::FontVariantEastAsian,
 | |
|             PropertyID::FontVariantEmoji,
 | |
|             PropertyID::FontVariantLigatures,
 | |
|             PropertyID::FontVariantNumeric,
 | |
|             PropertyID::FontVariantPosition,
 | |
|             PropertyID::FontVariationSettings,
 | |
|         },
 | |
|         {
 | |
|             // Set explicitly
 | |
|             font_families.release_nonnull(),
 | |
|             font_size.release_nonnull(),
 | |
|             font_width.release_nonnull(),
 | |
|             // FIXME: font-stretch
 | |
|             font_style.release_nonnull(),
 | |
|             font_variant.release_nonnull(),
 | |
|             font_weight.release_nonnull(),
 | |
|             line_height.release_nonnull(),
 | |
| 
 | |
|             // Reset implicitly
 | |
|             initial_value, // font-feature-settings
 | |
|                            // FIXME: font-kerning,
 | |
|             initial_value, // font-language-override
 | |
|                            // FIXME: font-optical-sizing,
 | |
|                            // FIXME: font-size-adjust,
 | |
|             initial_value, // font-variant-alternates
 | |
|             initial_value, // font-variant-caps
 | |
|             initial_value, // font-variant-east-asian
 | |
|             initial_value, // font-variant-emoji
 | |
|             initial_value, // font-variant-ligatures
 | |
|             initial_value, // font-variant-numeric
 | |
|             initial_value, // font-variant-position
 | |
|             initial_value, // font-variation-settings
 | |
|         });
 | |
| }
 | |
| 
 | |
| // https://drafts.csswg.org/css-fonts-4/#font-family-prop
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_family_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // [ <family-name> | <generic-family> ]#
 | |
|     // FIXME: We currently require font-family to always be a list, even with one item.
 | |
|     //        Maybe change that?
 | |
|     auto result = parse_comma_separated_value_list(tokens, [this](auto& inner_tokens) -> RefPtr<CSSStyleValue const> {
 | |
|         inner_tokens.discard_whitespace();
 | |
| 
 | |
|         // <generic-family>
 | |
|         if (inner_tokens.next_token().is(Token::Type::Ident)) {
 | |
|             auto maybe_keyword = keyword_from_string(inner_tokens.next_token().token().ident());
 | |
|             if (maybe_keyword.has_value() && keyword_to_generic_font_family(maybe_keyword.value()).has_value()) {
 | |
|                 inner_tokens.discard_a_token(); // Ident
 | |
|                 inner_tokens.discard_whitespace();
 | |
|                 return CSSKeywordValue::create(maybe_keyword.value());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // <family-name>
 | |
|         return parse_family_name_value(inner_tokens);
 | |
|     });
 | |
| 
 | |
|     if (!result)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (result->is_value_list())
 | |
|         return result.release_nonnull();
 | |
| 
 | |
|     // It's a single value, so wrap it in a list - see FIXME above.
 | |
|     return StyleValueList::create(StyleValueVector { result.release_nonnull() }, StyleValueList::Separator::Comma);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_language_override_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://drafts.csswg.org/css-fonts/#propdef-font-language-override
 | |
|     // This is `normal | <string>` but with the constraint that the string has to be 4 characters long:
 | |
|     // Shorter strings are right-padded with spaces, and longer strings are invalid.
 | |
| 
 | |
|     if (auto normal = parse_all_as_single_keyword_value(tokens, Keyword::Normal))
 | |
|         return normal;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     tokens.discard_whitespace();
 | |
|     if (auto string = parse_string_value(tokens)) {
 | |
|         auto string_value = string->string_value();
 | |
|         tokens.discard_whitespace();
 | |
|         if (tokens.has_next_token()) {
 | |
|             dbgln_if(CSS_PARSER_DEBUG, "CSSParser: Failed to parse font-language-override: unexpected trailing tokens");
 | |
|             return nullptr;
 | |
|         }
 | |
|         auto length = string_value.code_points().length();
 | |
|         if (length > 4) {
 | |
|             dbgln_if(CSS_PARSER_DEBUG, "CSSParser: Failed to parse font-language-override: <string> value \"{}\" is too long", string_value);
 | |
|             return nullptr;
 | |
|         }
 | |
|         transaction.commit();
 | |
|         if (length < 4)
 | |
|             return StringStyleValue::create(MUST(String::formatted("{:<4}", string_value)));
 | |
|         return string;
 | |
|     }
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_feature_settings_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://drafts.csswg.org/css-fonts/#propdef-font-feature-settings
 | |
|     // normal | <feature-tag-value>#
 | |
| 
 | |
|     // normal
 | |
|     if (auto normal = parse_all_as_single_keyword_value(tokens, Keyword::Normal))
 | |
|         return normal;
 | |
| 
 | |
|     // <feature-tag-value>#
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto tag_values = parse_a_comma_separated_list_of_component_values(tokens);
 | |
| 
 | |
|     // "The computed value of font-feature-settings is a map, so any duplicates in the specified value must not be preserved.
 | |
|     // If the same feature tag appears more than once, the value associated with the last appearance supersedes any previous
 | |
|     // value for that axis."
 | |
|     // So, we deduplicate them here using a HashSet.
 | |
| 
 | |
|     OrderedHashMap<FlyString, NonnullRefPtr<OpenTypeTaggedStyleValue const>> feature_tags_map;
 | |
|     for (auto const& values : tag_values) {
 | |
|         // <feature-tag-value> = <opentype-tag> [ <integer [0,∞]> | on | off ]?
 | |
|         TokenStream tag_tokens { values };
 | |
|         tag_tokens.discard_whitespace();
 | |
|         auto opentype_tag = parse_opentype_tag_value(tag_tokens);
 | |
|         tag_tokens.discard_whitespace();
 | |
|         RefPtr<CSSStyleValue const> value;
 | |
|         if (tag_tokens.has_next_token()) {
 | |
|             if (auto integer = parse_integer_value(tag_tokens)) {
 | |
|                 if (integer->is_integer() && integer->as_integer().value() < 0)
 | |
|                     return nullptr;
 | |
|                 value = integer;
 | |
|             } else {
 | |
|                 // A value of on is synonymous with 1 and off is synonymous with 0.
 | |
|                 auto keyword = parse_keyword_value(tag_tokens);
 | |
|                 if (!keyword)
 | |
|                     return nullptr;
 | |
|                 switch (keyword->to_keyword()) {
 | |
|                 case Keyword::On:
 | |
|                     value = IntegerStyleValue::create(1);
 | |
|                     break;
 | |
|                 case Keyword::Off:
 | |
|                     value = IntegerStyleValue::create(0);
 | |
|                     break;
 | |
|                 default:
 | |
|                     return nullptr;
 | |
|                 }
 | |
|             }
 | |
|             tag_tokens.discard_whitespace();
 | |
|         } else {
 | |
|             // "If the value is omitted, a value of 1 is assumed."
 | |
|             value = IntegerStyleValue::create(1);
 | |
|         }
 | |
| 
 | |
|         if (!opentype_tag || !value || tag_tokens.has_next_token())
 | |
|             return nullptr;
 | |
| 
 | |
|         feature_tags_map.set(opentype_tag->string_value(), OpenTypeTaggedStyleValue::create(OpenTypeTaggedStyleValue::Mode::FontFeatureSettings, opentype_tag->string_value(), value.release_nonnull()));
 | |
|     }
 | |
| 
 | |
|     // "The computed value contains the de-duplicated feature tags, sorted in ascending order by code unit."
 | |
|     StyleValueVector feature_tags;
 | |
|     feature_tags.ensure_capacity(feature_tags_map.size());
 | |
|     for (auto const& [key, feature_tag] : feature_tags_map)
 | |
|         feature_tags.append(feature_tag);
 | |
| 
 | |
|     quick_sort(feature_tags, [](auto& a, auto& b) {
 | |
|         return a->as_open_type_tagged().tag() < b->as_open_type_tagged().tag();
 | |
|     });
 | |
| 
 | |
|     transaction.commit();
 | |
|     return StyleValueList::create(move(feature_tags), StyleValueList::Separator::Comma);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_style_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://drafts.csswg.org/css-fonts/#font-style-prop
 | |
|     // normal | italic | left | right | oblique <angle [-90deg,90deg]>?
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto keyword_value = parse_css_value_for_property(PropertyID::FontStyle, tokens);
 | |
|     if (!keyword_value)
 | |
|         return nullptr;
 | |
|     auto font_style = keyword_to_font_style(keyword_value->to_keyword());
 | |
|     VERIFY(font_style.has_value());
 | |
|     if (tokens.has_next_token() && keyword_value->to_keyword() == Keyword::Oblique) {
 | |
|         if (auto angle_value = parse_angle_value(tokens)) {
 | |
|             if (angle_value->is_angle()) {
 | |
|                 auto angle = angle_value->as_angle().angle();
 | |
|                 auto angle_degrees = angle.to_degrees();
 | |
|                 if (angle_degrees < -90 || angle_degrees > 90)
 | |
|                     return nullptr;
 | |
|             }
 | |
| 
 | |
|             transaction.commit();
 | |
|             return FontStyleStyleValue::create(*font_style, angle_value);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     transaction.commit();
 | |
|     return FontStyleStyleValue::create(*font_style);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_variation_settings_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://drafts.csswg.org/css-fonts/#propdef-font-variation-settings
 | |
|     // normal | [ <opentype-tag> <number>]#
 | |
| 
 | |
|     // normal
 | |
|     if (auto normal = parse_all_as_single_keyword_value(tokens, Keyword::Normal))
 | |
|         return normal;
 | |
| 
 | |
|     // [ <opentype-tag> <number>]#
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto tag_values = parse_a_comma_separated_list_of_component_values(tokens);
 | |
| 
 | |
|     // "If the same axis name appears more than once, the value associated with the last appearance supersedes any
 | |
|     // previous value for that axis. This deduplication is observable by accessing the computed value of this property."
 | |
|     // So, we deduplicate them here using a HashSet.
 | |
| 
 | |
|     OrderedHashMap<FlyString, NonnullRefPtr<OpenTypeTaggedStyleValue const>> axis_tags_map;
 | |
|     for (auto const& values : tag_values) {
 | |
|         TokenStream tag_tokens { values };
 | |
|         tag_tokens.discard_whitespace();
 | |
|         auto opentype_tag = parse_opentype_tag_value(tag_tokens);
 | |
|         tag_tokens.discard_whitespace();
 | |
|         auto number = parse_number_value(tag_tokens);
 | |
|         tag_tokens.discard_whitespace();
 | |
| 
 | |
|         if (!opentype_tag || !number || tag_tokens.has_next_token())
 | |
|             return nullptr;
 | |
| 
 | |
|         axis_tags_map.set(opentype_tag->string_value(), OpenTypeTaggedStyleValue::create(OpenTypeTaggedStyleValue::Mode::FontVariationSettings, opentype_tag->string_value(), number.release_nonnull()));
 | |
|     }
 | |
| 
 | |
|     // "The computed value contains the de-duplicated axis names, sorted in ascending order by code unit."
 | |
|     StyleValueVector axis_tags;
 | |
|     axis_tags.ensure_capacity(axis_tags_map.size());
 | |
|     for (auto const& [key, axis_tag] : axis_tags_map)
 | |
|         axis_tags.append(axis_tag);
 | |
| 
 | |
|     quick_sort(axis_tags, [](auto& a, auto& b) {
 | |
|         return a->as_open_type_tagged().tag() < b->as_open_type_tagged().tag();
 | |
|     });
 | |
| 
 | |
|     transaction.commit();
 | |
|     return StyleValueList::create(move(axis_tags), StyleValueList::Separator::Comma);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_variant(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // 6.11 https://drafts.csswg.org/css-fonts/#propdef-font-variant
 | |
|     // normal | none |
 | |
|     // [ [ <common-lig-values> || <discretionary-lig-values> || <historical-lig-values> || <contextual-alt-values> ]
 | |
|     // || [ small-caps | all-small-caps | petite-caps | all-petite-caps | unicase | titling-caps ] ||
 | |
|     // [ FIXME: stylistic(<feature-value-name>) ||
 | |
|     // historical-forms ||
 | |
|     // FIXME: styleset(<feature-value-name>#) ||
 | |
|     // FIXME: character-variant(<feature-value-name>#) ||
 | |
|     // FIXME: swash(<feature-value-name>) ||
 | |
|     // FIXME: ornaments(<feature-value-name>) ||
 | |
|     // FIXME: annotation(<feature-value-name>) ] ||
 | |
|     // [ <numeric-figure-values> || <numeric-spacing-values> || <numeric-fraction-values> ||
 | |
|     // ordinal || slashed-zero ] || [ <east-asian-variant-values> || <east-asian-width-values> || ruby ] ||
 | |
|     // [ sub | super ] || [ text | emoji | unicode ] ]
 | |
| 
 | |
|     bool has_common_ligatures = false;
 | |
|     bool has_discretionary_ligatures = false;
 | |
|     bool has_historical_ligatures = false;
 | |
|     bool has_contextual = false;
 | |
|     bool has_numeric_figures = false;
 | |
|     bool has_numeric_spacing = false;
 | |
|     bool has_numeric_fractions = false;
 | |
|     bool has_numeric_ordinals = false;
 | |
|     bool has_numeric_slashed_zero = false;
 | |
|     bool has_east_asian_variant = false;
 | |
|     bool has_east_asian_width = false;
 | |
|     bool has_east_asian_ruby = false;
 | |
|     RefPtr<CSSStyleValue const> alternates_value {};
 | |
|     RefPtr<CSSStyleValue const> caps_value {};
 | |
|     RefPtr<CSSStyleValue const> emoji_value {};
 | |
|     RefPtr<CSSStyleValue const> position_value {};
 | |
|     StyleValueVector east_asian_values;
 | |
|     StyleValueVector ligatures_values;
 | |
|     StyleValueVector numeric_values;
 | |
| 
 | |
|     if (auto parsed_value = parse_all_as_single_keyword_value(tokens, Keyword::Normal)) {
 | |
|         // normal, do nothing
 | |
|     } else if (auto parsed_value = parse_all_as_single_keyword_value(tokens, Keyword::None)) {
 | |
|         // none
 | |
|         ligatures_values.append(parsed_value.release_nonnull());
 | |
|     } else {
 | |
| 
 | |
|         while (tokens.has_next_token()) {
 | |
|             auto maybe_value = parse_keyword_value(tokens);
 | |
|             if (!maybe_value)
 | |
|                 break;
 | |
|             auto value = maybe_value.release_nonnull();
 | |
|             if (!value->is_keyword()) {
 | |
|                 // FIXME: alternate functions such as stylistic()
 | |
|                 return nullptr;
 | |
|             }
 | |
|             auto keyword = value->to_keyword();
 | |
| 
 | |
|             switch (keyword) {
 | |
|             // <common-lig-values>       = [ common-ligatures | no-common-ligatures ]
 | |
|             case Keyword::CommonLigatures:
 | |
|             case Keyword::NoCommonLigatures:
 | |
|                 if (has_common_ligatures)
 | |
|                     return nullptr;
 | |
|                 ligatures_values.append(move(value));
 | |
|                 has_common_ligatures = true;
 | |
|                 break;
 | |
|             // <discretionary-lig-values> = [ discretionary-ligatures | no-discretionary-ligatures ]
 | |
|             case Keyword::DiscretionaryLigatures:
 | |
|             case Keyword::NoDiscretionaryLigatures:
 | |
|                 if (has_discretionary_ligatures)
 | |
|                     return nullptr;
 | |
|                 ligatures_values.append(move(value));
 | |
|                 has_discretionary_ligatures = true;
 | |
|                 break;
 | |
|             // <historical-lig-values>   = [ historical-ligatures | no-historical-ligatures ]
 | |
|             case Keyword::HistoricalLigatures:
 | |
|             case Keyword::NoHistoricalLigatures:
 | |
|                 if (has_historical_ligatures)
 | |
|                     return nullptr;
 | |
|                 ligatures_values.append(move(value));
 | |
|                 has_historical_ligatures = true;
 | |
|                 break;
 | |
|             // <contextual-alt-values>   = [ contextual | no-contextual ]
 | |
|             case Keyword::Contextual:
 | |
|             case Keyword::NoContextual:
 | |
|                 if (has_contextual)
 | |
|                     return nullptr;
 | |
|                 ligatures_values.append(move(value));
 | |
|                 has_contextual = true;
 | |
|                 break;
 | |
|             // historical-forms
 | |
|             case Keyword::HistoricalForms:
 | |
|                 if (alternates_value != nullptr)
 | |
|                     return nullptr;
 | |
|                 alternates_value = value.ptr();
 | |
|                 break;
 | |
|             // [ small-caps | all-small-caps | petite-caps | all-petite-caps | unicase | titling-caps ]
 | |
|             case Keyword::SmallCaps:
 | |
|             case Keyword::AllSmallCaps:
 | |
|             case Keyword::PetiteCaps:
 | |
|             case Keyword::AllPetiteCaps:
 | |
|             case Keyword::Unicase:
 | |
|             case Keyword::TitlingCaps:
 | |
|                 if (caps_value != nullptr)
 | |
|                     return nullptr;
 | |
|                 caps_value = value.ptr();
 | |
|                 break;
 | |
|             // <numeric-figure-values>       = [ lining-nums | oldstyle-nums ]
 | |
|             case Keyword::LiningNums:
 | |
|             case Keyword::OldstyleNums:
 | |
|                 if (has_numeric_figures)
 | |
|                     return nullptr;
 | |
|                 numeric_values.append(move(value));
 | |
|                 has_numeric_figures = true;
 | |
|                 break;
 | |
|             // <numeric-spacing-values>      = [ proportional-nums | tabular-nums ]
 | |
|             case Keyword::ProportionalNums:
 | |
|             case Keyword::TabularNums:
 | |
|                 if (has_numeric_spacing)
 | |
|                     return nullptr;
 | |
|                 numeric_values.append(move(value));
 | |
|                 has_numeric_spacing = true;
 | |
|                 break;
 | |
|             // <numeric-fraction-values>     = [ diagonal-fractions | stacked-fractions]
 | |
|             case Keyword::DiagonalFractions:
 | |
|             case Keyword::StackedFractions:
 | |
|                 if (has_numeric_fractions)
 | |
|                     return nullptr;
 | |
|                 numeric_values.append(move(value));
 | |
|                 has_numeric_fractions = true;
 | |
|                 break;
 | |
|             // ordinal
 | |
|             case Keyword::Ordinal:
 | |
|                 if (has_numeric_ordinals)
 | |
|                     return nullptr;
 | |
|                 numeric_values.append(move(value));
 | |
|                 has_numeric_ordinals = true;
 | |
|                 break;
 | |
|             case Keyword::SlashedZero:
 | |
|                 if (has_numeric_slashed_zero)
 | |
|                     return nullptr;
 | |
|                 numeric_values.append(move(value));
 | |
|                 has_numeric_slashed_zero = true;
 | |
|                 break;
 | |
|             // <east-asian-variant-values> = [ jis78 | jis83 | jis90 | jis04 | simplified | traditional ]
 | |
|             case Keyword::Jis78:
 | |
|             case Keyword::Jis83:
 | |
|             case Keyword::Jis90:
 | |
|             case Keyword::Jis04:
 | |
|             case Keyword::Simplified:
 | |
|             case Keyword::Traditional:
 | |
|                 if (has_east_asian_variant)
 | |
|                     return nullptr;
 | |
|                 east_asian_values.append(move(value));
 | |
|                 has_east_asian_variant = true;
 | |
|                 break;
 | |
|             // <east-asian-width-values>   = [ full-width | proportional-width ]
 | |
|             case Keyword::FullWidth:
 | |
|             case Keyword::ProportionalWidth:
 | |
|                 if (has_east_asian_width)
 | |
|                     return nullptr;
 | |
|                 east_asian_values.append(move(value));
 | |
|                 has_east_asian_width = true;
 | |
|                 break;
 | |
|             // ruby
 | |
|             case Keyword::Ruby:
 | |
|                 if (has_east_asian_ruby)
 | |
|                     return nullptr;
 | |
|                 east_asian_values.append(move(value));
 | |
|                 has_east_asian_ruby = true;
 | |
|                 break;
 | |
|             // text | emoji | unicode
 | |
|             case Keyword::Text:
 | |
|             case Keyword::Emoji:
 | |
|             case Keyword::Unicode:
 | |
|                 if (emoji_value != nullptr)
 | |
|                     return nullptr;
 | |
|                 emoji_value = value.ptr();
 | |
|                 break;
 | |
|             // sub | super
 | |
|             case Keyword::Sub:
 | |
|             case Keyword::Super:
 | |
|                 if (position_value != nullptr)
 | |
|                     return nullptr;
 | |
|                 position_value = value.ptr();
 | |
|                 break;
 | |
|             default:
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     auto normal_value = CSSKeywordValue::create(Keyword::Normal);
 | |
|     auto resolve_list = [&normal_value](StyleValueVector values) -> NonnullRefPtr<CSSStyleValue const> {
 | |
|         if (values.is_empty())
 | |
|             return normal_value;
 | |
|         if (values.size() == 1)
 | |
|             return *values.first();
 | |
|         return StyleValueList::create(move(values), StyleValueList::Separator::Space);
 | |
|     };
 | |
| 
 | |
|     if (!alternates_value)
 | |
|         alternates_value = normal_value;
 | |
|     if (!caps_value)
 | |
|         caps_value = normal_value;
 | |
|     if (!emoji_value)
 | |
|         emoji_value = normal_value;
 | |
|     if (!position_value)
 | |
|         position_value = normal_value;
 | |
| 
 | |
|     quick_sort(east_asian_values, [](auto& left, auto& right) { return *keyword_to_font_variant_east_asian(left->to_keyword()) < *keyword_to_font_variant_east_asian(right->to_keyword()); });
 | |
|     auto east_asian_value = resolve_list(east_asian_values);
 | |
| 
 | |
|     quick_sort(ligatures_values, [](auto& left, auto& right) { return *keyword_to_font_variant_ligatures(left->to_keyword()) < *keyword_to_font_variant_ligatures(right->to_keyword()); });
 | |
|     auto ligatures_value = resolve_list(ligatures_values);
 | |
| 
 | |
|     quick_sort(numeric_values, [](auto& left, auto& right) { return *keyword_to_font_variant_numeric(left->to_keyword()) < *keyword_to_font_variant_numeric(right->to_keyword()); });
 | |
|     auto numeric_value = resolve_list(numeric_values);
 | |
| 
 | |
|     return ShorthandStyleValue::create(PropertyID::FontVariant,
 | |
|         { PropertyID::FontVariantAlternates,
 | |
|             PropertyID::FontVariantCaps,
 | |
|             PropertyID::FontVariantEastAsian,
 | |
|             PropertyID::FontVariantEmoji,
 | |
|             PropertyID::FontVariantLigatures,
 | |
|             PropertyID::FontVariantNumeric,
 | |
|             PropertyID::FontVariantPosition },
 | |
|         {
 | |
|             alternates_value.release_nonnull(),
 | |
|             caps_value.release_nonnull(),
 | |
|             move(east_asian_value),
 | |
|             emoji_value.release_nonnull(),
 | |
|             move(ligatures_value),
 | |
|             move(numeric_value),
 | |
|             position_value.release_nonnull(),
 | |
|         });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_variant_alternates_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // 6.8 https://drafts.csswg.org/css-fonts/#font-variant-alternates-prop
 | |
|     // normal |
 | |
|     // [ FIXME: stylistic(<feature-value-name>) ||
 | |
|     //   historical-forms ||
 | |
|     //   FIXME: styleset(<feature-value-name>#) ||
 | |
|     //   FIXME: character-variant(<feature-value-name>#) ||
 | |
|     //   FIXME: swash(<feature-value-name>) ||
 | |
|     //   FIXME: ornaments(<feature-value-name>) ||
 | |
|     //   FIXME: annotation(<feature-value-name>) ]
 | |
| 
 | |
|     // normal
 | |
|     if (auto normal = parse_all_as_single_keyword_value(tokens, Keyword::Normal))
 | |
|         return normal;
 | |
| 
 | |
|     // historical-forms
 | |
|     // FIXME: Support this together with other values when we parse them.
 | |
|     if (auto historical_forms = parse_all_as_single_keyword_value(tokens, Keyword::HistoricalForms))
 | |
|         return historical_forms;
 | |
| 
 | |
|     dbgln_if(CSS_PARSER_DEBUG, "CSSParser: @font-variant-alternate: parsing {} not implemented.", tokens.next_token().to_debug_string());
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_variant_east_asian_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // 6.10 https://drafts.csswg.org/css-fonts/#propdef-font-variant-east-asian
 | |
|     // normal | [ <east-asian-variant-values> || <east-asian-width-values> || ruby ]
 | |
|     // <east-asian-variant-values> = [ jis78 | jis83 | jis90 | jis04 | simplified | traditional ]
 | |
|     // <east-asian-width-values>   = [ full-width | proportional-width ]
 | |
| 
 | |
|     // normal
 | |
|     if (auto normal = parse_all_as_single_keyword_value(tokens, Keyword::Normal))
 | |
|         return normal.release_nonnull();
 | |
| 
 | |
|     // [ <east-asian-variant-values> || <east-asian-width-values> || ruby ]
 | |
|     RefPtr<CSSStyleValue const> ruby_value;
 | |
|     RefPtr<CSSStyleValue const> variant_value;
 | |
|     RefPtr<CSSStyleValue const> width_value;
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto maybe_value = parse_keyword_value(tokens);
 | |
|         if (!maybe_value)
 | |
|             break;
 | |
|         auto font_variant_east_asian = keyword_to_font_variant_east_asian(maybe_value->to_keyword());
 | |
|         if (!font_variant_east_asian.has_value())
 | |
|             return nullptr;
 | |
| 
 | |
|         switch (font_variant_east_asian.value()) {
 | |
|         case FontVariantEastAsian::Ruby:
 | |
|             if (ruby_value)
 | |
|                 return nullptr;
 | |
|             ruby_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         case FontVariantEastAsian::FullWidth:
 | |
|         case FontVariantEastAsian::ProportionalWidth:
 | |
|             if (width_value)
 | |
|                 return nullptr;
 | |
|             width_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         case FontVariantEastAsian::Jis78:
 | |
|         case FontVariantEastAsian::Jis83:
 | |
|         case FontVariantEastAsian::Jis90:
 | |
|         case FontVariantEastAsian::Jis04:
 | |
|         case FontVariantEastAsian::Simplified:
 | |
|         case FontVariantEastAsian::Traditional:
 | |
|             if (variant_value)
 | |
|                 return nullptr;
 | |
|             variant_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         case FontVariantEastAsian::Normal:
 | |
|             return nullptr;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     StyleValueVector values;
 | |
|     if (variant_value)
 | |
|         values.append(variant_value.release_nonnull());
 | |
|     if (width_value)
 | |
|         values.append(width_value.release_nonnull());
 | |
|     if (ruby_value)
 | |
|         values.append(ruby_value.release_nonnull());
 | |
| 
 | |
|     if (values.is_empty())
 | |
|         return nullptr;
 | |
|     if (values.size() == 1)
 | |
|         return *values.first();
 | |
| 
 | |
|     return StyleValueList::create(move(values), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_variant_ligatures_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // 6.4 https://drafts.csswg.org/css-fonts/#propdef-font-variant-ligatures
 | |
|     // normal | none | [ <common-lig-values> || <discretionary-lig-values> || <historical-lig-values> || <contextual-alt-values> ]
 | |
|     // <common-lig-values>       = [ common-ligatures | no-common-ligatures ]
 | |
|     // <discretionary-lig-values> = [ discretionary-ligatures | no-discretionary-ligatures ]
 | |
|     // <historical-lig-values>   = [ historical-ligatures | no-historical-ligatures ]
 | |
|     // <contextual-alt-values>   = [ contextual | no-contextual ]
 | |
| 
 | |
|     // normal
 | |
|     if (auto normal = parse_all_as_single_keyword_value(tokens, Keyword::Normal))
 | |
|         return normal.release_nonnull();
 | |
| 
 | |
|     // none
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none.release_nonnull();
 | |
| 
 | |
|     // [ <common-lig-values> || <discretionary-lig-values> || <historical-lig-values> || <contextual-alt-values> ]
 | |
|     RefPtr<CSSStyleValue const> common_ligatures_value;
 | |
|     RefPtr<CSSStyleValue const> discretionary_ligatures_value;
 | |
|     RefPtr<CSSStyleValue const> historical_ligatures_value;
 | |
|     RefPtr<CSSStyleValue const> contextual_value;
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto maybe_value = parse_keyword_value(tokens);
 | |
|         if (!maybe_value)
 | |
|             break;
 | |
|         auto font_variant_ligatures = keyword_to_font_variant_ligatures(maybe_value->to_keyword());
 | |
|         if (!font_variant_ligatures.has_value())
 | |
|             return nullptr;
 | |
| 
 | |
|         switch (font_variant_ligatures.value()) {
 | |
|         // <common-lig-values>       = [ common-ligatures | no-common-ligatures ]
 | |
|         case FontVariantLigatures::CommonLigatures:
 | |
|         case FontVariantLigatures::NoCommonLigatures:
 | |
|             if (common_ligatures_value)
 | |
|                 return nullptr;
 | |
|             common_ligatures_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         // <discretionary-lig-values> = [ discretionary-ligatures | no-discretionary-ligatures ]
 | |
|         case FontVariantLigatures::DiscretionaryLigatures:
 | |
|         case FontVariantLigatures::NoDiscretionaryLigatures:
 | |
|             if (discretionary_ligatures_value)
 | |
|                 return nullptr;
 | |
|             discretionary_ligatures_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         // <historical-lig-values> = [ historical-ligatures | no-historical-ligatures ]
 | |
|         case FontVariantLigatures::HistoricalLigatures:
 | |
|         case FontVariantLigatures::NoHistoricalLigatures:
 | |
|             if (historical_ligatures_value)
 | |
|                 return nullptr;
 | |
|             historical_ligatures_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         // <contextual-alt-values> = [ contextual | no-contextual ]
 | |
|         case FontVariantLigatures::Contextual:
 | |
|         case FontVariantLigatures::NoContextual:
 | |
|             if (contextual_value)
 | |
|                 return nullptr;
 | |
|             contextual_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         case FontVariantLigatures::Normal:
 | |
|         case FontVariantLigatures::None:
 | |
|             return nullptr;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     StyleValueVector values;
 | |
|     if (common_ligatures_value)
 | |
|         values.append(common_ligatures_value.release_nonnull());
 | |
|     if (discretionary_ligatures_value)
 | |
|         values.append(discretionary_ligatures_value.release_nonnull());
 | |
|     if (historical_ligatures_value)
 | |
|         values.append(historical_ligatures_value.release_nonnull());
 | |
|     if (contextual_value)
 | |
|         values.append(contextual_value.release_nonnull());
 | |
| 
 | |
|     if (values.is_empty())
 | |
|         return nullptr;
 | |
|     if (values.size() == 1)
 | |
|         return *values.first();
 | |
| 
 | |
|     return StyleValueList::create(move(values), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_font_variant_numeric_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // 6.7 https://drafts.csswg.org/css-fonts/#propdef-font-variant-numeric
 | |
|     // normal | [ <numeric-figure-values> || <numeric-spacing-values> || <numeric-fraction-values> || ordinal || slashed-zero]
 | |
|     // <numeric-figure-values>       = [ lining-nums | oldstyle-nums ]
 | |
|     // <numeric-spacing-values>      = [ proportional-nums | tabular-nums ]
 | |
|     // <numeric-fraction-values>     = [ diagonal-fractions | stacked-fractions ]
 | |
| 
 | |
|     // normal
 | |
|     if (auto normal = parse_all_as_single_keyword_value(tokens, Keyword::Normal))
 | |
|         return normal.release_nonnull();
 | |
| 
 | |
|     RefPtr<CSSStyleValue const> figures_value;
 | |
|     RefPtr<CSSStyleValue const> spacing_value;
 | |
|     RefPtr<CSSStyleValue const> fractions_value;
 | |
|     RefPtr<CSSStyleValue const> ordinals_value;
 | |
|     RefPtr<CSSStyleValue const> slashed_zero_value;
 | |
| 
 | |
|     // [ <numeric-figure-values> || <numeric-spacing-values> || <numeric-fraction-values> || ordinal || slashed-zero]
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto maybe_value = parse_keyword_value(tokens);
 | |
|         if (!maybe_value)
 | |
|             break;
 | |
|         auto font_variant_numeric = keyword_to_font_variant_numeric(maybe_value->to_keyword());
 | |
|         if (!font_variant_numeric.has_value())
 | |
|             return nullptr;
 | |
|         switch (font_variant_numeric.value()) {
 | |
|         // ... || ordinal
 | |
|         case FontVariantNumeric::Ordinal:
 | |
|             if (ordinals_value)
 | |
|                 return nullptr;
 | |
|             ordinals_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         // ... || slashed-zero
 | |
|         case FontVariantNumeric::SlashedZero:
 | |
|             if (slashed_zero_value)
 | |
|                 return nullptr;
 | |
|             slashed_zero_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         // <numeric-figure-values> = [ lining-nums | oldstyle-nums ]
 | |
|         case FontVariantNumeric::LiningNums:
 | |
|         case FontVariantNumeric::OldstyleNums:
 | |
|             if (figures_value)
 | |
|                 return nullptr;
 | |
|             figures_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         // <numeric-spacing-values> = [ proportional-nums | tabular-nums ]
 | |
|         case FontVariantNumeric::ProportionalNums:
 | |
|         case FontVariantNumeric::TabularNums:
 | |
|             if (spacing_value)
 | |
|                 return nullptr;
 | |
|             spacing_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         // <numeric-fraction-values> = [ diagonal-fractions | stacked-fractions ]
 | |
|         case FontVariantNumeric::DiagonalFractions:
 | |
|         case FontVariantNumeric::StackedFractions:
 | |
|             if (fractions_value)
 | |
|                 return nullptr;
 | |
|             fractions_value = maybe_value.release_nonnull();
 | |
|             break;
 | |
|         case FontVariantNumeric::Normal:
 | |
|             return nullptr;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     StyleValueVector values;
 | |
|     if (figures_value)
 | |
|         values.append(figures_value.release_nonnull());
 | |
|     if (spacing_value)
 | |
|         values.append(spacing_value.release_nonnull());
 | |
|     if (fractions_value)
 | |
|         values.append(fractions_value.release_nonnull());
 | |
|     if (ordinals_value)
 | |
|         values.append(ordinals_value.release_nonnull());
 | |
|     if (slashed_zero_value)
 | |
|         values.append(slashed_zero_value.release_nonnull());
 | |
| 
 | |
|     if (values.is_empty())
 | |
|         return nullptr;
 | |
|     if (values.size() == 1)
 | |
|         return *values.first();
 | |
| 
 | |
|     return StyleValueList::create(move(values), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_list_style_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     RefPtr<CSSStyleValue const> list_position;
 | |
|     RefPtr<CSSStyleValue const> list_image;
 | |
|     RefPtr<CSSStyleValue const> list_type;
 | |
|     int found_nones = 0;
 | |
| 
 | |
|     Vector<PropertyID> remaining_longhands { PropertyID::ListStyleImage, PropertyID::ListStylePosition, PropertyID::ListStyleType };
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     while (tokens.has_next_token()) {
 | |
|         if (auto const& peek = tokens.next_token(); peek.is_ident("none"sv)) {
 | |
|             tokens.discard_a_token();
 | |
|             found_nones++;
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         auto property_and_value = parse_css_value_for_properties(remaining_longhands, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = property_and_value->style_value;
 | |
|         remove_property(remaining_longhands, property_and_value->property);
 | |
| 
 | |
|         switch (property_and_value->property) {
 | |
|         case PropertyID::ListStylePosition: {
 | |
|             VERIFY(!list_position);
 | |
|             list_position = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::ListStyleImage: {
 | |
|             VERIFY(!list_image);
 | |
|             list_image = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::ListStyleType: {
 | |
|             VERIFY(!list_type);
 | |
|             list_type = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (found_nones > 2)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (found_nones == 2) {
 | |
|         if (list_image || list_type)
 | |
|             return nullptr;
 | |
|         auto none = CSSKeywordValue::create(Keyword::None);
 | |
|         list_image = none;
 | |
|         list_type = none;
 | |
| 
 | |
|     } else if (found_nones == 1) {
 | |
|         if (list_image && list_type)
 | |
|             return nullptr;
 | |
|         auto none = CSSKeywordValue::create(Keyword::None);
 | |
|         if (!list_image)
 | |
|             list_image = none;
 | |
|         if (!list_type)
 | |
|             list_type = none;
 | |
|     }
 | |
| 
 | |
|     if (!list_position)
 | |
|         list_position = property_initial_value(PropertyID::ListStylePosition);
 | |
|     if (!list_image)
 | |
|         list_image = property_initial_value(PropertyID::ListStyleImage);
 | |
|     if (!list_type)
 | |
|         list_type = property_initial_value(PropertyID::ListStyleType);
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::ListStyle,
 | |
|         { PropertyID::ListStylePosition, PropertyID::ListStyleImage, PropertyID::ListStyleType },
 | |
|         { list_position.release_nonnull(), list_image.release_nonnull(), list_type.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_math_depth_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://w3c.github.io/mathml-core/#propdef-math-depth
 | |
|     // auto-add | add(<integer>) | <integer>
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     // auto-add
 | |
|     if (tokens.next_token().is_ident("auto-add"sv)) {
 | |
|         tokens.discard_a_token(); // auto-add
 | |
|         transaction.commit();
 | |
|         return MathDepthStyleValue::create_auto_add();
 | |
|     }
 | |
| 
 | |
|     // add(<integer>)
 | |
|     if (tokens.next_token().is_function("add"sv)) {
 | |
|         auto const& function = tokens.next_token().function();
 | |
|         auto context_guard = push_temporary_value_parsing_context(FunctionContext { function.name });
 | |
| 
 | |
|         auto add_tokens = TokenStream { function.value };
 | |
|         add_tokens.discard_whitespace();
 | |
|         if (auto integer_value = parse_integer_value(add_tokens)) {
 | |
|             add_tokens.discard_whitespace();
 | |
|             if (add_tokens.has_next_token())
 | |
|                 return nullptr;
 | |
|             tokens.discard_a_token(); // add()
 | |
|             transaction.commit();
 | |
|             return MathDepthStyleValue::create_add(integer_value.release_nonnull());
 | |
|         }
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     // <integer>
 | |
|     if (auto integer_value = parse_integer_value(tokens)) {
 | |
|         transaction.commit();
 | |
|         return MathDepthStyleValue::create_integer(integer_value.release_nonnull());
 | |
|     }
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_overflow_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto maybe_x_value = parse_css_value_for_property(PropertyID::OverflowX, tokens);
 | |
|     if (!maybe_x_value)
 | |
|         return nullptr;
 | |
|     auto maybe_y_value = parse_css_value_for_property(PropertyID::OverflowY, tokens);
 | |
|     transaction.commit();
 | |
|     if (maybe_y_value) {
 | |
|         return ShorthandStyleValue::create(PropertyID::Overflow,
 | |
|             { PropertyID::OverflowX, PropertyID::OverflowY },
 | |
|             { maybe_x_value.release_nonnull(), maybe_y_value.release_nonnull() });
 | |
|     }
 | |
|     return ShorthandStyleValue::create(PropertyID::Overflow,
 | |
|         { PropertyID::OverflowX, PropertyID::OverflowY },
 | |
|         { *maybe_x_value, *maybe_x_value });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_place_content_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto maybe_align_content_value = parse_css_value_for_property(PropertyID::AlignContent, tokens);
 | |
|     if (!maybe_align_content_value)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!tokens.has_next_token()) {
 | |
|         if (!property_accepts_keyword(PropertyID::JustifyContent, maybe_align_content_value->to_keyword()))
 | |
|             return nullptr;
 | |
|         transaction.commit();
 | |
|         return ShorthandStyleValue::create(PropertyID::PlaceContent,
 | |
|             { PropertyID::AlignContent, PropertyID::JustifyContent },
 | |
|             { *maybe_align_content_value, *maybe_align_content_value });
 | |
|     }
 | |
| 
 | |
|     auto maybe_justify_content_value = parse_css_value_for_property(PropertyID::JustifyContent, tokens);
 | |
|     if (!maybe_justify_content_value)
 | |
|         return nullptr;
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::PlaceContent,
 | |
|         { PropertyID::AlignContent, PropertyID::JustifyContent },
 | |
|         { maybe_align_content_value.release_nonnull(), maybe_justify_content_value.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_place_items_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto maybe_align_items_value = parse_css_value_for_property(PropertyID::AlignItems, tokens);
 | |
|     if (!maybe_align_items_value)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!tokens.has_next_token()) {
 | |
|         if (!property_accepts_keyword(PropertyID::JustifyItems, maybe_align_items_value->to_keyword()))
 | |
|             return nullptr;
 | |
|         transaction.commit();
 | |
|         return ShorthandStyleValue::create(PropertyID::PlaceItems,
 | |
|             { PropertyID::AlignItems, PropertyID::JustifyItems },
 | |
|             { *maybe_align_items_value, *maybe_align_items_value });
 | |
|     }
 | |
| 
 | |
|     auto maybe_justify_items_value = parse_css_value_for_property(PropertyID::JustifyItems, tokens);
 | |
|     if (!maybe_justify_items_value)
 | |
|         return nullptr;
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::PlaceItems,
 | |
|         { PropertyID::AlignItems, PropertyID::JustifyItems },
 | |
|         { *maybe_align_items_value, *maybe_justify_items_value });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_place_self_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto maybe_align_self_value = parse_css_value_for_property(PropertyID::AlignSelf, tokens);
 | |
|     if (!maybe_align_self_value)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!tokens.has_next_token()) {
 | |
|         if (!property_accepts_keyword(PropertyID::JustifySelf, maybe_align_self_value->to_keyword()))
 | |
|             return nullptr;
 | |
|         transaction.commit();
 | |
|         return ShorthandStyleValue::create(PropertyID::PlaceSelf,
 | |
|             { PropertyID::AlignSelf, PropertyID::JustifySelf },
 | |
|             { *maybe_align_self_value, *maybe_align_self_value });
 | |
|     }
 | |
| 
 | |
|     auto maybe_justify_self_value = parse_css_value_for_property(PropertyID::JustifySelf, tokens);
 | |
|     if (!maybe_justify_self_value)
 | |
|         return nullptr;
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::PlaceSelf,
 | |
|         { PropertyID::AlignSelf, PropertyID::JustifySelf },
 | |
|         { *maybe_align_self_value, *maybe_justify_self_value });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_quotes_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://www.w3.org/TR/css-content-3/#quotes-property
 | |
|     // auto | none | [ <string> <string> ]+
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     if (tokens.remaining_token_count() == 1) {
 | |
|         auto keyword = parse_keyword_value(tokens);
 | |
|         if (keyword && property_accepts_keyword(PropertyID::Quotes, keyword->to_keyword())) {
 | |
|             transaction.commit();
 | |
|             return keyword;
 | |
|         }
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     // Parse an even number of <string> values.
 | |
|     if (tokens.remaining_token_count() % 2 != 0)
 | |
|         return nullptr;
 | |
| 
 | |
|     StyleValueVector string_values;
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto maybe_string = parse_string_value(tokens);
 | |
|         if (!maybe_string)
 | |
|             return nullptr;
 | |
| 
 | |
|         string_values.append(maybe_string.release_nonnull());
 | |
|     }
 | |
| 
 | |
|     transaction.commit();
 | |
|     return StyleValueList::create(move(string_values), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_text_decoration_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     RefPtr<CSSStyleValue const> decoration_line;
 | |
|     RefPtr<CSSStyleValue const> decoration_thickness;
 | |
|     RefPtr<CSSStyleValue const> decoration_style;
 | |
|     RefPtr<CSSStyleValue const> decoration_color;
 | |
| 
 | |
|     auto remaining_longhands = Vector { PropertyID::TextDecorationColor, PropertyID::TextDecorationLine, PropertyID::TextDecorationStyle, PropertyID::TextDecorationThickness };
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto property_and_value = parse_css_value_for_properties(remaining_longhands, tokens);
 | |
|         if (!property_and_value.has_value())
 | |
|             return nullptr;
 | |
|         auto& value = property_and_value->style_value;
 | |
|         remove_property(remaining_longhands, property_and_value->property);
 | |
| 
 | |
|         switch (property_and_value->property) {
 | |
|         case PropertyID::TextDecorationColor: {
 | |
|             VERIFY(!decoration_color);
 | |
|             decoration_color = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::TextDecorationLine: {
 | |
|             VERIFY(!decoration_line);
 | |
|             tokens.reconsume_current_input_token();
 | |
|             auto parsed_decoration_line = parse_text_decoration_line_value(tokens);
 | |
|             if (!parsed_decoration_line)
 | |
|                 return nullptr;
 | |
|             decoration_line = parsed_decoration_line.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::TextDecorationThickness: {
 | |
|             VERIFY(!decoration_thickness);
 | |
|             decoration_thickness = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         case PropertyID::TextDecorationStyle: {
 | |
|             VERIFY(!decoration_style);
 | |
|             decoration_style = value.release_nonnull();
 | |
|             continue;
 | |
|         }
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!decoration_line)
 | |
|         decoration_line = property_initial_value(PropertyID::TextDecorationLine);
 | |
|     if (!decoration_thickness)
 | |
|         decoration_thickness = property_initial_value(PropertyID::TextDecorationThickness);
 | |
|     if (!decoration_style)
 | |
|         decoration_style = property_initial_value(PropertyID::TextDecorationStyle);
 | |
|     if (!decoration_color)
 | |
|         decoration_color = property_initial_value(PropertyID::TextDecorationColor);
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::TextDecoration,
 | |
|         { PropertyID::TextDecorationLine, PropertyID::TextDecorationThickness, PropertyID::TextDecorationStyle, PropertyID::TextDecorationColor },
 | |
|         { decoration_line.release_nonnull(), decoration_thickness.release_nonnull(), decoration_style.release_nonnull(), decoration_color.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_text_decoration_line_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     StyleValueVector style_values;
 | |
| 
 | |
|     bool includes_spelling_or_grammar_error_value = false;
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto maybe_value = parse_css_value_for_property(PropertyID::TextDecorationLine, tokens);
 | |
|         if (!maybe_value)
 | |
|             break;
 | |
|         auto value = maybe_value.release_nonnull();
 | |
| 
 | |
|         if (auto maybe_line = keyword_to_text_decoration_line(value->to_keyword()); maybe_line.has_value()) {
 | |
|             if (maybe_line == TextDecorationLine::None) {
 | |
|                 if (!style_values.is_empty())
 | |
|                     break;
 | |
|                 return value;
 | |
|             }
 | |
|             if (first_is_one_of(*maybe_line, TextDecorationLine::SpellingError, TextDecorationLine::GrammarError)) {
 | |
|                 includes_spelling_or_grammar_error_value = true;
 | |
|             }
 | |
|             if (style_values.contains_slow(value))
 | |
|                 break;
 | |
|             style_values.append(move(value));
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     if (style_values.is_empty())
 | |
|         return nullptr;
 | |
| 
 | |
|     // These can only appear on their own.
 | |
|     if (style_values.size() > 1 && includes_spelling_or_grammar_error_value)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (style_values.size() == 1)
 | |
|         return *style_values.first();
 | |
| 
 | |
|     quick_sort(style_values, [](auto& left, auto& right) {
 | |
|         return *keyword_to_text_decoration_line(left->to_keyword()) < *keyword_to_text_decoration_line(right->to_keyword());
 | |
|     });
 | |
| 
 | |
|     return StyleValueList::create(move(style_values), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/pointerevents/#the-touch-action-css-property
 | |
| RefPtr<CSSStyleValue const> Parser::parse_touch_action_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // auto | none | [ [ pan-x | pan-left | pan-right ] || [ pan-y | pan-up | pan-down ] ] | manipulation
 | |
| 
 | |
|     if (auto value = parse_all_as_single_keyword_value(tokens, Keyword::Auto))
 | |
|         return value;
 | |
|     if (auto value = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return value;
 | |
|     if (auto value = parse_all_as_single_keyword_value(tokens, Keyword::Manipulation))
 | |
|         return value;
 | |
| 
 | |
|     StyleValueVector parsed_values;
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     // We will verify that we have up to one vertical and one horizontal value
 | |
|     bool has_horizontal = false;
 | |
|     bool has_vertical = false;
 | |
| 
 | |
|     // Were the values specified in y/x order? (we need to store them in canonical x/y order)
 | |
|     bool swap_order = false;
 | |
| 
 | |
|     while (auto parsed_value = parse_css_value_for_property(PropertyID::TouchAction, tokens)) {
 | |
|         switch (parsed_value->as_keyword().keyword()) {
 | |
|         case Keyword::PanX:
 | |
|         case Keyword::PanLeft:
 | |
|         case Keyword::PanRight:
 | |
|             if (has_horizontal)
 | |
|                 return {};
 | |
|             if (has_vertical)
 | |
|                 swap_order = true;
 | |
|             has_horizontal = true;
 | |
|             break;
 | |
|         case Keyword::PanY:
 | |
|         case Keyword::PanUp:
 | |
|         case Keyword::PanDown:
 | |
|             if (has_vertical)
 | |
|                 return {};
 | |
|             has_vertical = true;
 | |
|             break;
 | |
|         case Keyword::Auto:
 | |
|         case Keyword::None:
 | |
|         case Keyword::Manipulation:
 | |
|             // Not valid as part of a list
 | |
|             return {};
 | |
|         default:
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
| 
 | |
|         parsed_values.append(parsed_value.release_nonnull());
 | |
|         if (!tokens.has_next_token())
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     if (swap_order)
 | |
|         swap(parsed_values[0], parsed_values[1]);
 | |
| 
 | |
|     transaction.commit();
 | |
|     return StyleValueList::create(move(parsed_values), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/css-transforms-1/#transform-property
 | |
| RefPtr<CSSStyleValue const> Parser::parse_transform_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // <transform> = none | <transform-list>
 | |
|     // <transform-list> = <transform-function>+
 | |
| 
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     StyleValueVector transformations;
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto const& part = tokens.consume_a_token();
 | |
|         if (!part.is_function())
 | |
|             return nullptr;
 | |
|         auto maybe_function = transform_function_from_string(part.function().name);
 | |
|         if (!maybe_function.has_value())
 | |
|             return nullptr;
 | |
| 
 | |
|         auto context_guard = push_temporary_value_parsing_context(FunctionContext { part.function().name });
 | |
| 
 | |
|         auto function = maybe_function.release_value();
 | |
|         auto function_metadata = transform_function_metadata(function);
 | |
| 
 | |
|         auto function_tokens = TokenStream { part.function().value };
 | |
|         auto arguments = parse_a_comma_separated_list_of_component_values(function_tokens);
 | |
| 
 | |
|         if (arguments.size() > function_metadata.parameters.size()) {
 | |
|             dbgln_if(CSS_PARSER_DEBUG, "Too many arguments to {}. max: {}", part.function().name, function_metadata.parameters.size());
 | |
|             return nullptr;
 | |
|         }
 | |
| 
 | |
|         if (arguments.size() < function_metadata.parameters.size() && function_metadata.parameters[arguments.size()].required) {
 | |
|             dbgln_if(CSS_PARSER_DEBUG, "Required parameter at position {} is missing", arguments.size());
 | |
|             return nullptr;
 | |
|         }
 | |
| 
 | |
|         StyleValueVector values;
 | |
|         for (auto argument_index = 0u; argument_index < arguments.size(); ++argument_index) {
 | |
|             TokenStream argument_tokens { arguments[argument_index] };
 | |
|             argument_tokens.discard_whitespace();
 | |
| 
 | |
|             switch (function_metadata.parameters[argument_index].type) {
 | |
|             case TransformFunctionParameterType::Angle: {
 | |
|                 // These are `<angle> | <zero>` in the spec, so we have to check for both kinds.
 | |
|                 if (auto angle_value = parse_angle_value(argument_tokens)) {
 | |
|                     values.append(angle_value.release_nonnull());
 | |
|                     break;
 | |
|                 }
 | |
|                 if (argument_tokens.next_token().is(Token::Type::Number) && argument_tokens.next_token().token().number_value() == 0) {
 | |
|                     argument_tokens.discard_a_token(); // 0
 | |
|                     values.append(AngleStyleValue::create(Angle::make_degrees(0)));
 | |
|                     break;
 | |
|                 }
 | |
|                 return nullptr;
 | |
|             }
 | |
|             case TransformFunctionParameterType::Length:
 | |
|             case TransformFunctionParameterType::LengthNone: {
 | |
|                 if (auto length_value = parse_length_value(argument_tokens)) {
 | |
|                     values.append(length_value.release_nonnull());
 | |
|                     break;
 | |
|                 }
 | |
|                 if (function_metadata.parameters[argument_index].type == TransformFunctionParameterType::LengthNone
 | |
|                     && argument_tokens.next_token().is_ident("none"sv)) {
 | |
| 
 | |
|                     argument_tokens.discard_a_token(); // none
 | |
|                     values.append(CSSKeywordValue::create(Keyword::None));
 | |
|                     break;
 | |
|                 }
 | |
|                 return nullptr;
 | |
|             }
 | |
|             case TransformFunctionParameterType::LengthPercentage: {
 | |
|                 if (auto length_percentage_value = parse_length_percentage_value(argument_tokens)) {
 | |
|                     values.append(length_percentage_value.release_nonnull());
 | |
|                     break;
 | |
|                 }
 | |
|                 return nullptr;
 | |
|             }
 | |
|             case TransformFunctionParameterType::Number: {
 | |
|                 if (auto number_value = parse_number_value(argument_tokens)) {
 | |
|                     values.append(number_value.release_nonnull());
 | |
|                     break;
 | |
|                 }
 | |
|                 return nullptr;
 | |
|             }
 | |
|             case TransformFunctionParameterType::NumberPercentage: {
 | |
|                 if (auto number_percentage_value = parse_number_percentage_value(argument_tokens)) {
 | |
|                     values.append(number_percentage_value.release_nonnull());
 | |
|                     break;
 | |
|                 }
 | |
|                 return nullptr;
 | |
|             }
 | |
|             }
 | |
| 
 | |
|             argument_tokens.discard_whitespace();
 | |
|             if (argument_tokens.has_next_token())
 | |
|                 return nullptr;
 | |
|         }
 | |
| 
 | |
|         transformations.append(TransformationStyleValue::create(PropertyID::Transform, function, move(values)));
 | |
|     }
 | |
|     transaction.commit();
 | |
|     return StyleValueList::create(move(transformations), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/css-transforms-1/#propdef-transform-origin
 | |
| // FIXME: This only supports a 2D position
 | |
| RefPtr<CSSStyleValue const> Parser::parse_transform_origin_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     enum class Axis {
 | |
|         None,
 | |
|         X,
 | |
|         Y,
 | |
|     };
 | |
| 
 | |
|     struct AxisOffset {
 | |
|         Axis axis;
 | |
|         NonnullRefPtr<CSSStyleValue const> offset;
 | |
|     };
 | |
| 
 | |
|     auto to_axis_offset = [](RefPtr<CSSStyleValue const> value) -> Optional<AxisOffset> {
 | |
|         if (!value)
 | |
|             return OptionalNone {};
 | |
|         if (value->is_percentage())
 | |
|             return AxisOffset { Axis::None, value->as_percentage() };
 | |
|         if (value->is_length())
 | |
|             return AxisOffset { Axis::None, value->as_length() };
 | |
|         if (value->is_keyword()) {
 | |
|             switch (value->to_keyword()) {
 | |
|             case Keyword::Top:
 | |
|                 return AxisOffset { Axis::Y, PercentageStyleValue::create(Percentage(0)) };
 | |
|             case Keyword::Left:
 | |
|                 return AxisOffset { Axis::X, PercentageStyleValue::create(Percentage(0)) };
 | |
|             case Keyword::Center:
 | |
|                 return AxisOffset { Axis::None, PercentageStyleValue::create(Percentage(50)) };
 | |
|             case Keyword::Bottom:
 | |
|                 return AxisOffset { Axis::Y, PercentageStyleValue::create(Percentage(100)) };
 | |
|             case Keyword::Right:
 | |
|                 return AxisOffset { Axis::X, PercentageStyleValue::create(Percentage(100)) };
 | |
|             default:
 | |
|                 return OptionalNone {};
 | |
|             }
 | |
|         }
 | |
|         if (value->is_calculated()) {
 | |
|             return AxisOffset { Axis::None, value->as_calculated() };
 | |
|         }
 | |
|         return OptionalNone {};
 | |
|     };
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto make_list = [&transaction](NonnullRefPtr<CSSStyleValue const> const& x_value, NonnullRefPtr<CSSStyleValue const> const& y_value) -> NonnullRefPtr<StyleValueList> {
 | |
|         transaction.commit();
 | |
|         return StyleValueList::create(StyleValueVector { x_value, y_value }, StyleValueList::Separator::Space);
 | |
|     };
 | |
| 
 | |
|     switch (tokens.remaining_token_count()) {
 | |
|     case 1: {
 | |
|         auto single_value = to_axis_offset(parse_css_value_for_property(PropertyID::TransformOrigin, tokens));
 | |
|         if (!single_value.has_value())
 | |
|             return nullptr;
 | |
|         // If only one value is specified, the second value is assumed to be center.
 | |
|         // FIXME: If one or two values are specified, the third value is assumed to be 0px.
 | |
|         switch (single_value->axis) {
 | |
|         case Axis::None:
 | |
|         case Axis::X:
 | |
|             return make_list(single_value->offset, PercentageStyleValue::create(Percentage(50)));
 | |
|         case Axis::Y:
 | |
|             return make_list(PercentageStyleValue::create(Percentage(50)), single_value->offset);
 | |
|         }
 | |
|         VERIFY_NOT_REACHED();
 | |
|     }
 | |
|     case 2: {
 | |
|         auto first_value = to_axis_offset(parse_css_value_for_property(PropertyID::TransformOrigin, tokens));
 | |
|         auto second_value = to_axis_offset(parse_css_value_for_property(PropertyID::TransformOrigin, tokens));
 | |
|         if (!first_value.has_value() || !second_value.has_value())
 | |
|             return nullptr;
 | |
| 
 | |
|         RefPtr<CSSStyleValue const> x_value;
 | |
|         RefPtr<CSSStyleValue const> y_value;
 | |
| 
 | |
|         if (first_value->axis == Axis::X) {
 | |
|             x_value = first_value->offset;
 | |
|         } else if (first_value->axis == Axis::Y) {
 | |
|             y_value = first_value->offset;
 | |
|         }
 | |
| 
 | |
|         if (second_value->axis == Axis::X) {
 | |
|             if (x_value)
 | |
|                 return nullptr;
 | |
|             x_value = second_value->offset;
 | |
|             // Put the other in Y since its axis can't have been X
 | |
|             y_value = first_value->offset;
 | |
|         } else if (second_value->axis == Axis::Y) {
 | |
|             if (y_value)
 | |
|                 return nullptr;
 | |
|             y_value = second_value->offset;
 | |
|             // Put the other in X since its axis can't have been Y
 | |
|             x_value = first_value->offset;
 | |
|         } else {
 | |
|             if (x_value) {
 | |
|                 VERIFY(!y_value);
 | |
|                 y_value = second_value->offset;
 | |
|             } else {
 | |
|                 VERIFY(!x_value);
 | |
|                 x_value = second_value->offset;
 | |
|             }
 | |
|         }
 | |
|         // If two or more values are defined and either no value is a keyword, or the only used keyword is center,
 | |
|         // then the first value represents the horizontal position (or offset) and the second represents the vertical position (or offset).
 | |
|         // FIXME: A third value always represents the Z position (or offset) and must be of type <length>.
 | |
|         if (first_value->axis == Axis::None && second_value->axis == Axis::None) {
 | |
|             x_value = first_value->offset;
 | |
|             y_value = second_value->offset;
 | |
|         }
 | |
|         return make_list(x_value.release_nonnull(), y_value.release_nonnull());
 | |
|     }
 | |
|     }
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_transition_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     Vector<TransitionStyleValue::Transition> transitions;
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         TransitionStyleValue::Transition transition;
 | |
|         auto time_value_count = 0;
 | |
|         bool transition_behavior_found = false;
 | |
|         while (tokens.has_next_token() && !tokens.next_token().is(Token::Type::Comma)) {
 | |
|             if (auto maybe_time = parse_time(tokens); maybe_time.has_value()) {
 | |
|                 auto time = maybe_time.release_value();
 | |
|                 switch (time_value_count) {
 | |
|                 case 0:
 | |
|                     if (!time.is_calculated() && !property_accepts_time(PropertyID::TransitionDuration, time.value()))
 | |
|                         return nullptr;
 | |
|                     transition.duration = move(time);
 | |
|                     break;
 | |
|                 case 1:
 | |
|                     if (!time.is_calculated() && !property_accepts_time(PropertyID::TransitionDelay, time.value()))
 | |
|                         return nullptr;
 | |
|                     transition.delay = move(time);
 | |
|                     break;
 | |
|                 default:
 | |
|                     dbgln_if(CSS_PARSER_DEBUG, "Transition property has more than two time values");
 | |
|                     return {};
 | |
|                 }
 | |
|                 time_value_count++;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             if (auto easing = parse_easing_value(tokens)) {
 | |
|                 if (transition.easing) {
 | |
|                     dbgln_if(CSS_PARSER_DEBUG, "Transition property has multiple easing values");
 | |
|                     return {};
 | |
|                 }
 | |
| 
 | |
|                 transition.easing = easing->as_easing();
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             if (!transition_behavior_found && (tokens.peek_token().is_ident("normal"sv) || tokens.peek_token().is_ident("allow-discrete"sv))) {
 | |
|                 transition_behavior_found = true;
 | |
|                 auto ident = tokens.consume_a_token().token().ident();
 | |
|                 if (ident == "allow-discrete"sv)
 | |
|                     transition.transition_behavior = TransitionBehavior::AllowDiscrete;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             if (auto token = tokens.peek_token(); token.is_ident("all"sv)) {
 | |
|                 auto transition_keyword = parse_keyword_value(tokens);
 | |
|                 VERIFY(transition_keyword->to_keyword() == Keyword::All);
 | |
|                 if (transition.property_name) {
 | |
|                     dbgln_if(CSS_PARSER_DEBUG, "Transition property has multiple property identifiers");
 | |
|                     return {};
 | |
|                 }
 | |
|                 transition.property_name = transition_keyword.release_nonnull();
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             if (auto transition_property = parse_custom_ident_value(tokens, { { "all"sv, "none"sv } })) {
 | |
|                 if (transition.property_name) {
 | |
|                     dbgln_if(CSS_PARSER_DEBUG, "Transition property has multiple property identifiers");
 | |
|                     return {};
 | |
|                 }
 | |
| 
 | |
|                 auto custom_ident = transition_property->custom_ident();
 | |
|                 if (auto property = property_id_from_string(custom_ident); property.has_value()) {
 | |
|                     transition.property_name = CustomIdentStyleValue::create(custom_ident);
 | |
|                     continue;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             dbgln_if(CSS_PARSER_DEBUG, "Transition property has unexpected token \"{}\"", tokens.next_token().to_string());
 | |
|             return {};
 | |
|         }
 | |
| 
 | |
|         if (!transition.property_name)
 | |
|             transition.property_name = CSSKeywordValue::create(Keyword::All);
 | |
| 
 | |
|         if (!transition.easing)
 | |
|             transition.easing = EasingStyleValue::create(EasingStyleValue::CubicBezier::ease());
 | |
| 
 | |
|         transitions.append(move(transition));
 | |
| 
 | |
|         if (!tokens.next_token().is(Token::Type::Comma))
 | |
|             break;
 | |
| 
 | |
|         tokens.discard_a_token();
 | |
|     }
 | |
| 
 | |
|     transaction.commit();
 | |
|     return TransitionStyleValue::create(move(transitions));
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_list_of_time_values(PropertyID property_id, TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto time_values = parse_a_comma_separated_list_of_component_values(tokens);
 | |
|     StyleValueVector time_value_list;
 | |
|     for (auto const& value : time_values) {
 | |
|         TokenStream time_value_tokens { value };
 | |
|         auto time_style_value = parse_time_value(time_value_tokens);
 | |
|         if (!time_style_value)
 | |
|             return nullptr;
 | |
|         if (time_value_tokens.has_next_token())
 | |
|             return nullptr;
 | |
|         if (!time_style_value->is_calculated() && !property_accepts_time(property_id, time_style_value->as_time().time()))
 | |
|             return nullptr;
 | |
|         time_value_list.append(*time_style_value);
 | |
|     }
 | |
| 
 | |
|     transaction.commit();
 | |
|     return StyleValueList::create(move(time_value_list), StyleValueList::Separator::Comma);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_transition_property_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://drafts.csswg.org/css-transitions/#transition-property-property
 | |
|     // none | <single-transition-property>#
 | |
| 
 | |
|     // none
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     // <single-transition-property>#
 | |
|     // <single-transition-property> = all | <custom-ident>
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     auto transition_property_values = parse_a_comma_separated_list_of_component_values(tokens);
 | |
| 
 | |
|     StyleValueVector transition_properties;
 | |
|     for (auto const& value : transition_property_values) {
 | |
|         TokenStream transition_property_tokens { value };
 | |
|         if (auto all_keyword_value = parse_all_as_single_keyword_value(transition_property_tokens, Keyword::All)) {
 | |
|             transition_properties.append(*all_keyword_value);
 | |
|         } else {
 | |
|             auto custom_ident = parse_custom_ident_value(transition_property_tokens, { { "all"sv, "none"sv } });
 | |
|             if (!custom_ident || transition_property_tokens.has_next_token())
 | |
|                 return nullptr;
 | |
| 
 | |
|             transition_properties.append(custom_ident.release_nonnull());
 | |
|         }
 | |
|     }
 | |
|     transaction.commit();
 | |
|     return StyleValueList::create(move(transition_properties), StyleValueList::Separator::Comma);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_translate_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     if (tokens.remaining_token_count() == 1) {
 | |
|         // "none"
 | |
|         if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|             return none;
 | |
|     }
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto maybe_x = parse_length_percentage_value(tokens);
 | |
|     if (!maybe_x)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!tokens.has_next_token()) {
 | |
|         transaction.commit();
 | |
|         return TransformationStyleValue::create(PropertyID::Translate, TransformFunction::Translate, { maybe_x.release_nonnull(), LengthStyleValue::create(Length::make_px(0)) });
 | |
|     }
 | |
| 
 | |
|     auto maybe_y = parse_length_percentage_value(tokens);
 | |
|     if (!maybe_y)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!tokens.has_next_token()) {
 | |
|         transaction.commit();
 | |
|         return TransformationStyleValue::create(PropertyID::Translate, TransformFunction::Translate, { maybe_x.release_nonnull(), maybe_y.release_nonnull() });
 | |
|     }
 | |
| 
 | |
|     auto maybe_z = parse_length_value(tokens);
 | |
|     if (!maybe_z)
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return TransformationStyleValue::create(PropertyID::Translate, TransformFunction::Translate3d, { maybe_x.release_nonnull(), maybe_y.release_nonnull(), maybe_z.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_scale_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     if (tokens.remaining_token_count() == 1) {
 | |
|         // "none"
 | |
|         if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|             return none;
 | |
|     }
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto maybe_x = parse_number_percentage_value(tokens);
 | |
|     if (!maybe_x)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!tokens.has_next_token()) {
 | |
|         transaction.commit();
 | |
|         return TransformationStyleValue::create(PropertyID::Scale, TransformFunction::Scale, { *maybe_x, *maybe_x });
 | |
|     }
 | |
| 
 | |
|     auto maybe_y = parse_number_percentage_value(tokens);
 | |
|     if (!maybe_y)
 | |
|         return nullptr;
 | |
| 
 | |
|     if (!tokens.has_next_token()) {
 | |
|         transaction.commit();
 | |
|         return TransformationStyleValue::create(PropertyID::Scale, TransformFunction::Scale, { maybe_x.release_nonnull(), maybe_y.release_nonnull() });
 | |
|     }
 | |
| 
 | |
|     auto maybe_z = parse_number_percentage_value(tokens);
 | |
|     if (!maybe_z)
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return TransformationStyleValue::create(PropertyID::Scale, TransformFunction::Scale, { maybe_x.release_nonnull(), maybe_y.release_nonnull(), maybe_z.release_nonnull() });
 | |
| }
 | |
| 
 | |
| // https://drafts.csswg.org/css-overflow/#propdef-scrollbar-gutter
 | |
| RefPtr<CSSStyleValue const> Parser::parse_scrollbar_gutter_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // auto | stable && both-edges?
 | |
|     if (!tokens.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto parse_stable = [&]() -> Optional<bool> {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         auto const& token = tokens.consume_a_token();
 | |
|         if (!token.is(Token::Type::Ident))
 | |
|             return {};
 | |
|         auto const& ident = token.token().ident();
 | |
|         if (ident.equals_ignoring_ascii_case("auto"sv)) {
 | |
|             transaction.commit();
 | |
|             return false;
 | |
|         } else if (ident.equals_ignoring_ascii_case("stable"sv)) {
 | |
|             transaction.commit();
 | |
|             return true;
 | |
|         }
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     auto parse_both_edges = [&]() -> Optional<bool> {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         auto const& token = tokens.consume_a_token();
 | |
|         if (!token.is(Token::Type::Ident))
 | |
|             return {};
 | |
|         auto const& ident = token.token().ident();
 | |
|         if (ident.equals_ignoring_ascii_case("both-edges"sv)) {
 | |
|             transaction.commit();
 | |
|             return true;
 | |
|         }
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     Optional<bool> stable;
 | |
|     Optional<bool> both_edges;
 | |
|     if (stable = parse_stable(); stable.has_value()) {
 | |
|         if (stable.value())
 | |
|             both_edges = parse_both_edges();
 | |
|     } else if (both_edges = parse_both_edges(); both_edges.has_value()) {
 | |
|         stable = parse_stable();
 | |
|         if (!stable.has_value() || !stable.value())
 | |
|             return nullptr;
 | |
|     }
 | |
| 
 | |
|     if (tokens.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
| 
 | |
|     ScrollbarGutter gutter_value;
 | |
|     if (both_edges.has_value())
 | |
|         gutter_value = ScrollbarGutter::BothEdges;
 | |
|     else if (stable.has_value() && stable.value())
 | |
|         gutter_value = ScrollbarGutter::Stable;
 | |
|     else
 | |
|         gutter_value = ScrollbarGutter::Auto;
 | |
|     return ScrollbarGutterStyleValue::create(gutter_value);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_grid_track_placement_shorthand_value(PropertyID property_id, TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto start_property = (property_id == PropertyID::GridColumn) ? PropertyID::GridColumnStart : PropertyID::GridRowStart;
 | |
|     auto end_property = (property_id == PropertyID::GridColumn) ? PropertyID::GridColumnEnd : PropertyID::GridRowEnd;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     NonnullRawPtr<ComponentValue const> current_token = tokens.consume_a_token();
 | |
| 
 | |
|     Vector<ComponentValue> track_start_placement_tokens;
 | |
|     while (true) {
 | |
|         if (current_token->is_delim('/')) {
 | |
|             if (!tokens.has_next_token())
 | |
|                 return nullptr;
 | |
|             break;
 | |
|         }
 | |
|         track_start_placement_tokens.append(current_token);
 | |
|         if (!tokens.has_next_token())
 | |
|             break;
 | |
|         current_token = tokens.consume_a_token();
 | |
|     }
 | |
| 
 | |
|     Vector<ComponentValue> track_end_placement_tokens;
 | |
|     if (tokens.has_next_token()) {
 | |
|         current_token = tokens.consume_a_token();
 | |
|         while (true) {
 | |
|             track_end_placement_tokens.append(current_token);
 | |
|             if (!tokens.has_next_token())
 | |
|                 break;
 | |
|             current_token = tokens.consume_a_token();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     TokenStream track_start_placement_token_stream { track_start_placement_tokens };
 | |
|     auto parsed_start_value = parse_grid_track_placement(track_start_placement_token_stream);
 | |
|     if (parsed_start_value && track_end_placement_tokens.is_empty()) {
 | |
|         transaction.commit();
 | |
|         if (parsed_start_value->grid_track_placement().has_identifier()) {
 | |
|             auto custom_ident = parsed_start_value.release_nonnull();
 | |
|             return ShorthandStyleValue::create(property_id, { start_property, end_property }, { custom_ident, custom_ident });
 | |
|         }
 | |
|         return ShorthandStyleValue::create(property_id,
 | |
|             { start_property, end_property },
 | |
|             { parsed_start_value.release_nonnull(), GridTrackPlacementStyleValue::create(GridTrackPlacement::make_auto()) });
 | |
|     }
 | |
| 
 | |
|     TokenStream track_end_placement_token_stream { track_end_placement_tokens };
 | |
|     auto parsed_end_value = parse_grid_track_placement(track_end_placement_token_stream);
 | |
|     if (parsed_start_value && parsed_end_value) {
 | |
|         transaction.commit();
 | |
|         return ShorthandStyleValue::create(property_id,
 | |
|             { start_property, end_property },
 | |
|             { parsed_start_value.release_nonnull(), parsed_end_value.release_nonnull() });
 | |
|     }
 | |
| 
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/css-grid-2/#explicit-grid-shorthand
 | |
| // 7.4. Explicit Grid Shorthand: the grid-template property
 | |
| RefPtr<CSSStyleValue const> Parser::parse_grid_track_size_list_shorthand_value(PropertyID property_id, TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // The grid-template property is a shorthand for setting grid-template-columns, grid-template-rows,
 | |
|     // and grid-template-areas in a single declaration. It has several distinct syntax forms:
 | |
|     // none
 | |
|     //    - Sets all three properties to their initial values (none).
 | |
|     // <'grid-template-rows'> / <'grid-template-columns'>
 | |
|     //    - Sets grid-template-rows and grid-template-columns to the specified values, respectively, and sets grid-template-areas to none.
 | |
|     // [ <line-names>? <string> <track-size>? <line-names>? ]+ [ / <explicit-track-list> ]?
 | |
|     //    - Sets grid-template-areas to the strings listed.
 | |
|     //    - Sets grid-template-rows to the <track-size>s following each string (filling in auto for any missing sizes),
 | |
|     //      and splicing in the named lines defined before/after each size.
 | |
|     //    - Sets grid-template-columns to the track listing specified after the slash (or none, if not specified).
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     // FIXME: Read the parts in place if possible, instead of constructing separate vectors and streams.
 | |
|     Vector<ComponentValue> template_rows_tokens;
 | |
|     Vector<ComponentValue> template_columns_tokens;
 | |
|     Vector<ComponentValue> template_area_tokens;
 | |
| 
 | |
|     bool found_forward_slash = false;
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto& token = tokens.consume_a_token();
 | |
|         if (token.is_delim('/')) {
 | |
|             if (found_forward_slash)
 | |
|                 return nullptr;
 | |
|             found_forward_slash = true;
 | |
|             continue;
 | |
|         }
 | |
|         if (found_forward_slash) {
 | |
|             template_columns_tokens.append(token);
 | |
|             continue;
 | |
|         }
 | |
|         if (token.is(Token::Type::String))
 | |
|             template_area_tokens.append(token);
 | |
|         else
 | |
|             template_rows_tokens.append(token);
 | |
|     }
 | |
| 
 | |
|     TokenStream template_area_token_stream { template_area_tokens };
 | |
|     TokenStream template_rows_token_stream { template_rows_tokens };
 | |
|     TokenStream template_columns_token_stream { template_columns_tokens };
 | |
|     auto parsed_template_areas_values = parse_grid_template_areas_value(template_area_token_stream);
 | |
|     auto parsed_template_rows_values = parse_grid_track_size_list(template_rows_token_stream, true);
 | |
|     auto parsed_template_columns_values = parse_grid_track_size_list(template_columns_token_stream);
 | |
| 
 | |
|     if (template_area_token_stream.has_next_token()
 | |
|         || template_rows_token_stream.has_next_token()
 | |
|         || template_columns_token_stream.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(property_id,
 | |
|         { PropertyID::GridTemplateAreas, PropertyID::GridTemplateRows, PropertyID::GridTemplateColumns },
 | |
|         { parsed_template_areas_values.release_nonnull(), parsed_template_rows_values.release_nonnull(), parsed_template_columns_values.release_nonnull() });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_grid_area_shorthand_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto parse_placement_tokens = [&](Vector<ComponentValue>& placement_tokens, bool check_for_delimiter = true) -> void {
 | |
|         while (tokens.has_next_token()) {
 | |
|             auto& current_token = tokens.consume_a_token();
 | |
|             if (check_for_delimiter && current_token.is_delim('/'))
 | |
|                 break;
 | |
|             placement_tokens.append(current_token);
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     Vector<ComponentValue> row_start_placement_tokens;
 | |
|     parse_placement_tokens(row_start_placement_tokens);
 | |
| 
 | |
|     Vector<ComponentValue> column_start_placement_tokens;
 | |
|     if (tokens.has_next_token())
 | |
|         parse_placement_tokens(column_start_placement_tokens);
 | |
| 
 | |
|     Vector<ComponentValue> row_end_placement_tokens;
 | |
|     if (tokens.has_next_token())
 | |
|         parse_placement_tokens(row_end_placement_tokens);
 | |
| 
 | |
|     Vector<ComponentValue> column_end_placement_tokens;
 | |
|     if (tokens.has_next_token())
 | |
|         parse_placement_tokens(column_end_placement_tokens, false);
 | |
| 
 | |
|     // https://www.w3.org/TR/css-grid-2/#placement-shorthands
 | |
|     // The grid-area property is a shorthand for grid-row-start, grid-column-start, grid-row-end and
 | |
|     // grid-column-end.
 | |
|     TokenStream row_start_placement_token_stream { row_start_placement_tokens };
 | |
|     auto row_start_style_value = parse_grid_track_placement(row_start_placement_token_stream);
 | |
|     if (row_start_placement_token_stream.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     TokenStream column_start_placement_token_stream { column_start_placement_tokens };
 | |
|     auto column_start_style_value = parse_grid_track_placement(column_start_placement_token_stream);
 | |
|     if (column_start_placement_token_stream.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     TokenStream row_end_placement_token_stream { row_end_placement_tokens };
 | |
|     auto row_end_style_value = parse_grid_track_placement(row_end_placement_token_stream);
 | |
|     if (row_end_placement_token_stream.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     TokenStream column_end_placement_token_stream { column_end_placement_tokens };
 | |
|     auto column_end_style_value = parse_grid_track_placement(column_end_placement_token_stream);
 | |
|     if (column_end_placement_token_stream.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     // If four <grid-line> values are specified, grid-row-start is set to the first value, grid-column-start
 | |
|     // is set to the second value, grid-row-end is set to the third value, and grid-column-end is set to the
 | |
|     // fourth value.
 | |
|     auto row_start = GridTrackPlacement::make_auto();
 | |
|     auto column_start = GridTrackPlacement::make_auto();
 | |
|     auto row_end = GridTrackPlacement::make_auto();
 | |
|     auto column_end = GridTrackPlacement::make_auto();
 | |
| 
 | |
|     if (row_start_style_value)
 | |
|         row_start = row_start_style_value.release_nonnull()->as_grid_track_placement().grid_track_placement();
 | |
| 
 | |
|     // When grid-column-start is omitted, if grid-row-start is a <custom-ident>, all four longhands are set to
 | |
|     // that value. Otherwise, it is set to auto.
 | |
|     if (column_start_style_value)
 | |
|         column_start = column_start_style_value.release_nonnull()->as_grid_track_placement().grid_track_placement();
 | |
|     else
 | |
|         column_start = row_start;
 | |
| 
 | |
|     // When grid-row-end is omitted, if grid-row-start is a <custom-ident>, grid-row-end is set to that
 | |
|     // <custom-ident>; otherwise, it is set to auto.
 | |
|     if (row_end_style_value)
 | |
|         row_end = row_end_style_value.release_nonnull()->as_grid_track_placement().grid_track_placement();
 | |
|     else
 | |
|         row_end = row_start;
 | |
| 
 | |
|     // When grid-column-end is omitted, if grid-column-start is a <custom-ident>, grid-column-end is set to
 | |
|     // that <custom-ident>; otherwise, it is set to auto.
 | |
|     if (column_end_style_value)
 | |
|         column_end = column_end_style_value.release_nonnull()->as_grid_track_placement().grid_track_placement();
 | |
|     else
 | |
|         column_end = column_start;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return ShorthandStyleValue::create(PropertyID::GridArea,
 | |
|         { PropertyID::GridRowStart, PropertyID::GridColumnStart, PropertyID::GridRowEnd, PropertyID::GridColumnEnd },
 | |
|         { GridTrackPlacementStyleValue::create(row_start), GridTrackPlacementStyleValue::create(column_start), GridTrackPlacementStyleValue::create(row_end), GridTrackPlacementStyleValue::create(column_end) });
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_grid_shorthand_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // <'grid-template'> |
 | |
|     // FIXME: <'grid-template-rows'> / [ auto-flow && dense? ] <'grid-auto-columns'>? |
 | |
|     // FIXME: [ auto-flow && dense? ] <'grid-auto-rows'>? / <'grid-template-columns'>
 | |
|     return parse_grid_track_size_list_shorthand_value(PropertyID::Grid, tokens);
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/css-grid-1/#grid-template-areas-property
 | |
| RefPtr<CSSStyleValue const> Parser::parse_grid_template_areas_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // none | <string>+
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return GridTemplateAreaStyleValue::create({});
 | |
| 
 | |
|     auto is_full_stop = [](u32 code_point) {
 | |
|         return code_point == '.';
 | |
|     };
 | |
| 
 | |
|     auto consume_while = [](Utf8CodePointIterator& code_points, AK::Function<bool(u32)> predicate) {
 | |
|         StringBuilder builder;
 | |
|         while (!code_points.done() && predicate(*code_points)) {
 | |
|             builder.append_code_point(*code_points);
 | |
|             ++code_points;
 | |
|         }
 | |
|         return MUST(builder.to_string());
 | |
|     };
 | |
| 
 | |
|     Vector<Vector<String>> grid_area_rows;
 | |
|     Optional<size_t> column_count;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     while (tokens.has_next_token() && tokens.next_token().is(Token::Type::String)) {
 | |
|         Vector<String> grid_area_columns;
 | |
|         auto string = tokens.consume_a_token().token().string().code_points();
 | |
|         auto code_points = string.begin();
 | |
| 
 | |
|         while (!code_points.done()) {
 | |
|             if (is_whitespace(*code_points)) {
 | |
|                 consume_while(code_points, is_whitespace);
 | |
|             } else if (is_full_stop(*code_points)) {
 | |
|                 consume_while(code_points, *is_full_stop);
 | |
|                 grid_area_columns.append("."_string);
 | |
|             } else if (is_ident_code_point(*code_points)) {
 | |
|                 auto token = consume_while(code_points, is_ident_code_point);
 | |
|                 grid_area_columns.append(move(token));
 | |
|             } else {
 | |
|                 return nullptr;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (grid_area_columns.is_empty())
 | |
|             return nullptr;
 | |
| 
 | |
|         if (column_count.has_value()) {
 | |
|             if (grid_area_columns.size() != column_count)
 | |
|                 return nullptr;
 | |
|         } else {
 | |
|             column_count = grid_area_columns.size();
 | |
|         }
 | |
| 
 | |
|         grid_area_rows.append(move(grid_area_columns));
 | |
|     }
 | |
| 
 | |
|     // FIXME: If a named grid area spans multiple grid cells, but those cells do not form a single filled-in rectangle, the declaration is invalid.
 | |
| 
 | |
|     transaction.commit();
 | |
|     return GridTemplateAreaStyleValue::create(grid_area_rows);
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_grid_auto_track_sizes(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // https://www.w3.org/TR/css-grid-2/#auto-tracks
 | |
|     // <track-size>+
 | |
|     Vector<Variant<ExplicitGridTrack, GridLineNames>> track_list;
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto const& token = tokens.consume_a_token();
 | |
|         auto track_sizing_function = parse_track_sizing_function(token);
 | |
|         if (!track_sizing_function.has_value()) {
 | |
|             transaction.commit();
 | |
|             return GridTrackSizeListStyleValue::make_auto();
 | |
|         }
 | |
|         // FIXME: Handle multiple repeat values (should combine them here, or remove
 | |
|         //        any other ones if the first one is auto-fill, etc.)
 | |
|         track_list.append(track_sizing_function.value());
 | |
|     }
 | |
|     transaction.commit();
 | |
|     return GridTrackSizeListStyleValue::create(GridTrackSizeList(move(track_list)));
 | |
| }
 | |
| 
 | |
| // https://www.w3.org/TR/css-grid-1/#grid-auto-flow-property
 | |
| RefPtr<GridAutoFlowStyleValue const> Parser::parse_grid_auto_flow_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // [ row | column ] || dense
 | |
|     if (!tokens.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     auto parse_axis = [&]() -> Optional<GridAutoFlowStyleValue::Axis> {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         auto const& token = tokens.consume_a_token();
 | |
|         if (!token.is(Token::Type::Ident))
 | |
|             return {};
 | |
|         auto const& ident = token.token().ident();
 | |
|         if (ident.equals_ignoring_ascii_case("row"sv)) {
 | |
|             transaction.commit();
 | |
|             return GridAutoFlowStyleValue::Axis::Row;
 | |
|         } else if (ident.equals_ignoring_ascii_case("column"sv)) {
 | |
|             transaction.commit();
 | |
|             return GridAutoFlowStyleValue::Axis::Column;
 | |
|         }
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     auto parse_dense = [&]() -> Optional<GridAutoFlowStyleValue::Dense> {
 | |
|         auto transaction = tokens.begin_transaction();
 | |
|         auto const& token = tokens.consume_a_token();
 | |
|         if (!token.is(Token::Type::Ident))
 | |
|             return {};
 | |
|         auto const& ident = token.token().ident();
 | |
|         if (ident.equals_ignoring_ascii_case("dense"sv)) {
 | |
|             transaction.commit();
 | |
|             return GridAutoFlowStyleValue::Dense::Yes;
 | |
|         }
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     Optional<GridAutoFlowStyleValue::Axis> axis;
 | |
|     Optional<GridAutoFlowStyleValue::Dense> dense;
 | |
|     if (axis = parse_axis(); axis.has_value()) {
 | |
|         dense = parse_dense();
 | |
|     } else if (dense = parse_dense(); dense.has_value()) {
 | |
|         axis = parse_axis();
 | |
|     }
 | |
| 
 | |
|     if (tokens.has_next_token())
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return GridAutoFlowStyleValue::create(axis.value_or(GridAutoFlowStyleValue::Axis::Row), dense.value_or(GridAutoFlowStyleValue::Dense::No));
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_grid_track_size_list(TokenStream<ComponentValue>& tokens, bool allow_separate_line_name_blocks)
 | |
| {
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return GridTrackSizeListStyleValue::make_none();
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     Vector<Variant<ExplicitGridTrack, GridLineNames>> track_list;
 | |
|     auto last_object_was_line_names = false;
 | |
|     while (tokens.has_next_token()) {
 | |
|         auto const& token = tokens.consume_a_token();
 | |
|         if (token.is_block()) {
 | |
|             if (last_object_was_line_names && !allow_separate_line_name_blocks) {
 | |
|                 transaction.commit();
 | |
|                 return GridTrackSizeListStyleValue::make_auto();
 | |
|             }
 | |
|             last_object_was_line_names = true;
 | |
|             Vector<String> line_names;
 | |
|             if (!token.block().is_square()) {
 | |
|                 transaction.commit();
 | |
|                 return GridTrackSizeListStyleValue::make_auto();
 | |
|             }
 | |
|             TokenStream block_tokens { token.block().value };
 | |
|             block_tokens.discard_whitespace();
 | |
|             while (block_tokens.has_next_token()) {
 | |
|                 auto const& current_block_token = block_tokens.consume_a_token();
 | |
|                 line_names.append(current_block_token.token().ident().to_string());
 | |
|                 block_tokens.discard_whitespace();
 | |
|             }
 | |
|             track_list.append(GridLineNames { move(line_names) });
 | |
|         } else {
 | |
|             last_object_was_line_names = false;
 | |
|             auto track_sizing_function = parse_track_sizing_function(token);
 | |
|             if (!track_sizing_function.has_value()) {
 | |
|                 transaction.commit();
 | |
|                 return GridTrackSizeListStyleValue::make_auto();
 | |
|             }
 | |
|             // FIXME: Handle multiple repeat values (should combine them here, or remove
 | |
|             // any other ones if the first one is auto-fill, etc.)
 | |
|             track_list.append(track_sizing_function.value());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     transaction.commit();
 | |
|     return GridTrackSizeListStyleValue::create(GridTrackSizeList(move(track_list)));
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_filter_value_list_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None))
 | |
|         return none;
 | |
| 
 | |
|     auto transaction = tokens.begin_transaction();
 | |
| 
 | |
|     // FIXME: <url>s are ignored for now
 | |
|     // <filter-value-list> = [ <filter-function> | <url> ]+
 | |
| 
 | |
|     enum class FilterToken {
 | |
|         // Color filters:
 | |
|         Brightness,
 | |
|         Contrast,
 | |
|         Grayscale,
 | |
|         Invert,
 | |
|         Opacity,
 | |
|         Saturate,
 | |
|         Sepia,
 | |
|         // Special filters:
 | |
|         Blur,
 | |
|         DropShadow,
 | |
|         HueRotate
 | |
|     };
 | |
| 
 | |
|     auto filter_token_to_operation = [&](auto filter) {
 | |
|         VERIFY(to_underlying(filter) < to_underlying(FilterToken::Blur));
 | |
|         return static_cast<Gfx::ColorFilter::Type>(filter);
 | |
|     };
 | |
| 
 | |
|     auto parse_filter_function_name = [&](auto name) -> Optional<FilterToken> {
 | |
|         if (name.equals_ignoring_ascii_case("blur"sv))
 | |
|             return FilterToken::Blur;
 | |
|         if (name.equals_ignoring_ascii_case("brightness"sv))
 | |
|             return FilterToken::Brightness;
 | |
|         if (name.equals_ignoring_ascii_case("contrast"sv))
 | |
|             return FilterToken::Contrast;
 | |
|         if (name.equals_ignoring_ascii_case("drop-shadow"sv))
 | |
|             return FilterToken::DropShadow;
 | |
|         if (name.equals_ignoring_ascii_case("grayscale"sv))
 | |
|             return FilterToken::Grayscale;
 | |
|         if (name.equals_ignoring_ascii_case("hue-rotate"sv))
 | |
|             return FilterToken::HueRotate;
 | |
|         if (name.equals_ignoring_ascii_case("invert"sv))
 | |
|             return FilterToken::Invert;
 | |
|         if (name.equals_ignoring_ascii_case("opacity"sv))
 | |
|             return FilterToken::Opacity;
 | |
|         if (name.equals_ignoring_ascii_case("saturate"sv))
 | |
|             return FilterToken::Saturate;
 | |
|         if (name.equals_ignoring_ascii_case("sepia"sv))
 | |
|             return FilterToken::Sepia;
 | |
|         return {};
 | |
|     };
 | |
| 
 | |
|     auto parse_filter_function = [&](auto filter_token, auto const& function_values) -> Optional<FilterFunction> {
 | |
|         TokenStream tokens { function_values };
 | |
|         tokens.discard_whitespace();
 | |
| 
 | |
|         auto if_no_more_tokens_return = [&](auto filter) -> Optional<FilterFunction> {
 | |
|             tokens.discard_whitespace();
 | |
|             if (tokens.has_next_token())
 | |
|                 return {};
 | |
|             return filter;
 | |
|         };
 | |
| 
 | |
|         if (filter_token == FilterToken::Blur) {
 | |
|             // blur( <length>? )
 | |
|             if (!tokens.has_next_token())
 | |
|                 return FilterOperation::Blur {};
 | |
|             auto blur_radius = parse_length(tokens);
 | |
|             tokens.discard_whitespace();
 | |
|             if (!blur_radius.has_value() || (!blur_radius->is_calculated() && blur_radius->value().raw_value() < 0))
 | |
|                 return {};
 | |
|             return if_no_more_tokens_return(FilterOperation::Blur { blur_radius.value() });
 | |
|         } else if (filter_token == FilterToken::DropShadow) {
 | |
|             if (!tokens.has_next_token())
 | |
|                 return {};
 | |
|             // drop-shadow( [ <color>? && <length>{2,3} ] )
 | |
|             // Note: The following code is a little awkward to allow the color to be before or after the lengths.
 | |
|             Optional<LengthOrCalculated> maybe_radius = {};
 | |
|             auto maybe_color = parse_color_value(tokens);
 | |
|             tokens.discard_whitespace();
 | |
|             auto x_offset = parse_length(tokens);
 | |
|             tokens.discard_whitespace();
 | |
|             if (!x_offset.has_value() || !tokens.has_next_token())
 | |
|                 return {};
 | |
| 
 | |
|             auto y_offset = parse_length(tokens);
 | |
|             tokens.discard_whitespace();
 | |
|             if (!y_offset.has_value())
 | |
|                 return {};
 | |
| 
 | |
|             if (tokens.has_next_token()) {
 | |
|                 maybe_radius = parse_length(tokens);
 | |
|                 tokens.discard_whitespace();
 | |
|                 if (!maybe_color && (!maybe_radius.has_value() || tokens.has_next_token())) {
 | |
|                     maybe_color = parse_color_value(tokens);
 | |
|                     if (!maybe_color)
 | |
|                         return {};
 | |
|                 } else if (!maybe_radius.has_value()) {
 | |
|                     return {};
 | |
|                 }
 | |
|             }
 | |
|             Optional<Color> color = {};
 | |
|             if (maybe_color)
 | |
|                 color = maybe_color->to_color({});
 | |
| 
 | |
|             return if_no_more_tokens_return(FilterOperation::DropShadow { x_offset.value(), y_offset.value(), maybe_radius, color });
 | |
|         } else if (filter_token == FilterToken::HueRotate) {
 | |
|             // hue-rotate( [ <angle> | <zero> ]? )
 | |
|             if (!tokens.has_next_token())
 | |
|                 return FilterOperation::HueRotate {};
 | |
| 
 | |
|             if (tokens.next_token().is(Token::Type::Number)) {
 | |
|                 // hue-rotate(0)
 | |
|                 auto number = tokens.consume_a_token().token().number();
 | |
|                 if (number.is_integer() && number.integer_value() == 0)
 | |
|                     return if_no_more_tokens_return(FilterOperation::HueRotate { FilterOperation::HueRotate::Zero {} });
 | |
|                 return {};
 | |
|             }
 | |
| 
 | |
|             if (auto angle = parse_angle(tokens); angle.has_value())
 | |
|                 return if_no_more_tokens_return(FilterOperation::HueRotate { angle.value() });
 | |
| 
 | |
|             return {};
 | |
|         } else {
 | |
|             // Simple filters:
 | |
|             // brightness( <number-percentage>? )
 | |
|             // contrast( <number-percentage>? )
 | |
|             // grayscale( <number-percentage>? )
 | |
|             // invert( <number-percentage>? )
 | |
|             // opacity( <number-percentage>? )
 | |
|             // sepia( <number-percentage>? )
 | |
|             // saturate( <number-percentage>? )
 | |
|             if (!tokens.has_next_token())
 | |
|                 return FilterOperation::Color { filter_token_to_operation(filter_token) };
 | |
|             auto amount = parse_number_percentage(tokens);
 | |
|             if (amount.has_value()) {
 | |
|                 if (amount->is_percentage() && amount->percentage().value() < 0)
 | |
|                     return {};
 | |
|                 if (amount->is_number() && amount->number().value() < 0)
 | |
|                     return {};
 | |
|                 if (first_is_one_of(filter_token, FilterToken::Grayscale, FilterToken::Invert, FilterToken::Opacity, FilterToken::Sepia)) {
 | |
|                     if (amount->is_percentage() && amount->percentage().value() > 100)
 | |
|                         amount = Percentage { 100 };
 | |
|                     if (amount->is_number() && amount->number().value() > 1)
 | |
|                         amount = Number { Number::Type::Integer, 1.0 };
 | |
|                 }
 | |
|             }
 | |
|             return if_no_more_tokens_return(FilterOperation::Color { filter_token_to_operation(filter_token), amount.value_or(Number { Number::Type::Integer, 1 }) });
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     Vector<FilterFunction> filter_value_list {};
 | |
| 
 | |
|     while (tokens.has_next_token()) {
 | |
|         tokens.discard_whitespace();
 | |
|         if (!tokens.has_next_token())
 | |
|             break;
 | |
|         auto& token = tokens.consume_a_token();
 | |
|         if (!token.is_function())
 | |
|             return nullptr;
 | |
|         auto filter_token = parse_filter_function_name(token.function().name);
 | |
|         if (!filter_token.has_value())
 | |
|             return nullptr;
 | |
| 
 | |
|         auto context_guard = push_temporary_value_parsing_context(FunctionContext { token.function().name });
 | |
|         auto filter_function = parse_filter_function(*filter_token, token.function().value);
 | |
|         if (!filter_function.has_value())
 | |
|             return nullptr;
 | |
|         filter_value_list.append(*filter_function);
 | |
|     }
 | |
| 
 | |
|     if (filter_value_list.is_empty())
 | |
|         return nullptr;
 | |
| 
 | |
|     transaction.commit();
 | |
|     return FilterValueListStyleValue::create(move(filter_value_list));
 | |
| }
 | |
| 
 | |
| RefPtr<CSSStyleValue const> Parser::parse_contain_value(TokenStream<ComponentValue>& tokens)
 | |
| {
 | |
|     // none | strict | content | [ [size | inline-size] || layout || style || paint ]
 | |
|     auto transaction = tokens.begin_transaction();
 | |
|     tokens.discard_whitespace();
 | |
| 
 | |
|     // none
 | |
|     if (auto none = parse_all_as_single_keyword_value(tokens, Keyword::None)) {
 | |
|         transaction.commit();
 | |
|         return none;
 | |
|     }
 | |
| 
 | |
|     // strict
 | |
|     if (auto strict = parse_all_as_single_keyword_value(tokens, Keyword::Strict)) {
 | |
|         transaction.commit();
 | |
|         return strict;
 | |
|     }
 | |
| 
 | |
|     // content
 | |
|     if (auto content = parse_all_as_single_keyword_value(tokens, Keyword::Content)) {
 | |
|         transaction.commit();
 | |
|         return content;
 | |
|     }
 | |
| 
 | |
|     // [ [size | inline-size] || layout || style || paint ]
 | |
|     if (!tokens.has_next_token())
 | |
|         return {};
 | |
| 
 | |
|     bool has_size = false;
 | |
|     bool has_layout = false;
 | |
|     bool has_style = false;
 | |
|     bool has_paint = false;
 | |
| 
 | |
|     StyleValueVector containments;
 | |
|     while (tokens.has_next_token()) {
 | |
|         tokens.discard_whitespace();
 | |
|         auto keyword = parse_keyword_value(tokens);
 | |
|         if (!keyword)
 | |
|             return {};
 | |
|         switch (keyword->to_keyword()) {
 | |
|         case Keyword::Size:
 | |
|         case Keyword::InlineSize:
 | |
|             if (has_size)
 | |
|                 return {};
 | |
|             has_size = true;
 | |
|             break;
 | |
|         case Keyword::Layout:
 | |
|             if (has_layout)
 | |
|                 return {};
 | |
|             has_layout = true;
 | |
|             break;
 | |
|         case Keyword::Style:
 | |
|             if (has_style)
 | |
|                 return {};
 | |
|             has_style = true;
 | |
|             break;
 | |
|         case Keyword::Paint:
 | |
|             if (has_paint)
 | |
|                 return {};
 | |
|             has_paint = true;
 | |
|             break;
 | |
|         default:
 | |
|             return {};
 | |
|         }
 | |
|         containments.append(*keyword);
 | |
|     }
 | |
|     transaction.commit();
 | |
| 
 | |
|     return StyleValueList::create(move(containments), StyleValueList::Separator::Space);
 | |
| }
 | |
| 
 | |
| }
 |