/* * Copyright (c) 2021, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include namespace Web::CSS { // https://www.w3.org/TR/mediaqueries-4/#typedef-mf-value class MediaFeatureValue { public: explicit MediaFeatureValue(String ident) : m_value(move(ident)) { } explicit MediaFeatureValue(Length length) : m_value(move(length)) { } explicit MediaFeatureValue(double number) : m_value(number) { } String to_string() const; bool is_ident() const { return m_value.has(); } bool is_length() const { return m_value.has(); } bool is_number() const { return m_value.has(); } bool is_same_type(MediaFeatureValue const& other) const; String const& ident() const { VERIFY(is_ident()); return m_value.get(); } Length const& length() const { VERIFY(is_length()); return m_value.get(); } double number() const { VERIFY(is_number()); return m_value.get(); } bool operator==(MediaFeatureValue const& other) const { return equals(other); } bool operator!=(MediaFeatureValue const& other) const { return !(*this == other); } bool equals(MediaFeatureValue const& other) const; private: // TODO: Support once we have that. Variant m_value; }; class MediaQuery : public RefCounted { friend class Parser; public: ~MediaQuery() = default; // https://www.w3.org/TR/mediaqueries-4/#media-types enum class MediaType { All, Print, Screen, // Deprecated, must never match: TTY, TV, Projection, Handheld, Braille, Embossed, Aural, Speech, }; // https://www.w3.org/TR/mediaqueries-4/#mq-features struct MediaFeature { // FIXME: Implement range syntax: https://www.w3.org/TR/mediaqueries-4/#mq-ranges enum class Type { IsTrue, ExactValue, MinValue, MaxValue, }; Type type; FlyString name; Optional value {}; bool evaluate(DOM::Window const&) const; String to_string() const; }; // https://www.w3.org/TR/mediaqueries-4/#media-conditions struct MediaCondition { enum class Type { Single, And, Or, Not, GeneralEnclosed, }; Type type; MediaFeature feature; NonnullOwnPtrVector conditions; Optional general_enclosed; MatchResult evaluate(DOM::Window const&) const; String to_string() const; }; static NonnullRefPtr create_not_all(); static NonnullRefPtr create() { return adopt_ref(*new MediaQuery); } bool matches() const { return m_matches; } bool evaluate(DOM::Window const&); String to_string() const; private: MediaQuery() = default; // https://www.w3.org/TR/mediaqueries-4/#mq-not bool m_negated { false }; MediaType m_media_type { MediaType::All }; OwnPtr m_media_condition { nullptr }; // Cached value, updated by evaluate() bool m_matches { false }; }; String serialize_a_media_query_list(NonnullRefPtrVector const&); } namespace AK { template<> struct Formatter : Formatter { ErrorOr format(FormatBuilder& builder, Web::CSS::MediaQuery::MediaFeature const& media_feature) { return Formatter::format(builder, media_feature.to_string()); } }; template<> struct Formatter : Formatter { ErrorOr format(FormatBuilder& builder, Web::CSS::MediaQuery::MediaCondition const& media_condition) { return Formatter::format(builder, media_condition.to_string()); } }; template<> struct Formatter : Formatter { ErrorOr format(FormatBuilder& builder, Web::CSS::MediaQuery const& media_query) { return Formatter::format(builder, media_query.to_string()); } }; }