mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-03 23:00:58 +00:00 
			
		
		
		
	Currently, `evaluate()` recalculates whether the MediaQuery matches or not, and stores it in `m_matches`, which users can query using `matches()`. This allows us to know when the match-state changes, which is required to fire MediaQueryList's change event.
		
			
				
	
	
		
			230 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			230 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#include <LibWeb/CSS/MediaQuery.h>
 | 
						|
#include <LibWeb/DOM/Document.h>
 | 
						|
#include <LibWeb/DOM/Window.h>
 | 
						|
 | 
						|
namespace Web::CSS {
 | 
						|
 | 
						|
NonnullRefPtr<MediaQuery> MediaQuery::create_not_all()
 | 
						|
{
 | 
						|
    auto media_query = new MediaQuery;
 | 
						|
    media_query->m_negated = true;
 | 
						|
    media_query->m_media_type = MediaType::All;
 | 
						|
 | 
						|
    return adopt_ref(*media_query);
 | 
						|
}
 | 
						|
 | 
						|
String MediaQuery::MediaFeature::to_string() const
 | 
						|
{
 | 
						|
    switch (type) {
 | 
						|
    case Type::IsTrue:
 | 
						|
        return name;
 | 
						|
    case Type::ExactValue:
 | 
						|
        return String::formatted("{}:{}", name, value->to_string());
 | 
						|
    case Type::MinValue:
 | 
						|
        return String::formatted("min-{}:{}", name, value->to_string());
 | 
						|
    case Type::MaxValue:
 | 
						|
        return String::formatted("max-{}:{}", name, value->to_string());
 | 
						|
    }
 | 
						|
 | 
						|
    VERIFY_NOT_REACHED();
 | 
						|
}
 | 
						|
 | 
						|
bool MediaQuery::MediaFeature::evaluate(DOM::Window const& window) const
 | 
						|
{
 | 
						|
    auto queried_value = window.query_media_feature(name);
 | 
						|
    if (!queried_value)
 | 
						|
        return false;
 | 
						|
 | 
						|
    switch (type) {
 | 
						|
    case Type::IsTrue:
 | 
						|
        if (queried_value->has_number())
 | 
						|
            return queried_value->to_number() != 0;
 | 
						|
        if (queried_value->has_length())
 | 
						|
            return queried_value->to_length().raw_value() != 0;
 | 
						|
        if (queried_value->has_identifier())
 | 
						|
            return queried_value->to_identifier() != ValueID::None;
 | 
						|
        return false;
 | 
						|
 | 
						|
    case Type::ExactValue:
 | 
						|
        return queried_value->equals(*value);
 | 
						|
 | 
						|
    case Type::MinValue:
 | 
						|
        if (queried_value->has_number() && value->has_number())
 | 
						|
            return queried_value->to_number() >= value->to_number();
 | 
						|
        if (queried_value->has_length() && value->has_length()) {
 | 
						|
            auto queried_length = queried_value->to_length();
 | 
						|
            auto value_length = value->to_length();
 | 
						|
            // FIXME: We should be checking that lengths are valid during parsing
 | 
						|
            if (!value_length.is_absolute()) {
 | 
						|
                dbgln("Media feature was given a non-absolute length, which is invalid! {}", value_length.to_string());
 | 
						|
                return false;
 | 
						|
            }
 | 
						|
            return queried_length.absolute_length_to_px() >= value_length.absolute_length_to_px();
 | 
						|
        }
 | 
						|
        return false;
 | 
						|
 | 
						|
    case Type::MaxValue:
 | 
						|
        if (queried_value->has_number() && value->has_number())
 | 
						|
            return queried_value->to_number() <= value->to_number();
 | 
						|
        if (queried_value->has_length() && value->has_length()) {
 | 
						|
            auto queried_length = queried_value->to_length();
 | 
						|
            auto value_length = value->to_length();
 | 
						|
            // FIXME: We should be checking that lengths are valid during parsing
 | 
						|
            if (!value_length.is_absolute()) {
 | 
						|
                dbgln("Media feature was given a non-absolute length, which is invalid! {}", value_length.to_string());
 | 
						|
                return false;
 | 
						|
            }
 | 
						|
            return queried_length.absolute_length_to_px() <= value_length.absolute_length_to_px();
 | 
						|
        }
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    VERIFY_NOT_REACHED();
 | 
						|
}
 | 
						|
 | 
						|
String MediaQuery::MediaCondition::to_string() const
 | 
						|
{
 | 
						|
    StringBuilder builder;
 | 
						|
    builder.append('(');
 | 
						|
    switch (type) {
 | 
						|
    case Type::Single:
 | 
						|
        builder.append(feature.to_string());
 | 
						|
        break;
 | 
						|
    case Type::Not:
 | 
						|
        builder.append("not ");
 | 
						|
        builder.append(conditions.first().to_string());
 | 
						|
        break;
 | 
						|
    case Type::And:
 | 
						|
        builder.join(" and ", conditions);
 | 
						|
        break;
 | 
						|
    case Type::Or:
 | 
						|
        builder.join(" or ", conditions);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
    builder.append(')');
 | 
						|
    return builder.to_string();
 | 
						|
}
 | 
						|
 | 
						|
bool MediaQuery::MediaCondition::evaluate(DOM::Window const& window) const
 | 
						|
{
 | 
						|
    switch (type) {
 | 
						|
    case Type::Single:
 | 
						|
        return feature.evaluate(window);
 | 
						|
 | 
						|
    case Type::Not:
 | 
						|
        return !conditions.first().evaluate(window);
 | 
						|
 | 
						|
    case Type::And:
 | 
						|
        for (auto& child : conditions) {
 | 
						|
            if (!child.evaluate(window))
 | 
						|
                return false;
 | 
						|
        }
 | 
						|
        return true;
 | 
						|
 | 
						|
    case Type::Or:
 | 
						|
        for (auto& child : conditions) {
 | 
						|
            if (child.evaluate(window))
 | 
						|
                return true;
 | 
						|
        }
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    VERIFY_NOT_REACHED();
 | 
						|
}
 | 
						|
 | 
						|
String MediaQuery::to_string() const
 | 
						|
{
 | 
						|
    StringBuilder builder;
 | 
						|
 | 
						|
    if (m_negated)
 | 
						|
        builder.append("not ");
 | 
						|
 | 
						|
    if (m_negated || m_media_type != MediaType::All || !m_media_condition) {
 | 
						|
        switch (m_media_type) {
 | 
						|
        case MediaType::All:
 | 
						|
            builder.append("all");
 | 
						|
            break;
 | 
						|
        case MediaType::Aural:
 | 
						|
            builder.append("aural");
 | 
						|
            break;
 | 
						|
        case MediaType::Braille:
 | 
						|
            builder.append("braille");
 | 
						|
            break;
 | 
						|
        case MediaType::Embossed:
 | 
						|
            builder.append("embossed");
 | 
						|
            break;
 | 
						|
        case MediaType::Handheld:
 | 
						|
            builder.append("handheld");
 | 
						|
            break;
 | 
						|
        case MediaType::Print:
 | 
						|
            builder.append("print");
 | 
						|
            break;
 | 
						|
        case MediaType::Projection:
 | 
						|
            builder.append("projection");
 | 
						|
            break;
 | 
						|
        case MediaType::Screen:
 | 
						|
            builder.append("screen");
 | 
						|
            break;
 | 
						|
        case MediaType::Speech:
 | 
						|
            builder.append("speech");
 | 
						|
            break;
 | 
						|
        case MediaType::TTY:
 | 
						|
            builder.append("tty");
 | 
						|
            break;
 | 
						|
        case MediaType::TV:
 | 
						|
            builder.append("tv");
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        if (m_media_condition)
 | 
						|
            builder.append(" and ");
 | 
						|
    }
 | 
						|
 | 
						|
    if (m_media_condition) {
 | 
						|
        builder.append(m_media_condition->to_string());
 | 
						|
    }
 | 
						|
 | 
						|
    return builder.to_string();
 | 
						|
}
 | 
						|
 | 
						|
bool MediaQuery::evaluate(DOM::Window const& window)
 | 
						|
{
 | 
						|
    auto matches_media = [](MediaType media) -> bool {
 | 
						|
        switch (media) {
 | 
						|
        case MediaType::All:
 | 
						|
            return true;
 | 
						|
        case MediaType::Print:
 | 
						|
            // FIXME: Enable for printing, when we have printing!
 | 
						|
            return false;
 | 
						|
        case MediaType::Screen:
 | 
						|
            // FIXME: Disable for printing, when we have printing!
 | 
						|
            return true;
 | 
						|
        // Deprecated, must never match:
 | 
						|
        case MediaType::TTY:
 | 
						|
        case MediaType::TV:
 | 
						|
        case MediaType::Projection:
 | 
						|
        case MediaType::Handheld:
 | 
						|
        case MediaType::Braille:
 | 
						|
        case MediaType::Embossed:
 | 
						|
        case MediaType::Aural:
 | 
						|
        case MediaType::Speech:
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
        VERIFY_NOT_REACHED();
 | 
						|
    };
 | 
						|
 | 
						|
    bool result = matches_media(m_media_type);
 | 
						|
 | 
						|
    if (result && m_media_condition)
 | 
						|
        result = m_media_condition->evaluate(window);
 | 
						|
 | 
						|
    m_matches = m_negated ? !result : result;
 | 
						|
    return m_matches;
 | 
						|
}
 | 
						|
 | 
						|
}
 |