mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-04 07:10:57 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			176 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			176 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org>
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#include <LibJS/Runtime/Realm.h>
 | 
						|
#include <LibWeb/CSS/Parser/Parser.h>
 | 
						|
#include <LibWeb/CSS/Supports.h>
 | 
						|
 | 
						|
namespace Web::CSS {
 | 
						|
 | 
						|
static void indent(StringBuilder& builder, int levels)
 | 
						|
{
 | 
						|
    for (int i = 0; i < levels; i++)
 | 
						|
        builder.append("  "sv);
 | 
						|
}
 | 
						|
 | 
						|
Supports::Supports(JS::Realm& realm, NonnullOwnPtr<Condition>&& condition)
 | 
						|
    : m_condition(move(condition))
 | 
						|
{
 | 
						|
    m_matches = m_condition->evaluate(realm);
 | 
						|
}
 | 
						|
 | 
						|
bool Supports::Condition::evaluate(JS::Realm& realm) const
 | 
						|
{
 | 
						|
    switch (type) {
 | 
						|
    case Type::Not:
 | 
						|
        return !children.first().evaluate(realm);
 | 
						|
    case Type::And:
 | 
						|
        for (auto& child : children) {
 | 
						|
            if (!child.evaluate(realm))
 | 
						|
                return false;
 | 
						|
        }
 | 
						|
        return true;
 | 
						|
    case Type::Or:
 | 
						|
        for (auto& child : children) {
 | 
						|
            if (child.evaluate(realm))
 | 
						|
                return true;
 | 
						|
        }
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
    VERIFY_NOT_REACHED();
 | 
						|
}
 | 
						|
 | 
						|
bool Supports::InParens::evaluate(JS::Realm& realm) const
 | 
						|
{
 | 
						|
    return value.visit(
 | 
						|
        [&](NonnullOwnPtr<Condition> const& condition) {
 | 
						|
            return condition->evaluate(realm);
 | 
						|
        },
 | 
						|
        [&](Feature const& feature) {
 | 
						|
            return feature.evaluate(realm);
 | 
						|
        },
 | 
						|
        [&](GeneralEnclosed const&) {
 | 
						|
            return false;
 | 
						|
        });
 | 
						|
}
 | 
						|
 | 
						|
bool Supports::Declaration::evaluate(JS::Realm& realm) const
 | 
						|
{
 | 
						|
    auto style_property = parse_css_supports_condition(Parser::ParsingContext { realm }, declaration);
 | 
						|
    return style_property.has_value();
 | 
						|
}
 | 
						|
 | 
						|
bool Supports::Selector::evaluate(JS::Realm& realm) const
 | 
						|
{
 | 
						|
    auto style_property = parse_selector(Parser::ParsingContext { realm }, selector);
 | 
						|
    return style_property.has_value();
 | 
						|
}
 | 
						|
 | 
						|
bool Supports::Feature::evaluate(JS::Realm& realm) const
 | 
						|
{
 | 
						|
    return value.visit(
 | 
						|
        [&](Declaration const& declaration) {
 | 
						|
            return declaration.evaluate(realm);
 | 
						|
        },
 | 
						|
        [&](Selector const& selector) {
 | 
						|
            return selector.evaluate(realm);
 | 
						|
        });
 | 
						|
}
 | 
						|
 | 
						|
String Supports::Declaration::to_string() const
 | 
						|
{
 | 
						|
    return MUST(String::formatted("({})", declaration));
 | 
						|
}
 | 
						|
 | 
						|
String Supports::Selector::to_string() const
 | 
						|
{
 | 
						|
    return MUST(String::formatted("selector({})", selector));
 | 
						|
}
 | 
						|
 | 
						|
String Supports::Feature::to_string() const
 | 
						|
{
 | 
						|
    return value.visit([](auto& it) { return it.to_string(); });
 | 
						|
}
 | 
						|
 | 
						|
String Supports::InParens::to_string() const
 | 
						|
{
 | 
						|
    return value.visit(
 | 
						|
        [](NonnullOwnPtr<Condition> const& condition) { return MUST(String::formatted("({})", condition->to_string())); },
 | 
						|
        [](Supports::Feature const& it) { return it.to_string(); },
 | 
						|
        [](GeneralEnclosed const& it) { return it.to_string(); });
 | 
						|
}
 | 
						|
 | 
						|
String Supports::Condition::to_string() const
 | 
						|
{
 | 
						|
    switch (type) {
 | 
						|
    case Type::Not:
 | 
						|
        return MUST(String::formatted("not {}", children.first().to_string()));
 | 
						|
    case Type::And:
 | 
						|
        return MUST(String::join(" and "sv, children));
 | 
						|
    case Type::Or:
 | 
						|
        return MUST(String::join(" or "sv, children));
 | 
						|
    }
 | 
						|
    VERIFY_NOT_REACHED();
 | 
						|
}
 | 
						|
 | 
						|
String Supports::to_string() const
 | 
						|
{
 | 
						|
    return m_condition->to_string();
 | 
						|
}
 | 
						|
 | 
						|
void Supports::Declaration::dump(StringBuilder& builder, int indent_levels) const
 | 
						|
{
 | 
						|
    indent(builder, indent_levels);
 | 
						|
    builder.appendff("Declaration: {}\n", declaration);
 | 
						|
}
 | 
						|
 | 
						|
void Supports::Selector::dump(StringBuilder& builder, int indent_levels) const
 | 
						|
{
 | 
						|
    indent(builder, indent_levels);
 | 
						|
    builder.appendff("Selector: {}\n", selector);
 | 
						|
}
 | 
						|
 | 
						|
void Supports::Feature::dump(StringBuilder& builder, int indent_levels) const
 | 
						|
{
 | 
						|
    value.visit([&](auto& it) { it.dump(builder, indent_levels); });
 | 
						|
}
 | 
						|
 | 
						|
void Supports::InParens::dump(StringBuilder& builder, int indent_levels) const
 | 
						|
{
 | 
						|
    value.visit(
 | 
						|
        [&](NonnullOwnPtr<Condition> const& condition) { condition->dump(builder, indent_levels); },
 | 
						|
        [&](Supports::Feature const& it) { it.dump(builder, indent_levels); },
 | 
						|
        [&](GeneralEnclosed const& it) {
 | 
						|
            indent(builder, indent_levels);
 | 
						|
            builder.appendff("GeneralEnclosed: {}\n", it.to_string());
 | 
						|
        });
 | 
						|
}
 | 
						|
 | 
						|
void Supports::Condition::dump(StringBuilder& builder, int indent_levels) const
 | 
						|
{
 | 
						|
    indent(builder, indent_levels);
 | 
						|
    StringView type_name = [](Type type) {
 | 
						|
        switch (type) {
 | 
						|
        case Type::And:
 | 
						|
            return "AND"sv;
 | 
						|
        case Type::Or:
 | 
						|
            return "OR"sv;
 | 
						|
        case Type::Not:
 | 
						|
            return "NOT"sv;
 | 
						|
        }
 | 
						|
        VERIFY_NOT_REACHED();
 | 
						|
    }(type);
 | 
						|
    builder.appendff("Condition: {}\n", type_name);
 | 
						|
    for (auto const& child : children)
 | 
						|
        child.dump(builder, indent_levels + 1);
 | 
						|
}
 | 
						|
 | 
						|
void Supports::dump(StringBuilder& builder, int indent_levels) const
 | 
						|
{
 | 
						|
    m_condition->dump(builder, indent_levels);
 | 
						|
}
 | 
						|
 | 
						|
}
 |