| 
									
										
										
										
											2019-06-21 20:54:13 +02:00
										 |  |  | #include <LibHTML/CSS/StyleSheet.h>
 | 
					
						
							|  |  |  | #include <LibHTML/Parser/CSSParser.h>
 | 
					
						
							|  |  |  | #include <ctype.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  | #define PARSE_ASSERT(x) \
 | 
					
						
							|  |  |  |     if (!(x)) { \ | 
					
						
							|  |  |  |         dbg() << "CSS PARSER ASSERTION FAILED: " << #x; \ | 
					
						
							|  |  |  |         dbg() << "At character# " << index << " in CSS: _" << css << "_"; \ | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); \ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 22:18:19 +02:00
										 |  |  | static Optional<Color> parse_css_color(const StringView& view) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto color = Color::from_string(view); | 
					
						
							|  |  |  |     if (color.has_value()) | 
					
						
							|  |  |  |         return color; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // FIXME: Parse all valid color strings :^)
 | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 15:49:16 +02:00
										 |  |  | NonnullRefPtr<StyleValue> parse_css_value(const StringView& view) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     String string(view); | 
					
						
							|  |  |  |     bool ok; | 
					
						
							|  |  |  |     int as_int = string.to_int(ok); | 
					
						
							|  |  |  |     if (ok) | 
					
						
							|  |  |  |         return LengthStyleValue::create(Length(as_int, Length::Type::Absolute)); | 
					
						
							|  |  |  |     unsigned as_uint = string.to_uint(ok); | 
					
						
							|  |  |  |     if (ok) | 
					
						
							|  |  |  |         return LengthStyleValue::create(Length(as_uint, Length::Type::Absolute)); | 
					
						
							| 
									
										
										
										
											2019-07-08 07:14:23 +02:00
										 |  |  |     if (string == "inherit") | 
					
						
							|  |  |  |         return InheritStyleValue::create(); | 
					
						
							|  |  |  |     if (string == "initial") | 
					
						
							|  |  |  |         return InitialStyleValue::create(); | 
					
						
							| 
									
										
										
										
											2019-07-04 15:49:16 +02:00
										 |  |  |     if (string == "auto") | 
					
						
							|  |  |  |         return LengthStyleValue::create(Length()); | 
					
						
							| 
									
										
										
										
											2019-09-28 22:18:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto color = parse_css_color(view); | 
					
						
							|  |  |  |     if (color.has_value()) | 
					
						
							|  |  |  |         return ColorStyleValue::create(color.value()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 15:49:16 +02:00
										 |  |  |     return StringStyleValue::create(string); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  | class CSSParser { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     CSSParser(const StringView& input) | 
					
						
							|  |  |  |         : css(input) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     char peek() const | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  |         if (index < css.length()) | 
					
						
							|  |  |  |             return css[index]; | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     char consume_specific(char ch) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         PARSE_ASSERT(peek() == ch); | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         ++index; | 
					
						
							|  |  |  |         return ch; | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     char consume_one() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         return css[index++]; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     void consume_whitespace() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         while (isspace(peek())) | 
					
						
							|  |  |  |             ++index; | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     bool is_valid_selector_char(char ch) const | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         return isalnum(ch) || ch == '-' || ch == '_' || ch == '(' || ch == ')' || ch == '@'; | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |     Optional<Selector::Component> parse_selector_component() | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         consume_whitespace(); | 
					
						
							|  |  |  |         Selector::Component::Type type; | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         Selector::Component::Relation relation = Selector::Component::Relation::None; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (peek() == '{') | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (peek() == '>') { | 
					
						
							|  |  |  |             relation = Selector::Component::Relation::ImmediateChild; | 
					
						
							|  |  |  |             consume_one(); | 
					
						
							|  |  |  |             consume_whitespace(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-29 17:26:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (peek() == '.') { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |             type = Selector::Component::Type::Class; | 
					
						
							| 
									
										
										
										
											2019-09-29 17:26:05 +02:00
										 |  |  |             consume_one(); | 
					
						
							|  |  |  |         } else if (peek() == '#') { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |             type = Selector::Component::Type::Id; | 
					
						
							| 
									
										
										
										
											2019-09-29 17:26:05 +02:00
										 |  |  |             consume_one(); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |             type = Selector::Component::Type::TagName; | 
					
						
							| 
									
										
										
										
											2019-09-29 17:26:05 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while (is_valid_selector_char(peek())) | 
					
						
							|  |  |  |             buffer.append(consume_one()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         PARSE_ASSERT(!buffer.is_null()); | 
					
						
							|  |  |  |         Selector::Component component { type, relation, String::copy(buffer) }; | 
					
						
							|  |  |  |         buffer.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (peek() == '[') { | 
					
						
							|  |  |  |             // FIXME: Implement attribute selectors.
 | 
					
						
							|  |  |  |             while (peek() != ']') { | 
					
						
							|  |  |  |                 consume_one(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             consume_one(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (peek() == ':') { | 
					
						
							|  |  |  |             // FIXME: Implement pseudo stuff.
 | 
					
						
							|  |  |  |             consume_one(); | 
					
						
							|  |  |  |             if (peek() == ':') | 
					
						
							|  |  |  |                 consume_one(); | 
					
						
							|  |  |  |             while (is_valid_selector_char(peek())) | 
					
						
							|  |  |  |                 consume_one(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         return component; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |     void parse_selector() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         Vector<Selector::Component> components; | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (;;) { | 
					
						
							|  |  |  |             auto component = parse_selector_component(); | 
					
						
							|  |  |  |             if (component.has_value()) | 
					
						
							|  |  |  |                 components.append(component.value()); | 
					
						
							|  |  |  |             consume_whitespace(); | 
					
						
							|  |  |  |             if (peek() == ',' || peek() == '{') | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         current_rule.selectors.append(Selector(move(components))); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     void parse_selector_list() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         for (;;) { | 
					
						
							|  |  |  |             parse_selector(); | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  |             consume_whitespace(); | 
					
						
							|  |  |  |             if (peek() == ',') { | 
					
						
							|  |  |  |                 consume_one(); | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |             if (peek() == '{') | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     bool is_valid_property_name_char(char ch) const | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  |         return !isspace(ch) && ch != ':'; | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     bool is_valid_property_value_char(char ch) const | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         return ch != '!' && ch != ';'; | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |     Optional<StyleProperty> parse_property() | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         consume_whitespace(); | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         if (peek() == ';') { | 
					
						
							|  |  |  |             consume_one(); | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  |         buffer.clear(); | 
					
						
							|  |  |  |         while (is_valid_property_name_char(peek())) | 
					
						
							|  |  |  |             buffer.append(consume_one()); | 
					
						
							|  |  |  |         auto property_name = String::copy(buffer); | 
					
						
							|  |  |  |         buffer.clear(); | 
					
						
							|  |  |  |         consume_whitespace(); | 
					
						
							|  |  |  |         consume_specific(':'); | 
					
						
							|  |  |  |         consume_whitespace(); | 
					
						
							|  |  |  |         while (is_valid_property_value_char(peek())) | 
					
						
							|  |  |  |             buffer.append(consume_one()); | 
					
						
							|  |  |  |         auto property_value = String::copy(buffer); | 
					
						
							|  |  |  |         buffer.clear(); | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         consume_whitespace(); | 
					
						
							|  |  |  |         bool is_important = false; | 
					
						
							|  |  |  |         if (peek() == '!') { | 
					
						
							|  |  |  |             consume_specific('!'); | 
					
						
							|  |  |  |             consume_specific('i'); | 
					
						
							|  |  |  |             consume_specific('m'); | 
					
						
							|  |  |  |             consume_specific('p'); | 
					
						
							|  |  |  |             consume_specific('o'); | 
					
						
							|  |  |  |             consume_specific('r'); | 
					
						
							|  |  |  |             consume_specific('t'); | 
					
						
							|  |  |  |             consume_specific('a'); | 
					
						
							|  |  |  |             consume_specific('n'); | 
					
						
							|  |  |  |             consume_specific('t'); | 
					
						
							|  |  |  |             consume_whitespace(); | 
					
						
							|  |  |  |             is_important = true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  |         consume_specific(';'); | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |         return StyleProperty { property_name, parse_css_value(property_value), is_important }; | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     void parse_declaration() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         for (;;) { | 
					
						
							| 
									
										
										
										
											2019-10-06 09:28:10 +02:00
										 |  |  |             auto property = parse_property(); | 
					
						
							|  |  |  |             if (property.has_value()) | 
					
						
							|  |  |  |                 current_rule.properties.append(property.value()); | 
					
						
							| 
									
										
										
										
											2019-06-22 21:48:21 +02:00
										 |  |  |             consume_whitespace(); | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |             if (peek() == '}') | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     void parse_rule() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         parse_selector_list(); | 
					
						
							|  |  |  |         consume_specific('{'); | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |         parse_declaration(); | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |         consume_specific('}'); | 
					
						
							| 
									
										
										
										
											2019-09-30 20:06:17 +02:00
										 |  |  |         rules.append(StyleRule::create(move(current_rule.selectors), StyleDeclaration::create(move(current_rule.properties)))); | 
					
						
							| 
									
										
										
										
											2019-09-29 17:26:05 +02:00
										 |  |  |         consume_whitespace(); | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NonnullRefPtr<StyleSheet> parse_sheet() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         while (index < css.length()) { | 
					
						
							|  |  |  |             parse_rule(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |         return StyleSheet::create(move(rules)); | 
					
						
							| 
									
										
										
										
											2019-06-22 09:27:39 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 20:24:38 +02:00
										 |  |  |     NonnullRefPtr<StyleDeclaration> parse_standalone_declaration() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         consume_whitespace(); | 
					
						
							|  |  |  |         for (;;) { | 
					
						
							|  |  |  |             parse_property(); | 
					
						
							|  |  |  |             consume_whitespace(); | 
					
						
							|  |  |  |             if (!peek()) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return StyleDeclaration::create(move(current_rule.properties)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     NonnullRefPtrVector<StyleRule> rules; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     enum class State { | 
					
						
							|  |  |  |         Free, | 
					
						
							|  |  |  |         InSelectorComponent, | 
					
						
							|  |  |  |         InPropertyName, | 
					
						
							|  |  |  |         InPropertyValue, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct CurrentRule { | 
					
						
							|  |  |  |         Vector<Selector> selectors; | 
					
						
							|  |  |  |         Vector<StyleProperty> properties; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CurrentRule current_rule; | 
					
						
							|  |  |  |     Vector<char> buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int index = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     String css; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NonnullRefPtr<StyleSheet> parse_css(const String& css) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CSSParser parser(css); | 
					
						
							|  |  |  |     return parser.parse_sheet(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NonnullRefPtr<StyleDeclaration> parse_css_declaration(const String& css) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CSSParser parser(css); | 
					
						
							|  |  |  |     return parser.parse_standalone_declaration(); | 
					
						
							| 
									
										
										
										
											2019-06-21 20:54:13 +02:00
										 |  |  | } |