/* * Copyright (c) 2018-2024, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include #include namespace Web::CSS { class StyleProperties { public: static constexpr size_t number_of_properties = to_underlying(CSS::last_property_id) + 1; private: struct Data : public RefCounted { friend class StyleComputer; NonnullRefPtr clone() const; // FIXME: These need protection from GC! GC::Ptr m_animation_name_source; GC::Ptr m_transition_property_source; Array, number_of_properties> m_property_values; Array m_property_important {}; Array m_property_inherited {}; HashMap> m_animated_property_values; int m_math_depth { InitialValues::math_depth() }; mutable RefPtr m_font_list; Optional m_line_height; }; public: StyleProperties() = default; template inline void for_each_property(Callback callback) const { for (size_t i = 0; i < m_data->m_property_values.size(); ++i) { if (m_data->m_property_values[i]) callback((CSS::PropertyID)i, *m_data->m_property_values[i]); } } enum class Inherited { No, Yes }; HashMap> const& animated_property_values() const { return m_data->m_animated_property_values; } void reset_animated_properties(); bool is_property_important(CSS::PropertyID property_id) const; bool is_property_inherited(CSS::PropertyID property_id) const; void set_property_important(CSS::PropertyID, Important); void set_property_inherited(CSS::PropertyID, Inherited); void set_property(CSS::PropertyID, NonnullRefPtr value, Inherited = Inherited::No, Important = Important::No); void set_animated_property(CSS::PropertyID, NonnullRefPtr value); enum class WithAnimationsApplied { No, Yes, }; CSSStyleValue const& property(CSS::PropertyID, WithAnimationsApplied = WithAnimationsApplied::Yes) const; CSSStyleValue const* maybe_null_property(CSS::PropertyID) const; void revert_property(CSS::PropertyID, StyleProperties const& style_for_revert); GC::Ptr animation_name_source() const { return m_data->m_animation_name_source; } void set_animation_name_source(GC::Ptr declaration) { m_data->m_animation_name_source = declaration; } GC::Ptr transition_property_source() const { return m_data->m_transition_property_source; } void set_transition_property_source(GC::Ptr declaration) { m_data->m_transition_property_source = declaration; } CSS::Size size_value(CSS::PropertyID) const; [[nodiscard]] Variant gap_value(CSS::PropertyID) const; LengthPercentage length_percentage_or_fallback(CSS::PropertyID, LengthPercentage const& fallback) const; Optional length_percentage(CSS::PropertyID) const; LengthBox length_box(CSS::PropertyID left_id, CSS::PropertyID top_id, CSS::PropertyID right_id, CSS::PropertyID bottom_id, const CSS::Length& default_value) const; Color color_or_fallback(CSS::PropertyID, Layout::NodeWithStyle const&, Color fallback) const; Optional text_anchor() const; Optional text_align() const; Optional text_justify() const; Optional text_overflow() const; CSS::Length border_spacing_horizontal(Layout::Node const&) const; CSS::Length border_spacing_vertical(Layout::Node const&) const; Optional caption_side() const; CSS::Clip clip() const; CSS::Display display() const; Optional float_() const; Optional clear() const; Optional column_span() const; struct ContentDataAndQuoteNestingLevel { CSS::ContentData content_data; u32 final_quote_nesting_level { 0 }; }; ContentDataAndQuoteNestingLevel content(DOM::Element&, u32 initial_quote_nesting_level) const; Optional content_visibility() const; Optional cursor() const; Variant tab_size() const; Optional white_space() const; Optional word_break() const; Optional word_spacing() const; Optional letter_spacing() const; Optional line_style(CSS::PropertyID) const; Optional outline_style() const; Vector text_decoration_line() const; Optional text_decoration_style() const; Optional text_transform() const; Vector text_shadow(Layout::Node const&) const; Optional list_style_type() const; Optional list_style_position() const; Optional flex_direction() const; Optional flex_wrap() const; Optional flex_basis() const; float flex_grow() const; float flex_shrink() const; int order() const; Optional accent_color(Layout::NodeWithStyle const&) const; Optional align_content() const; Optional align_items() const; Optional align_self() const; Optional appearance() const; CSS::Filter backdrop_filter() const; CSS::Filter filter() const; float opacity() const; Optional visibility() const; Optional image_rendering() const; Optional justify_content() const; Optional justify_items() const; Optional justify_self() const; Optional overflow_x() const; Optional overflow_y() const; Vector box_shadow(Layout::Node const&) const; Optional box_sizing() const; Optional pointer_events() const; Variant vertical_align() const; Optional font_variant() const; Optional font_language_override() const; Optional> font_feature_settings() const; Optional> font_variation_settings() const; CSS::GridTrackSizeList grid_auto_columns() const; CSS::GridTrackSizeList grid_auto_rows() const; CSS::GridTrackSizeList grid_template_columns() const; CSS::GridTrackSizeList grid_template_rows() const; [[nodiscard]] CSS::GridAutoFlow grid_auto_flow() const; CSS::GridTrackPlacement grid_column_end() const; CSS::GridTrackPlacement grid_column_start() const; CSS::GridTrackPlacement grid_row_end() const; CSS::GridTrackPlacement grid_row_start() const; Optional border_collapse() const; Vector> grid_template_areas() const; Optional object_fit() const; CSS::ObjectPosition object_position() const; Optional table_layout() const; Optional direction() const; Optional unicode_bidi() const; Optional writing_mode() const; static Vector transformations_for_style_value(CSSStyleValue const& value); Vector transformations() const; Optional transform_box() const; CSS::TransformOrigin transform_origin() const; Optional rotate(Layout::Node const&) const; Optional translate() const; Optional mask_type() const; Color stop_color() const; float stop_opacity() const; float fill_opacity() const; Optional stroke_linecap() const; Optional stroke_linejoin() const; NumberOrCalculated stroke_miterlimit() const; float stroke_opacity() const; Optional fill_rule() const; Optional clip_rule() const; Gfx::Font const& first_available_computed_font() const { return m_data->m_font_list->first(); } Gfx::FontCascadeList const& computed_font_list() const { VERIFY(m_data->m_font_list); return *m_data->m_font_list; } void set_computed_font_list(NonnullRefPtr font_list) const { m_data->m_font_list = move(font_list); } [[nodiscard]] CSSPixels compute_line_height(CSSPixelRect const& viewport_rect, Length::FontMetrics const& font_metrics, Length::FontMetrics const& root_font_metrics) const; [[nodiscard]] CSSPixels line_height() const { return *m_data->m_line_height; } void set_line_height(Badge const&, CSSPixels line_height) { m_data->m_line_height = line_height; } bool operator==(StyleProperties const&) const; Optional position() const; Optional z_index() const; void set_math_depth(int math_depth); int math_depth() const { return m_data->m_math_depth; } QuotesData quotes() const; Vector counter_data(PropertyID) const; Optional scrollbar_width() const; static NonnullRefPtr font_fallback(bool monospace, bool bold); static float resolve_opacity_value(CSSStyleValue const& value); private: friend class StyleComputer; Optional overflow(CSS::PropertyID) const; Vector shadow(CSS::PropertyID, Layout::Node const&) const; AK::CopyOnWrite m_data; }; }