mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 22:32:07 +00:00 
			
		
		
		
	 85a0772147
			
		
	
	
		85a0772147
		
	
	
	
	
		
			
			Until now, we've internally thought of the CSS "display" property as a single-value property. In practice, "display" is a much more complex property that comes in a number of configurations. The most interesting one is the two-part format that describes the outside and inside behavior of a box. Switching our own internal representation towards this model will allow for much cleaner abstractions around layout and the various formatting contexts. Note that we don't *parse* two-part "display" yet, this is only about changing the internal representation of the property. Spec: https://drafts.csswg.org/css-display
		
			
				
	
	
		
			241 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			241 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/Optional.h>
 | |
| #include <LibWeb/CSS/LengthBox.h>
 | |
| #include <LibWeb/CSS/StyleValue.h>
 | |
| 
 | |
| namespace Web::CSS {
 | |
| 
 | |
| class InitialValues {
 | |
| public:
 | |
|     static CSS::Float float_() { return CSS::Float::None; }
 | |
|     static CSS::Clear clear() { return CSS::Clear::None; }
 | |
|     static CSS::Cursor cursor() { return CSS::Cursor::Auto; }
 | |
|     static CSS::WhiteSpace white_space() { return CSS::WhiteSpace::Normal; }
 | |
|     static CSS::TextAlign text_align() { return CSS::TextAlign::Left; }
 | |
|     static CSS::Position position() { return CSS::Position::Static; }
 | |
|     static CSS::TextDecorationLine text_decoration_line() { return CSS::TextDecorationLine::None; }
 | |
|     static CSS::TextTransform text_transform() { return CSS::TextTransform::None; }
 | |
|     static CSS::Display display() { return CSS::Display { CSS::Display::Outside::Inline, CSS::Display::Inside::Flow }; }
 | |
|     static Color color() { return Color::Black; }
 | |
|     static Color background_color() { return Color::Transparent; }
 | |
|     static CSS::Repeat background_repeat() { return CSS::Repeat::Repeat; }
 | |
|     static CSS::ListStyleType list_style_type() { return CSS::ListStyleType::Disc; }
 | |
|     static CSS::FlexDirection flex_direction() { return CSS::FlexDirection::Row; }
 | |
|     static CSS::FlexWrap flex_wrap() { return CSS::FlexWrap::Nowrap; }
 | |
|     static CSS::JustifyContent justify_content() { return CSS::JustifyContent::FlexStart; }
 | |
|     static CSS::AlignItems align_items() { return CSS::AlignItems::FlexStart; }
 | |
|     static CSS::Overflow overflow() { return CSS::Overflow::Visible; }
 | |
|     static CSS::BoxSizing box_sizing() { return CSS::BoxSizing::ContentBox; }
 | |
| };
 | |
| 
 | |
| struct BorderData {
 | |
| public:
 | |
|     Color color { Color::Transparent };
 | |
|     CSS::LineStyle line_style { CSS::LineStyle::None };
 | |
|     float width { 0 };
 | |
| };
 | |
| 
 | |
| struct Transformation {
 | |
|     CSS::TransformFunction function;
 | |
|     Vector<Variant<CSS::Length, float>> values;
 | |
| };
 | |
| 
 | |
| struct FlexBasisData {
 | |
|     CSS::FlexBasis type { CSS::FlexBasis::Auto };
 | |
|     CSS::Length length {};
 | |
| };
 | |
| 
 | |
| struct BoxShadowData {
 | |
|     CSS::Length offset_x {};
 | |
|     CSS::Length offset_y {};
 | |
|     CSS::Length blur_radius {};
 | |
|     Color color {};
 | |
| };
 | |
| 
 | |
| class ComputedValues {
 | |
| public:
 | |
|     CSS::Float float_() const { return m_noninherited.float_; }
 | |
|     CSS::Clear clear() const { return m_noninherited.clear; }
 | |
|     CSS::Cursor cursor() const { return m_inherited.cursor; }
 | |
|     CSS::Display display() const { return m_noninherited.display; }
 | |
|     Optional<int> const& z_index() const { return m_noninherited.z_index; }
 | |
|     CSS::TextAlign text_align() const { return m_inherited.text_align; }
 | |
|     CSS::TextDecorationLine text_decoration_line() const { return m_noninherited.text_decoration_line; }
 | |
|     CSS::TextTransform text_transform() const { return m_inherited.text_transform; }
 | |
|     CSS::Position position() const { return m_noninherited.position; }
 | |
|     CSS::WhiteSpace white_space() const { return m_inherited.white_space; }
 | |
|     CSS::FlexDirection flex_direction() const { return m_noninherited.flex_direction; }
 | |
|     CSS::FlexWrap flex_wrap() const { return m_noninherited.flex_wrap; }
 | |
|     FlexBasisData const& flex_basis() const { return m_noninherited.flex_basis; }
 | |
|     Optional<float> const& flex_grow_factor() const { return m_noninherited.flex_grow_factor; }
 | |
|     Optional<float> const& flex_shrink_factor() const { return m_noninherited.flex_shrink_factor; }
 | |
|     CSS::AlignItems align_items() const { return m_noninherited.align_items; }
 | |
|     Optional<float> const& opacity() const { return m_noninherited.opacity; }
 | |
|     CSS::JustifyContent justify_content() const { return m_noninherited.justify_content; }
 | |
|     Optional<BoxShadowData> const& box_shadow() const { return m_noninherited.box_shadow; }
 | |
|     CSS::BoxSizing box_sizing() const { return m_noninherited.box_sizing; }
 | |
|     const CSS::Length& width() const { return m_noninherited.width; }
 | |
|     const CSS::Length& min_width() const { return m_noninherited.min_width; }
 | |
|     const CSS::Length& max_width() const { return m_noninherited.max_width; }
 | |
|     const CSS::Length& height() const { return m_noninherited.height; }
 | |
|     const CSS::Length& min_height() const { return m_noninherited.min_height; }
 | |
|     const CSS::Length& max_height() const { return m_noninherited.max_height; }
 | |
| 
 | |
|     const CSS::LengthBox& offset() const { return m_noninherited.offset; }
 | |
|     const CSS::LengthBox& margin() const { return m_noninherited.margin; }
 | |
|     const CSS::LengthBox& padding() const { return m_noninherited.padding; }
 | |
| 
 | |
|     const BorderData& border_left() const { return m_noninherited.border_left; }
 | |
|     const BorderData& border_top() const { return m_noninherited.border_top; }
 | |
|     const BorderData& border_right() const { return m_noninherited.border_right; }
 | |
|     const BorderData& border_bottom() const { return m_noninherited.border_bottom; }
 | |
| 
 | |
|     const CSS::Length& border_bottom_left_radius() const { return m_noninherited.border_bottom_left_radius; }
 | |
|     const CSS::Length& border_bottom_right_radius() const { return m_noninherited.border_bottom_right_radius; }
 | |
|     const CSS::Length& border_top_left_radius() const { return m_noninherited.border_top_left_radius; }
 | |
|     const CSS::Length& border_top_right_radius() const { return m_noninherited.border_top_right_radius; }
 | |
| 
 | |
|     CSS::Overflow overflow_x() const { return m_noninherited.overflow_x; }
 | |
|     CSS::Overflow overflow_y() const { return m_noninherited.overflow_y; }
 | |
| 
 | |
|     Color color() const { return m_inherited.color; }
 | |
|     Color background_color() const { return m_noninherited.background_color; }
 | |
|     CSS::Repeat background_repeat_x() const { return m_noninherited.background_repeat_x; }
 | |
|     CSS::Repeat background_repeat_y() const { return m_noninherited.background_repeat_y; }
 | |
| 
 | |
|     CSS::ListStyleType list_style_type() const { return m_inherited.list_style_type; }
 | |
| 
 | |
|     Optional<Color> fill() const { return m_inherited.fill; }
 | |
|     Optional<Color> stroke() const { return m_inherited.stroke; }
 | |
|     Optional<Length> const& stroke_width() const { return m_inherited.stroke_width; }
 | |
| 
 | |
|     Vector<CSS::Transformation> transformations() const { return m_noninherited.transformations; }
 | |
| 
 | |
|     ComputedValues clone_inherited_values() const
 | |
|     {
 | |
|         ComputedValues clone;
 | |
|         clone.m_inherited = m_inherited;
 | |
|         return clone;
 | |
|     }
 | |
| 
 | |
| protected:
 | |
|     struct {
 | |
|         Color color { InitialValues::color() };
 | |
|         CSS::Cursor cursor { InitialValues::cursor() };
 | |
|         CSS::TextAlign text_align { InitialValues::text_align() };
 | |
|         CSS::TextTransform text_transform { InitialValues::text_transform() };
 | |
|         CSS::WhiteSpace white_space { InitialValues::white_space() };
 | |
|         CSS::ListStyleType list_style_type { InitialValues::list_style_type() };
 | |
| 
 | |
|         Optional<Color> fill;
 | |
|         Optional<Color> stroke;
 | |
|         Optional<Length> stroke_width;
 | |
|     } m_inherited;
 | |
| 
 | |
|     struct {
 | |
|         CSS::Float float_ { InitialValues::float_() };
 | |
|         CSS::Clear clear { InitialValues::clear() };
 | |
|         CSS::Display display { InitialValues::display() };
 | |
|         Optional<int> z_index;
 | |
|         CSS::TextDecorationLine text_decoration_line { InitialValues::text_decoration_line() };
 | |
|         CSS::Position position { InitialValues::position() };
 | |
|         CSS::Length width;
 | |
|         CSS::Length min_width;
 | |
|         CSS::Length max_width;
 | |
|         CSS::Length height;
 | |
|         CSS::Length min_height;
 | |
|         CSS::Length max_height;
 | |
|         CSS::LengthBox offset;
 | |
|         CSS::LengthBox margin;
 | |
|         CSS::LengthBox padding;
 | |
|         BorderData border_left;
 | |
|         BorderData border_top;
 | |
|         BorderData border_right;
 | |
|         BorderData border_bottom;
 | |
|         Length border_bottom_left_radius;
 | |
|         Length border_bottom_right_radius;
 | |
|         Length border_top_left_radius;
 | |
|         Length border_top_right_radius;
 | |
|         Color background_color { InitialValues::background_color() };
 | |
|         CSS::Repeat background_repeat_x { InitialValues::background_repeat() };
 | |
|         CSS::Repeat background_repeat_y { InitialValues::background_repeat() };
 | |
|         CSS::FlexDirection flex_direction { InitialValues::flex_direction() };
 | |
|         CSS::FlexWrap flex_wrap { InitialValues::flex_wrap() };
 | |
|         CSS::FlexBasisData flex_basis {};
 | |
|         Optional<float> flex_grow_factor;
 | |
|         Optional<float> flex_shrink_factor;
 | |
|         CSS::AlignItems align_items { InitialValues::align_items() };
 | |
|         CSS::JustifyContent justify_content { InitialValues::justify_content() };
 | |
|         CSS::Overflow overflow_x { InitialValues::overflow() };
 | |
|         CSS::Overflow overflow_y { InitialValues::overflow() };
 | |
|         Optional<float> opacity;
 | |
|         Optional<BoxShadowData> box_shadow {};
 | |
|         Vector<CSS::Transformation> transformations {};
 | |
|         CSS::BoxSizing box_sizing { InitialValues::box_sizing() };
 | |
|     } m_noninherited;
 | |
| };
 | |
| 
 | |
| class ImmutableComputedValues final : public ComputedValues {
 | |
| };
 | |
| 
 | |
| class MutableComputedValues final : public ComputedValues {
 | |
| public:
 | |
|     void set_color(const Color& color) { m_inherited.color = color; }
 | |
|     void set_cursor(CSS::Cursor cursor) { m_inherited.cursor = cursor; }
 | |
|     void set_background_color(const Color& color) { m_noninherited.background_color = color; }
 | |
|     void set_background_repeat_x(CSS::Repeat repeat) { m_noninherited.background_repeat_x = repeat; }
 | |
|     void set_background_repeat_y(CSS::Repeat repeat) { m_noninherited.background_repeat_y = repeat; }
 | |
|     void set_float(CSS::Float value) { m_noninherited.float_ = value; }
 | |
|     void set_clear(CSS::Clear value) { m_noninherited.clear = value; }
 | |
|     void set_z_index(Optional<int> value) { m_noninherited.z_index = value; }
 | |
|     void set_text_align(CSS::TextAlign text_align) { m_inherited.text_align = text_align; }
 | |
|     void set_text_decoration_line(CSS::TextDecorationLine value) { m_noninherited.text_decoration_line = value; }
 | |
|     void set_text_transform(CSS::TextTransform value) { m_inherited.text_transform = value; }
 | |
|     void set_position(CSS::Position position) { m_noninherited.position = position; }
 | |
|     void set_white_space(CSS::WhiteSpace value) { m_inherited.white_space = value; }
 | |
|     void set_width(const CSS::Length& width) { m_noninherited.width = width; }
 | |
|     void set_min_width(const CSS::Length& width) { m_noninherited.min_width = width; }
 | |
|     void set_max_width(const CSS::Length& width) { m_noninherited.max_width = width; }
 | |
|     void set_height(const CSS::Length& height) { m_noninherited.height = height; }
 | |
|     void set_min_height(const CSS::Length& height) { m_noninherited.min_height = height; }
 | |
|     void set_max_height(const CSS::Length& height) { m_noninherited.max_height = height; }
 | |
|     void set_offset(const CSS::LengthBox& offset) { m_noninherited.offset = offset; }
 | |
|     void set_margin(const CSS::LengthBox& margin) { m_noninherited.margin = margin; }
 | |
|     void set_padding(const CSS::LengthBox& padding) { m_noninherited.padding = padding; }
 | |
|     void set_overflow_x(CSS::Overflow value) { m_noninherited.overflow_x = value; }
 | |
|     void set_overflow_y(CSS::Overflow value) { m_noninherited.overflow_y = value; }
 | |
|     void set_list_style_type(CSS::ListStyleType value) { m_inherited.list_style_type = value; }
 | |
|     void set_display(CSS::Display value) { m_noninherited.display = value; }
 | |
|     void set_border_bottom_left_radius(CSS::Length value) { m_noninherited.border_bottom_left_radius = value; }
 | |
|     void set_border_bottom_right_radius(CSS::Length value) { m_noninherited.border_bottom_right_radius = value; }
 | |
|     void set_border_top_left_radius(CSS::Length value) { m_noninherited.border_top_left_radius = value; }
 | |
|     void set_border_top_right_radius(CSS::Length value) { m_noninherited.border_top_right_radius = value; }
 | |
|     BorderData& border_left() { return m_noninherited.border_left; }
 | |
|     BorderData& border_top() { return m_noninherited.border_top; }
 | |
|     BorderData& border_right() { return m_noninherited.border_right; }
 | |
|     BorderData& border_bottom() { return m_noninherited.border_bottom; }
 | |
|     void set_flex_direction(CSS::FlexDirection value) { m_noninherited.flex_direction = value; }
 | |
|     void set_flex_wrap(CSS::FlexWrap value) { m_noninherited.flex_wrap = value; }
 | |
|     void set_flex_basis(FlexBasisData value) { m_noninherited.flex_basis = value; }
 | |
|     void set_flex_grow_factor(Optional<float> value) { m_noninherited.flex_grow_factor = value; }
 | |
|     void set_flex_shrink_factor(Optional<float> value) { m_noninherited.flex_shrink_factor = value; }
 | |
|     void set_align_items(CSS::AlignItems value) { m_noninherited.align_items = value; }
 | |
|     void set_opacity(Optional<float> value) { m_noninherited.opacity = value; }
 | |
|     void set_justify_content(CSS::JustifyContent value) { m_noninherited.justify_content = value; }
 | |
|     void set_box_shadow(Optional<BoxShadowData> value) { m_noninherited.box_shadow = move(value); }
 | |
|     void set_transformations(Vector<CSS::Transformation> value) { m_noninherited.transformations = move(value); }
 | |
|     void set_box_sizing(CSS::BoxSizing value) { m_noninherited.box_sizing = value; }
 | |
| 
 | |
|     void set_fill(Color value) { m_inherited.fill = value; }
 | |
|     void set_stroke(Color value) { m_inherited.stroke = value; }
 | |
|     void set_stroke_width(Length value) { m_inherited.stroke_width = value; }
 | |
| };
 | |
| 
 | |
| }
 |