mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 21:02:44 +00:00 
			
		
		
		
	 6034fc0ee6
			
		
	
	
		6034fc0ee6
		
	
	
	
	
		
			
			Parsing this pattern from CSS tokens turns out to be slightly crazy, but thankfully well documented in the spec. The spec lists the cases in order of simple -> complicated, but this would cause problems in code, since `<n-dimension> <signed-.integer>` would never by reached, as `<n-dimension>` comes before. Instead, I have grouped them by their first token. Also renamed the NthChildPattern class to ANPlusBPattern, to match spec terminology.
		
			
				
	
	
		
			225 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			225 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020-2021, the SerenityOS developers.
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/String.h>
 | |
| #include <LibWeb/CSS/Parser/Token.h>
 | |
| 
 | |
| namespace Web::CSS {
 | |
| 
 | |
| String Token::to_debug_string() const
 | |
| {
 | |
|     StringBuilder builder;
 | |
| 
 | |
|     switch (m_type) {
 | |
|     case Type::Invalid:
 | |
|         VERIFY_NOT_REACHED();
 | |
| 
 | |
|     case Type::EndOfFile:
 | |
|         builder.append("__EOF__");
 | |
|         break;
 | |
|     case Type::Ident:
 | |
|         builder.append("Identifier: ");
 | |
|         builder.append(m_value.to_string());
 | |
|         return builder.to_string();
 | |
|     case Type::Function:
 | |
|         builder.append("Function");
 | |
|         break;
 | |
|     case Type::AtKeyword:
 | |
|         builder.append("@");
 | |
|         break;
 | |
|     case Type::Hash:
 | |
|         builder.append("Hash: ");
 | |
|         builder.append(m_value.to_string());
 | |
|         return builder.to_string();
 | |
|     case Type::String:
 | |
|         builder.append("String: ");
 | |
|         builder.append(m_value.to_string());
 | |
|         return builder.to_string();
 | |
|     case Type::BadString:
 | |
|         builder.append("Invalid String");
 | |
|         break;
 | |
|     case Type::Url:
 | |
|         builder.append("Url");
 | |
|         break;
 | |
|     case Type::BadUrl:
 | |
|         builder.append("Invalid Url");
 | |
|         break;
 | |
|     case Type::Delim:
 | |
|         builder.append("Delimiter: ");
 | |
|         builder.append(m_value.to_string());
 | |
|         return builder.to_string();
 | |
|     case Type::Number:
 | |
|         builder.append("Number: ");
 | |
|         builder.append(m_value.to_string());
 | |
|         builder.append(m_number_type == NumberType::Integer ? " (int)" : " (float)");
 | |
|         return builder.to_string();
 | |
|     case Type::Percentage:
 | |
|         builder.append("Percentage: ");
 | |
|         builder.append(m_value.to_string());
 | |
|         builder.append(m_unit.to_string());
 | |
|         return builder.to_string();
 | |
|     case Type::Dimension:
 | |
|         builder.append("Dimension: ");
 | |
|         builder.append(m_value.to_string());
 | |
|         builder.append(m_unit.to_string());
 | |
|         return builder.to_string();
 | |
|     case Type::Whitespace:
 | |
|         builder.append("Whitespace");
 | |
|         break;
 | |
|     case Type::CDO:
 | |
|         builder.append("CDO");
 | |
|         break;
 | |
|     case Type::CDC:
 | |
|         builder.append("CDC");
 | |
|         break;
 | |
|     case Type::Colon:
 | |
|         builder.append(":");
 | |
|         break;
 | |
|     case Type::Semicolon:
 | |
|         builder.append(";");
 | |
|         break;
 | |
|     case Type::Comma:
 | |
|         builder.append(",");
 | |
|         break;
 | |
|     case Type::OpenSquare:
 | |
|         builder.append("[");
 | |
|         break;
 | |
|     case Type::CloseSquare:
 | |
|         builder.append("]");
 | |
|         break;
 | |
|     case Type::OpenParen:
 | |
|         builder.append("(");
 | |
|         break;
 | |
|     case Type::CloseParen:
 | |
|         builder.append(")");
 | |
|         break;
 | |
|     case Type::OpenCurly:
 | |
|         builder.append("{");
 | |
|         break;
 | |
|     case Type::CloseCurly:
 | |
|         builder.append("}");
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     if (m_value.is_empty()) {
 | |
|         return builder.to_string();
 | |
|     }
 | |
| 
 | |
|     builder.append(" ");
 | |
| 
 | |
|     builder.append(" { value: '");
 | |
|     builder.append(m_value.to_string());
 | |
| 
 | |
|     if (m_type == Token::Type::Hash) {
 | |
|         builder.append("', hash_type: '");
 | |
|         if (m_hash_type == Token::HashType::Unrestricted) {
 | |
|             builder.append("Unrestricted");
 | |
|         } else {
 | |
|             builder.append("Id");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (m_type == Token::Type::Number) {
 | |
|         builder.append("', number_type: '");
 | |
|         if (m_number_type == Token::NumberType::Integer) {
 | |
|             builder.append("Integer");
 | |
|         } else {
 | |
|             builder.append("Number");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (m_type == Token::Type::Dimension) {
 | |
|         builder.append("', number_type: '");
 | |
|         if (m_number_type == Token::NumberType::Integer) {
 | |
|             builder.append("Integer");
 | |
|         } else {
 | |
|             builder.append("Number");
 | |
|         }
 | |
| 
 | |
|         builder.append("', unit: '");
 | |
|         builder.append(m_unit.to_string());
 | |
|     }
 | |
| 
 | |
|     builder.append("' }");
 | |
|     return builder.to_string();
 | |
| }
 | |
| 
 | |
| Token::Type Token::mirror_variant() const
 | |
| {
 | |
|     if (is(Token::Type::OpenCurly)) {
 | |
|         return Type::CloseCurly;
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::OpenSquare)) {
 | |
|         return Type::CloseSquare;
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::OpenParen)) {
 | |
|         return Type::CloseParen;
 | |
|     }
 | |
| 
 | |
|     return Type::Invalid;
 | |
| }
 | |
| 
 | |
| String Token::bracket_string() const
 | |
| {
 | |
|     if (is(Token::Type::OpenCurly)) {
 | |
|         return "{";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::CloseCurly)) {
 | |
|         return "}";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::OpenSquare)) {
 | |
|         return "[";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::CloseSquare)) {
 | |
|         return "]";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::OpenParen)) {
 | |
|         return "(";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::CloseParen)) {
 | |
|         return ")";
 | |
|     }
 | |
| 
 | |
|     return "";
 | |
| }
 | |
| 
 | |
| String Token::bracket_mirror_string() const
 | |
| {
 | |
|     if (is(Token::Type::OpenCurly)) {
 | |
|         return "}";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::CloseCurly)) {
 | |
|         return "{";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::OpenSquare)) {
 | |
|         return "]";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::CloseSquare)) {
 | |
|         return "[";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::OpenParen)) {
 | |
|         return ")";
 | |
|     }
 | |
| 
 | |
|     if (is(Token::Type::CloseParen)) {
 | |
|         return "(";
 | |
|     }
 | |
| 
 | |
|     return "";
 | |
| }
 | |
| 
 | |
| }
 |