1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-10-23 07:22:06 +00:00
serenity/Userland/Libraries/LibWeb/CSS/Parser/StyleRules.cpp
Sam Atkins 23dc0dac88 LibWeb: Parse and resolve UnresolvedStyleValues
If a property is custom or contains a `var()` reference, it cannot be
parsed into a proper StyleValue immediately, so we store it as an
UnresolvedStyleValue until the property is compute. Then, at compute
time, we resolve them by expanding out any `var()` references, and
parsing the result.

The implementation here is very naive, and involves copying the
UnresolvedStyleValue's tree of StyleComponentValueRules while copying
the contents of any `var()`s it finds along the way. This is quite an
expensive operation to do every time that the style is computed.
2021-12-09 21:30:31 +01:00

191 lines
4.7 KiB
C++

/*
* Copyright (c) 2020-2021, the SerenityOS developers.
* Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibWeb/CSS/Parser/DeclarationOrAtRule.h>
#include <LibWeb/CSS/Parser/StyleBlockRule.h>
#include <LibWeb/CSS/Parser/StyleComponentValueRule.h>
#include <LibWeb/CSS/Parser/StyleDeclarationRule.h>
#include <LibWeb/CSS/Parser/StyleFunctionRule.h>
#include <LibWeb/CSS/Parser/StyleRule.h>
#include <LibWeb/CSS/Serialize.h>
namespace Web::CSS {
DeclarationOrAtRule::DeclarationOrAtRule(RefPtr<StyleRule> at)
: m_type(DeclarationType::At)
, m_at(move(at))
{
}
DeclarationOrAtRule::DeclarationOrAtRule(StyleDeclarationRule declaration)
: m_type(DeclarationType::Declaration)
, m_declaration(move(declaration))
{
}
DeclarationOrAtRule::~DeclarationOrAtRule() { }
StyleRule::StyleRule(StyleRule::Type type)
: m_type(type)
{
}
StyleRule::~StyleRule() { }
StyleBlockRule::StyleBlockRule() { }
StyleBlockRule::~StyleBlockRule() { }
StyleComponentValueRule::StyleComponentValueRule(Token token)
: m_type(StyleComponentValueRule::ComponentType::Token)
, m_token(token)
{
}
StyleComponentValueRule::StyleComponentValueRule(NonnullRefPtr<StyleFunctionRule> function)
: m_type(StyleComponentValueRule::ComponentType::Function)
, m_function(function)
{
}
StyleComponentValueRule::StyleComponentValueRule(NonnullRefPtr<StyleBlockRule> block)
: m_type(StyleComponentValueRule::ComponentType::Block)
, m_block(block)
{
}
StyleComponentValueRule::~StyleComponentValueRule() { }
StyleDeclarationRule::StyleDeclarationRule() { }
StyleDeclarationRule::~StyleDeclarationRule() { }
StyleFunctionRule::StyleFunctionRule(String name)
: m_name(move(name))
{
}
StyleFunctionRule::StyleFunctionRule(String name, Vector<StyleComponentValueRule>&& values)
: m_name(move(name))
, m_values(move(values))
{
}
StyleFunctionRule::~StyleFunctionRule() { }
template<class SeparatorType, class CollectionType>
void append_with_to_string(StringBuilder& builder, SeparatorType& separator, CollectionType& collection)
{
bool first = true;
for (auto& item : collection) {
if (first)
first = false;
else
builder.append(separator);
builder.append(item.to_string());
}
}
String DeclarationOrAtRule::to_string() const
{
StringBuilder builder;
switch (m_type) {
default:
case DeclarationType::At:
builder.append(m_at->to_string());
break;
case DeclarationType::Declaration:
builder.append(m_declaration.to_string());
break;
}
return builder.to_string();
}
String StyleRule::to_string() const
{
StringBuilder builder;
if (m_type == Type::At) {
builder.append("@");
serialize_an_identifier(builder, m_name);
}
append_with_to_string(builder, " ", m_prelude);
if (m_block)
builder.append(m_block->to_string());
else
builder.append(';');
return builder.to_string();
}
String StyleBlockRule::to_string() const
{
StringBuilder builder;
builder.append(m_token.bracket_string());
append_with_to_string(builder, " ", m_values);
builder.append(m_token.bracket_mirror_string());
return builder.to_string();
}
String StyleComponentValueRule::to_string() const
{
switch (m_type) {
case StyleComponentValueRule::ComponentType::Token:
return m_token.to_string();
case StyleComponentValueRule::ComponentType::Function:
return m_function->to_string();
case StyleComponentValueRule::ComponentType::Block:
return m_block->to_string();
default:
VERIFY_NOT_REACHED();
}
}
String StyleComponentValueRule::to_debug_string() const
{
StringBuilder builder;
switch (m_type) {
case ComponentType::Token:
builder.append("Token: ");
builder.append(m_token.to_debug_string());
break;
case ComponentType::Function:
builder.append("Function: ");
builder.append(m_function->to_string());
break;
case ComponentType::Block:
builder.append("Block: ");
builder.append(m_block->to_string());
break;
}
return builder.to_string();
}
String StyleDeclarationRule::to_string() const
{
StringBuilder builder;
serialize_an_identifier(builder, m_name);
builder.append(": ");
append_with_to_string(builder, " ", m_values);
if (m_important)
builder.append(" !important");
return builder.to_string();
}
String StyleFunctionRule::to_string() const
{
StringBuilder builder;
serialize_an_identifier(builder, m_name);
builder.append("(");
append_with_to_string(builder, " ", m_values);
builder.append(")");
return builder.to_string();
}
}