mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-04-30 16:28:48 +00:00
calc() sub-expressions can return a variety of different types, which then can be combined using the basic arithmetic operators. This class should make that easier to deal with, instead of having to handle all the possible combinations at each call site. :^) We take the Layout::Node as a pointer not a reference, since later we'll need to call these functions when resolving to `<number>` or `<integer>` which don't use those, and we don't want to force users to pass them in unnecessarily.
1417 lines
45 KiB
C++
1417 lines
45 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
* Copyright (c) 2021, Tobias Christiansen <tobyase@serenityos.org>
|
|
* Copyright (c) 2021-2022, Sam Atkins <atkinssj@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/Function.h>
|
|
#include <AK/NonnullOwnPtr.h>
|
|
#include <AK/NonnullOwnPtrVector.h>
|
|
#include <AK/NonnullRefPtrVector.h>
|
|
#include <AK/RefCounted.h>
|
|
#include <AK/RefPtr.h>
|
|
#include <AK/String.h>
|
|
#include <AK/StringView.h>
|
|
#include <AK/URL.h>
|
|
#include <AK/Variant.h>
|
|
#include <AK/Vector.h>
|
|
#include <AK/WeakPtr.h>
|
|
#include <LibGfx/Bitmap.h>
|
|
#include <LibGfx/Color.h>
|
|
#include <LibWeb/CSS/Display.h>
|
|
#include <LibWeb/CSS/Length.h>
|
|
#include <LibWeb/CSS/Parser/StyleComponentValueRule.h>
|
|
#include <LibWeb/CSS/Percentage.h>
|
|
#include <LibWeb/CSS/PropertyID.h>
|
|
#include <LibWeb/CSS/ValueID.h>
|
|
#include <LibWeb/Forward.h>
|
|
#include <LibWeb/Loader/ImageResource.h>
|
|
|
|
namespace Web::CSS {
|
|
|
|
enum class AlignItems {
|
|
FlexStart,
|
|
FlexEnd,
|
|
Center,
|
|
Baseline,
|
|
Stretch,
|
|
};
|
|
|
|
enum class BackgroundAttachment {
|
|
Fixed,
|
|
Local,
|
|
Scroll,
|
|
};
|
|
|
|
enum class BackgroundBox {
|
|
BorderBox,
|
|
ContentBox,
|
|
PaddingBox,
|
|
};
|
|
|
|
enum class BackgroundSize {
|
|
Contain,
|
|
Cover,
|
|
LengthPercentage,
|
|
};
|
|
|
|
enum class BoxSizing {
|
|
BorderBox,
|
|
ContentBox,
|
|
};
|
|
|
|
enum class Clear {
|
|
None,
|
|
Left,
|
|
Right,
|
|
Both,
|
|
};
|
|
|
|
enum class Cursor {
|
|
Auto,
|
|
Default,
|
|
None,
|
|
ContextMenu,
|
|
Help,
|
|
Pointer,
|
|
Progress,
|
|
Wait,
|
|
Cell,
|
|
Crosshair,
|
|
Text,
|
|
VerticalText,
|
|
Alias,
|
|
Copy,
|
|
Move,
|
|
NoDrop,
|
|
NotAllowed,
|
|
Grab,
|
|
Grabbing,
|
|
EResize,
|
|
NResize,
|
|
NeResize,
|
|
NwResize,
|
|
SResize,
|
|
SeResize,
|
|
SwResize,
|
|
WResize,
|
|
EwResize,
|
|
NsResize,
|
|
NeswResize,
|
|
NwseResize,
|
|
ColResize,
|
|
RowResize,
|
|
AllScroll,
|
|
ZoomIn,
|
|
ZoomOut,
|
|
};
|
|
|
|
enum class FlexBasis {
|
|
Content,
|
|
LengthPercentage,
|
|
Auto,
|
|
};
|
|
|
|
enum class FlexDirection {
|
|
Row,
|
|
RowReverse,
|
|
Column,
|
|
ColumnReverse,
|
|
};
|
|
|
|
enum class FlexWrap {
|
|
Nowrap,
|
|
Wrap,
|
|
WrapReverse
|
|
};
|
|
|
|
enum class Float {
|
|
None,
|
|
Left,
|
|
Right,
|
|
};
|
|
|
|
enum class JustifyContent {
|
|
FlexStart,
|
|
FlexEnd,
|
|
Center,
|
|
SpaceBetween,
|
|
SpaceAround,
|
|
};
|
|
|
|
enum class LineStyle {
|
|
None,
|
|
Hidden,
|
|
Dotted,
|
|
Dashed,
|
|
Solid,
|
|
Double,
|
|
Groove,
|
|
Ridge,
|
|
Inset,
|
|
Outset,
|
|
};
|
|
|
|
enum class ListStyleType {
|
|
None,
|
|
Disc,
|
|
Circle,
|
|
Square,
|
|
Decimal,
|
|
DecimalLeadingZero,
|
|
LowerAlpha,
|
|
LowerLatin,
|
|
LowerRoman,
|
|
UpperAlpha,
|
|
UpperLatin,
|
|
UpperRoman,
|
|
};
|
|
|
|
enum class Overflow : u8 {
|
|
Auto,
|
|
Clip,
|
|
Hidden,
|
|
Scroll,
|
|
Visible,
|
|
};
|
|
|
|
enum class Position {
|
|
Static,
|
|
Relative,
|
|
Absolute,
|
|
Fixed,
|
|
Sticky,
|
|
};
|
|
|
|
enum class PositionEdge {
|
|
Left,
|
|
Right,
|
|
Top,
|
|
Bottom,
|
|
};
|
|
|
|
enum class Repeat : u8 {
|
|
NoRepeat,
|
|
Repeat,
|
|
Round,
|
|
Space,
|
|
};
|
|
|
|
constexpr StringView to_string(Repeat value)
|
|
{
|
|
switch (value) {
|
|
case Repeat::NoRepeat:
|
|
return "no-repeat"sv;
|
|
case Repeat::Repeat:
|
|
return "repeat"sv;
|
|
case Repeat::Round:
|
|
return "round"sv;
|
|
case Repeat::Space:
|
|
return "space"sv;
|
|
default:
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
}
|
|
|
|
enum class TextAlign {
|
|
Left,
|
|
Center,
|
|
Right,
|
|
Justify,
|
|
LibwebCenter,
|
|
};
|
|
|
|
enum class TextDecorationLine {
|
|
None,
|
|
Underline,
|
|
Overline,
|
|
LineThrough,
|
|
Blink,
|
|
};
|
|
|
|
enum class TextDecorationStyle {
|
|
Solid,
|
|
Double,
|
|
Dotted,
|
|
Dashed,
|
|
Wavy,
|
|
};
|
|
|
|
enum class TextTransform {
|
|
None,
|
|
Capitalize,
|
|
Uppercase,
|
|
Lowercase,
|
|
FullWidth,
|
|
FullSizeKana,
|
|
};
|
|
|
|
enum class TransformFunction {
|
|
TranslateY,
|
|
};
|
|
|
|
enum class WhiteSpace {
|
|
Normal,
|
|
Pre,
|
|
Nowrap,
|
|
PreLine,
|
|
PreWrap,
|
|
};
|
|
|
|
enum class PointerEvents {
|
|
Auto,
|
|
None
|
|
};
|
|
|
|
class StyleValue : public RefCounted<StyleValue> {
|
|
public:
|
|
virtual ~StyleValue();
|
|
|
|
enum class Type {
|
|
Background,
|
|
BackgroundRepeat,
|
|
BackgroundSize,
|
|
Border,
|
|
BorderRadius,
|
|
BoxShadow,
|
|
Calculated,
|
|
Color,
|
|
CombinedBorderRadius,
|
|
Flex,
|
|
FlexFlow,
|
|
Font,
|
|
Identifier,
|
|
Image,
|
|
Inherit,
|
|
Initial,
|
|
Invalid,
|
|
Length,
|
|
ListStyle,
|
|
Numeric,
|
|
Overflow,
|
|
Percentage,
|
|
Position,
|
|
String,
|
|
TextDecoration,
|
|
Transformation,
|
|
Unresolved,
|
|
Unset,
|
|
ValueList,
|
|
};
|
|
|
|
Type type() const { return m_type; }
|
|
|
|
bool is_background() const { return type() == Type::Background; }
|
|
bool is_background_repeat() const { return type() == Type::BackgroundRepeat; }
|
|
bool is_background_size() const { return type() == Type::BackgroundSize; }
|
|
bool is_border() const { return type() == Type::Border; }
|
|
bool is_border_radius() const { return type() == Type::BorderRadius; }
|
|
bool is_box_shadow() const { return type() == Type::BoxShadow; }
|
|
bool is_calculated() const { return type() == Type::Calculated; }
|
|
bool is_color() const { return type() == Type::Color; }
|
|
bool is_flex() const { return type() == Type::Flex; }
|
|
bool is_flex_flow() const { return type() == Type::FlexFlow; }
|
|
bool is_font() const { return type() == Type::Font; }
|
|
bool is_identifier() const { return type() == Type::Identifier; }
|
|
bool is_image() const { return type() == Type::Image; }
|
|
bool is_inherit() const { return type() == Type::Inherit; }
|
|
bool is_initial() const { return type() == Type::Initial; }
|
|
bool is_length() const { return type() == Type::Length; }
|
|
bool is_list_style() const { return type() == Type::ListStyle; }
|
|
bool is_numeric() const { return type() == Type::Numeric; }
|
|
bool is_overflow() const { return type() == Type::Overflow; }
|
|
bool is_percentage() const { return type() == Type::Percentage; }
|
|
bool is_position() const { return type() == Type::Position; }
|
|
bool is_string() const { return type() == Type::String; }
|
|
bool is_text_decoration() const { return type() == Type::TextDecoration; }
|
|
bool is_transformation() const { return type() == Type::Transformation; }
|
|
bool is_unresolved() const { return type() == Type::Unresolved; }
|
|
bool is_unset() const { return type() == Type::Unset; }
|
|
bool is_value_list() const { return type() == Type::ValueList; }
|
|
|
|
bool is_builtin() const { return is_inherit() || is_initial() || is_unset(); }
|
|
|
|
BackgroundStyleValue const& as_background() const;
|
|
BackgroundRepeatStyleValue const& as_background_repeat() const;
|
|
BackgroundSizeStyleValue const& as_background_size() const;
|
|
BorderRadiusStyleValue const& as_border_radius() const;
|
|
BorderStyleValue const& as_border() const;
|
|
BoxShadowStyleValue const& as_box_shadow() const;
|
|
CalculatedStyleValue const& as_calculated() const;
|
|
ColorStyleValue const& as_color() const;
|
|
FlexFlowStyleValue const& as_flex_flow() const;
|
|
FlexStyleValue const& as_flex() const;
|
|
FontStyleValue const& as_font() const;
|
|
IdentifierStyleValue const& as_identifier() const;
|
|
ImageStyleValue const& as_image() const;
|
|
InheritStyleValue const& as_inherit() const;
|
|
InitialStyleValue const& as_initial() const;
|
|
LengthStyleValue const& as_length() const;
|
|
ListStyleStyleValue const& as_list_style() const;
|
|
NumericStyleValue const& as_numeric() const;
|
|
OverflowStyleValue const& as_overflow() const;
|
|
PercentageStyleValue const& as_percentage() const;
|
|
PositionStyleValue const& as_position() const;
|
|
StringStyleValue const& as_string() const;
|
|
TextDecorationStyleValue const& as_text_decoration() const;
|
|
TransformationStyleValue const& as_transformation() const;
|
|
UnresolvedStyleValue const& as_unresolved() const;
|
|
UnsetStyleValue const& as_unset() const;
|
|
StyleValueList const& as_value_list() const;
|
|
|
|
BackgroundStyleValue& as_background() { return const_cast<BackgroundStyleValue&>(const_cast<StyleValue const&>(*this).as_background()); }
|
|
BackgroundRepeatStyleValue& as_background_repeat() { return const_cast<BackgroundRepeatStyleValue&>(const_cast<StyleValue const&>(*this).as_background_repeat()); }
|
|
BackgroundSizeStyleValue& as_background_size() { return const_cast<BackgroundSizeStyleValue&>(const_cast<StyleValue const&>(*this).as_background_size()); }
|
|
BorderRadiusStyleValue& as_border_radius() { return const_cast<BorderRadiusStyleValue&>(const_cast<StyleValue const&>(*this).as_border_radius()); }
|
|
BorderStyleValue& as_border() { return const_cast<BorderStyleValue&>(const_cast<StyleValue const&>(*this).as_border()); }
|
|
BoxShadowStyleValue& as_box_shadow() { return const_cast<BoxShadowStyleValue&>(const_cast<StyleValue const&>(*this).as_box_shadow()); }
|
|
CalculatedStyleValue& as_calculated() { return const_cast<CalculatedStyleValue&>(const_cast<StyleValue const&>(*this).as_calculated()); }
|
|
ColorStyleValue& as_color() { return const_cast<ColorStyleValue&>(const_cast<StyleValue const&>(*this).as_color()); }
|
|
FlexFlowStyleValue& as_flex_flow() { return const_cast<FlexFlowStyleValue&>(const_cast<StyleValue const&>(*this).as_flex_flow()); }
|
|
FlexStyleValue& as_flex() { return const_cast<FlexStyleValue&>(const_cast<StyleValue const&>(*this).as_flex()); }
|
|
FontStyleValue& as_font() { return const_cast<FontStyleValue&>(const_cast<StyleValue const&>(*this).as_font()); }
|
|
IdentifierStyleValue& as_identifier() { return const_cast<IdentifierStyleValue&>(const_cast<StyleValue const&>(*this).as_identifier()); }
|
|
ImageStyleValue& as_image() { return const_cast<ImageStyleValue&>(const_cast<StyleValue const&>(*this).as_image()); }
|
|
InheritStyleValue& as_inherit() { return const_cast<InheritStyleValue&>(const_cast<StyleValue const&>(*this).as_inherit()); }
|
|
InitialStyleValue& as_initial() { return const_cast<InitialStyleValue&>(const_cast<StyleValue const&>(*this).as_initial()); }
|
|
LengthStyleValue& as_length() { return const_cast<LengthStyleValue&>(const_cast<StyleValue const&>(*this).as_length()); }
|
|
ListStyleStyleValue& as_list_style() { return const_cast<ListStyleStyleValue&>(const_cast<StyleValue const&>(*this).as_list_style()); }
|
|
NumericStyleValue& as_numeric() { return const_cast<NumericStyleValue&>(const_cast<StyleValue const&>(*this).as_numeric()); }
|
|
OverflowStyleValue& as_overflow() { return const_cast<OverflowStyleValue&>(const_cast<StyleValue const&>(*this).as_overflow()); }
|
|
PercentageStyleValue& as_percentage() { return const_cast<PercentageStyleValue&>(const_cast<StyleValue const&>(*this).as_percentage()); }
|
|
PositionStyleValue& as_position() { return const_cast<PositionStyleValue&>(const_cast<StyleValue const&>(*this).as_position()); }
|
|
StringStyleValue& as_string() { return const_cast<StringStyleValue&>(const_cast<StyleValue const&>(*this).as_string()); }
|
|
TextDecorationStyleValue& as_text_decoration() { return const_cast<TextDecorationStyleValue&>(const_cast<StyleValue const&>(*this).as_text_decoration()); }
|
|
TransformationStyleValue& as_transformation() { return const_cast<TransformationStyleValue&>(const_cast<StyleValue const&>(*this).as_transformation()); }
|
|
UnresolvedStyleValue& as_unresolved() { return const_cast<UnresolvedStyleValue&>(const_cast<StyleValue const&>(*this).as_unresolved()); }
|
|
UnsetStyleValue& as_unset() { return const_cast<UnsetStyleValue&>(const_cast<StyleValue const&>(*this).as_unset()); }
|
|
StyleValueList& as_value_list() { return const_cast<StyleValueList&>(const_cast<StyleValue const&>(*this).as_value_list()); }
|
|
|
|
virtual bool has_auto() const { return false; }
|
|
virtual bool has_color() const { return false; }
|
|
virtual bool has_identifier() const { return false; }
|
|
virtual bool has_length() const { return false; }
|
|
virtual bool has_number() const { return false; }
|
|
virtual bool has_integer() const { return false; }
|
|
|
|
virtual Color to_color(Layout::NodeWithStyle const&) const { return {}; }
|
|
virtual CSS::ValueID to_identifier() const { return ValueID::Invalid; }
|
|
virtual Length to_length() const { return {}; }
|
|
virtual float to_number() const { return 0; }
|
|
virtual float to_integer() const { return 0; }
|
|
virtual String to_string() const = 0;
|
|
|
|
bool operator==(StyleValue const& other) const { return equals(other); }
|
|
bool operator!=(StyleValue const& other) const { return !(*this == other); }
|
|
|
|
virtual bool equals(StyleValue const& other) const
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
return to_string() == other.to_string();
|
|
}
|
|
|
|
virtual void visit_lengths(Function<void(CSS::Length&)>) { }
|
|
|
|
protected:
|
|
explicit StyleValue(Type);
|
|
|
|
private:
|
|
Type m_type { Type::Invalid };
|
|
};
|
|
|
|
class BackgroundStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<BackgroundStyleValue> create(
|
|
NonnullRefPtr<StyleValue> color,
|
|
NonnullRefPtr<StyleValue> image,
|
|
NonnullRefPtr<StyleValue> position,
|
|
NonnullRefPtr<StyleValue> size,
|
|
NonnullRefPtr<StyleValue> repeat,
|
|
NonnullRefPtr<StyleValue> attachment,
|
|
NonnullRefPtr<StyleValue> origin,
|
|
NonnullRefPtr<StyleValue> clip)
|
|
{
|
|
return adopt_ref(*new BackgroundStyleValue(color, image, position, size, repeat, attachment, origin, clip));
|
|
}
|
|
virtual ~BackgroundStyleValue() override { }
|
|
|
|
size_t layer_count() const { return m_layer_count; }
|
|
|
|
NonnullRefPtr<StyleValue> attachment() const { return m_attachment; }
|
|
NonnullRefPtr<StyleValue> clip() const { return m_clip; }
|
|
NonnullRefPtr<StyleValue> color() const { return m_color; }
|
|
NonnullRefPtr<StyleValue> image() const { return m_image; }
|
|
NonnullRefPtr<StyleValue> origin() const { return m_origin; }
|
|
NonnullRefPtr<StyleValue> position() const { return m_position; }
|
|
NonnullRefPtr<StyleValue> repeat() const { return m_repeat; }
|
|
NonnullRefPtr<StyleValue> size() const { return m_size; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
BackgroundStyleValue(
|
|
NonnullRefPtr<StyleValue> color,
|
|
NonnullRefPtr<StyleValue> image,
|
|
NonnullRefPtr<StyleValue> position,
|
|
NonnullRefPtr<StyleValue> size,
|
|
NonnullRefPtr<StyleValue> repeat,
|
|
NonnullRefPtr<StyleValue> attachment,
|
|
NonnullRefPtr<StyleValue> origin,
|
|
NonnullRefPtr<StyleValue> clip);
|
|
|
|
NonnullRefPtr<StyleValue> m_color;
|
|
NonnullRefPtr<StyleValue> m_image;
|
|
NonnullRefPtr<StyleValue> m_position;
|
|
NonnullRefPtr<StyleValue> m_size;
|
|
NonnullRefPtr<StyleValue> m_repeat;
|
|
NonnullRefPtr<StyleValue> m_attachment;
|
|
NonnullRefPtr<StyleValue> m_origin;
|
|
NonnullRefPtr<StyleValue> m_clip;
|
|
|
|
size_t m_layer_count;
|
|
};
|
|
|
|
class BackgroundRepeatStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<BackgroundRepeatStyleValue> create(Repeat repeat_x, Repeat repeat_y)
|
|
{
|
|
return adopt_ref(*new BackgroundRepeatStyleValue(repeat_x, repeat_y));
|
|
}
|
|
virtual ~BackgroundRepeatStyleValue() override { }
|
|
|
|
Repeat repeat_x() const { return m_repeat_x; }
|
|
Repeat repeat_y() const { return m_repeat_y; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
BackgroundRepeatStyleValue(Repeat repeat_x, Repeat repeat_y)
|
|
: StyleValue(Type::BackgroundRepeat)
|
|
, m_repeat_x(repeat_x)
|
|
, m_repeat_y(repeat_y)
|
|
{
|
|
}
|
|
|
|
Repeat m_repeat_x;
|
|
Repeat m_repeat_y;
|
|
};
|
|
|
|
// NOTE: This is not used for identifier sizes, like `cover` and `contain`.
|
|
class BackgroundSizeStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<BackgroundSizeStyleValue> create(LengthPercentage size_x, LengthPercentage size_y)
|
|
{
|
|
return adopt_ref(*new BackgroundSizeStyleValue(size_x, size_y));
|
|
}
|
|
virtual ~BackgroundSizeStyleValue() override { }
|
|
|
|
LengthPercentage size_x() const { return m_size_x; }
|
|
LengthPercentage size_y() const { return m_size_y; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
BackgroundSizeStyleValue(LengthPercentage size_x, LengthPercentage size_y)
|
|
: StyleValue(Type::BackgroundSize)
|
|
, m_size_x(size_x)
|
|
, m_size_y(size_y)
|
|
{
|
|
}
|
|
|
|
LengthPercentage m_size_x;
|
|
LengthPercentage m_size_y;
|
|
};
|
|
|
|
class BorderStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<BorderStyleValue> create(
|
|
NonnullRefPtr<StyleValue> border_width,
|
|
NonnullRefPtr<StyleValue> border_style,
|
|
NonnullRefPtr<StyleValue> border_color)
|
|
{
|
|
return adopt_ref(*new BorderStyleValue(border_width, border_style, border_color));
|
|
}
|
|
virtual ~BorderStyleValue() override { }
|
|
|
|
NonnullRefPtr<StyleValue> border_width() const { return m_border_width; }
|
|
NonnullRefPtr<StyleValue> border_style() const { return m_border_style; }
|
|
NonnullRefPtr<StyleValue> border_color() const { return m_border_color; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
BorderStyleValue(
|
|
NonnullRefPtr<StyleValue> border_width,
|
|
NonnullRefPtr<StyleValue> border_style,
|
|
NonnullRefPtr<StyleValue> border_color)
|
|
: StyleValue(Type::Border)
|
|
, m_border_width(border_width)
|
|
, m_border_style(border_style)
|
|
, m_border_color(border_color)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_border_width;
|
|
NonnullRefPtr<StyleValue> m_border_style;
|
|
NonnullRefPtr<StyleValue> m_border_color;
|
|
};
|
|
|
|
class BorderRadiusStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<BorderRadiusStyleValue> create(LengthPercentage const& horizontal_radius, LengthPercentage const& vertical_radius)
|
|
{
|
|
return adopt_ref(*new BorderRadiusStyleValue(horizontal_radius, vertical_radius));
|
|
}
|
|
virtual ~BorderRadiusStyleValue() override { }
|
|
|
|
LengthPercentage const& horizontal_radius() const { return m_horizontal_radius; }
|
|
LengthPercentage const& vertical_radius() const { return m_vertical_radius; }
|
|
bool is_elliptical() const { return m_is_elliptical; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
virtual bool equals(StyleValue const& other) const override
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
auto& other_value = static_cast<BorderRadiusStyleValue const&>(other);
|
|
return m_is_elliptical == other_value.m_is_elliptical
|
|
&& m_horizontal_radius == other_value.m_horizontal_radius
|
|
&& m_vertical_radius == other_value.m_vertical_radius;
|
|
}
|
|
|
|
private:
|
|
BorderRadiusStyleValue(LengthPercentage const& horizontal_radius, LengthPercentage const& vertical_radius)
|
|
: StyleValue(Type::BorderRadius)
|
|
, m_horizontal_radius(horizontal_radius)
|
|
, m_vertical_radius(vertical_radius)
|
|
{
|
|
m_is_elliptical = (m_horizontal_radius != m_vertical_radius);
|
|
}
|
|
|
|
virtual void visit_lengths(Function<void(CSS::Length&)> visitor) override
|
|
{
|
|
if (!m_horizontal_radius.is_percentage()) {
|
|
Length temp = m_horizontal_radius.length();
|
|
visitor(temp);
|
|
m_horizontal_radius = move(temp);
|
|
}
|
|
if (!m_vertical_radius.is_percentage()) {
|
|
Length temp = m_vertical_radius.length();
|
|
visitor(temp);
|
|
m_vertical_radius = move(temp);
|
|
}
|
|
}
|
|
|
|
bool m_is_elliptical;
|
|
LengthPercentage m_horizontal_radius;
|
|
LengthPercentage m_vertical_radius;
|
|
};
|
|
|
|
class BoxShadowStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<BoxShadowStyleValue> create(Length const& offset_x, Length const& offset_y, Length const& blur_radius, Color const& color)
|
|
{
|
|
return adopt_ref(*new BoxShadowStyleValue(offset_x, offset_y, blur_radius, color));
|
|
}
|
|
virtual ~BoxShadowStyleValue() override { }
|
|
|
|
// FIXME: Spread-distance and "inset" flag
|
|
Length const& offset_x() const { return m_offset_x; }
|
|
Length const& offset_y() const { return m_offset_y; }
|
|
Length const& blur_radius() const { return m_blur_radius; }
|
|
Color const& color() const { return m_color; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
explicit BoxShadowStyleValue(Length const& offset_x, Length const& offset_y, Length const& blur_radius, Color const& color)
|
|
: StyleValue(Type::BoxShadow)
|
|
, m_offset_x(offset_x)
|
|
, m_offset_y(offset_y)
|
|
, m_blur_radius(blur_radius)
|
|
, m_color(color)
|
|
{
|
|
}
|
|
|
|
virtual void visit_lengths(Function<void(CSS::Length&)> visitor) override
|
|
{
|
|
visitor(m_offset_x);
|
|
visitor(m_offset_y);
|
|
visitor(m_blur_radius);
|
|
}
|
|
|
|
Length m_offset_x;
|
|
Length m_offset_y;
|
|
Length m_blur_radius;
|
|
Color m_color;
|
|
};
|
|
|
|
class CalculatedStyleValue : public StyleValue {
|
|
public:
|
|
enum class ResolvedType {
|
|
Angle,
|
|
Frequency,
|
|
Integer,
|
|
Length,
|
|
Number,
|
|
Percentage,
|
|
Time,
|
|
};
|
|
|
|
enum class SumOperation {
|
|
Add,
|
|
Subtract,
|
|
};
|
|
enum class ProductOperation {
|
|
Multiply,
|
|
Divide,
|
|
};
|
|
|
|
class CalculationResult {
|
|
public:
|
|
CalculationResult(Variant<float, Length, Percentage> value)
|
|
: m_value(move(value))
|
|
{
|
|
}
|
|
void add(CalculationResult const& other, Layout::Node const*, Length const& percentage_basis);
|
|
void subtract(CalculationResult const& other, Layout::Node const*, Length const& percentage_basis);
|
|
void multiply_by(CalculationResult const& other, Layout::Node const*);
|
|
void divide_by(CalculationResult const& other, Layout::Node const*);
|
|
|
|
Variant<float, Length, Percentage> const& value() const { return m_value; }
|
|
|
|
private:
|
|
void add_or_subtract_internal(SumOperation op, CalculationResult const& other, Layout::Node const*, Length const& percentage_basis);
|
|
Variant<float, Length, Percentage> m_value;
|
|
};
|
|
|
|
struct CalcSum;
|
|
struct CalcSumPartWithOperator;
|
|
struct CalcProduct;
|
|
struct CalcProductPartWithOperator;
|
|
struct CalcNumberSum;
|
|
struct CalcNumberSumPartWithOperator;
|
|
struct CalcNumberProduct;
|
|
struct CalcNumberProductPartWithOperator;
|
|
|
|
struct CalcNumberValue {
|
|
Variant<float, NonnullOwnPtr<CalcNumberSum>> value;
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcValue {
|
|
Variant<float, CSS::Length, NonnullOwnPtr<CalcSum>> value;
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
// This represents that: https://www.w3.org/TR/css-values-3/#calc-syntax
|
|
struct CalcSum {
|
|
CalcSum(NonnullOwnPtr<CalcProduct> first_calc_product, NonnullOwnPtrVector<CalcSumPartWithOperator> additional)
|
|
: first_calc_product(move(first_calc_product))
|
|
, zero_or_more_additional_calc_products(move(additional)) {};
|
|
|
|
NonnullOwnPtr<CalcProduct> first_calc_product;
|
|
NonnullOwnPtrVector<CalcSumPartWithOperator> zero_or_more_additional_calc_products;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcNumberSum {
|
|
CalcNumberSum(NonnullOwnPtr<CalcNumberProduct> first_calc_number_product, NonnullOwnPtrVector<CalcNumberSumPartWithOperator> additional)
|
|
: first_calc_number_product(move(first_calc_number_product))
|
|
, zero_or_more_additional_calc_number_products(move(additional)) {};
|
|
|
|
NonnullOwnPtr<CalcNumberProduct> first_calc_number_product;
|
|
NonnullOwnPtrVector<CalcNumberSumPartWithOperator> zero_or_more_additional_calc_number_products;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcProduct {
|
|
CalcValue first_calc_value;
|
|
NonnullOwnPtrVector<CalcProductPartWithOperator> zero_or_more_additional_calc_values;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcSumPartWithOperator {
|
|
CalcSumPartWithOperator(SumOperation op, NonnullOwnPtr<CalcProduct> calc_product)
|
|
: op(op)
|
|
, value(move(calc_product)) {};
|
|
|
|
SumOperation op;
|
|
NonnullOwnPtr<CalcProduct> value;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcProductPartWithOperator {
|
|
ProductOperation op;
|
|
Variant<CalcValue, CalcNumberValue> value;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcNumberProduct {
|
|
CalcNumberValue first_calc_number_value;
|
|
NonnullOwnPtrVector<CalcNumberProductPartWithOperator> zero_or_more_additional_calc_number_values;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcNumberProductPartWithOperator {
|
|
ProductOperation op;
|
|
CalcNumberValue value;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
struct CalcNumberSumPartWithOperator {
|
|
CalcNumberSumPartWithOperator(SumOperation op, NonnullOwnPtr<CalcNumberProduct> calc_number_product)
|
|
: op(op)
|
|
, value(move(calc_number_product)) {};
|
|
|
|
SumOperation op;
|
|
NonnullOwnPtr<CalcNumberProduct> value;
|
|
|
|
Optional<ResolvedType> resolved_type() const;
|
|
};
|
|
|
|
static NonnullRefPtr<CalculatedStyleValue> create(String const& expression_string, NonnullOwnPtr<CalcSum> calc_sum, ResolvedType resolved_type)
|
|
{
|
|
return adopt_ref(*new CalculatedStyleValue(expression_string, move(calc_sum), resolved_type));
|
|
}
|
|
|
|
String to_string() const override { return m_expression_string; }
|
|
ResolvedType resolved_type() const { return m_resolved_type; }
|
|
NonnullOwnPtr<CalcSum> const& expression() const { return m_expression; }
|
|
Optional<Length> resolve_length(Layout::Node const& layout_node) const;
|
|
|
|
private:
|
|
explicit CalculatedStyleValue(String const& expression_string, NonnullOwnPtr<CalcSum> calc_sum, ResolvedType resolved_type)
|
|
: StyleValue(Type::Calculated)
|
|
, m_resolved_type(resolved_type)
|
|
, m_expression_string(expression_string)
|
|
, m_expression(move(calc_sum))
|
|
{
|
|
}
|
|
|
|
ResolvedType m_resolved_type;
|
|
String m_expression_string;
|
|
NonnullOwnPtr<CalcSum> m_expression;
|
|
};
|
|
|
|
class ColorStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<ColorStyleValue> create(Color color)
|
|
{
|
|
return adopt_ref(*new ColorStyleValue(color));
|
|
}
|
|
virtual ~ColorStyleValue() override { }
|
|
|
|
Color color() const { return m_color; }
|
|
virtual String to_string() const override;
|
|
virtual bool has_color() const override { return true; }
|
|
virtual Color to_color(Layout::NodeWithStyle const&) const override { return m_color; }
|
|
|
|
virtual bool equals(StyleValue const& other) const override
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
return m_color == static_cast<const ColorStyleValue&>(other).m_color;
|
|
}
|
|
|
|
private:
|
|
explicit ColorStyleValue(Color color)
|
|
: StyleValue(Type::Color)
|
|
, m_color(color)
|
|
{
|
|
}
|
|
|
|
Color m_color;
|
|
};
|
|
|
|
class CombinedBorderRadiusStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<CombinedBorderRadiusStyleValue> create(NonnullRefPtr<BorderRadiusStyleValue> top_left, NonnullRefPtr<BorderRadiusStyleValue> top_right, NonnullRefPtr<BorderRadiusStyleValue> bottom_right, NonnullRefPtr<BorderRadiusStyleValue> bottom_left)
|
|
{
|
|
return adopt_ref(*new CombinedBorderRadiusStyleValue(top_left, top_right, bottom_right, bottom_left));
|
|
}
|
|
virtual ~CombinedBorderRadiusStyleValue() override { }
|
|
|
|
NonnullRefPtr<BorderRadiusStyleValue> top_left() const { return m_top_left; }
|
|
NonnullRefPtr<BorderRadiusStyleValue> top_right() const { return m_top_right; }
|
|
NonnullRefPtr<BorderRadiusStyleValue> bottom_right() const { return m_bottom_right; }
|
|
NonnullRefPtr<BorderRadiusStyleValue> bottom_left() const { return m_bottom_left; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
CombinedBorderRadiusStyleValue(NonnullRefPtr<BorderRadiusStyleValue> top_left, NonnullRefPtr<BorderRadiusStyleValue> top_right, NonnullRefPtr<BorderRadiusStyleValue> bottom_right, NonnullRefPtr<BorderRadiusStyleValue> bottom_left)
|
|
: StyleValue(Type::CombinedBorderRadius)
|
|
, m_top_left(top_left)
|
|
, m_top_right(top_right)
|
|
, m_bottom_right(bottom_right)
|
|
, m_bottom_left(bottom_left)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<BorderRadiusStyleValue> m_top_left;
|
|
NonnullRefPtr<BorderRadiusStyleValue> m_top_right;
|
|
NonnullRefPtr<BorderRadiusStyleValue> m_bottom_right;
|
|
NonnullRefPtr<BorderRadiusStyleValue> m_bottom_left;
|
|
};
|
|
|
|
class FlexStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<FlexStyleValue> create(
|
|
NonnullRefPtr<StyleValue> grow,
|
|
NonnullRefPtr<StyleValue> shrink,
|
|
NonnullRefPtr<StyleValue> basis)
|
|
{
|
|
return adopt_ref(*new FlexStyleValue(grow, shrink, basis));
|
|
}
|
|
virtual ~FlexStyleValue() override { }
|
|
|
|
NonnullRefPtr<StyleValue> grow() const { return m_grow; }
|
|
NonnullRefPtr<StyleValue> shrink() const { return m_shrink; }
|
|
NonnullRefPtr<StyleValue> basis() const { return m_basis; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
FlexStyleValue(
|
|
NonnullRefPtr<StyleValue> grow,
|
|
NonnullRefPtr<StyleValue> shrink,
|
|
NonnullRefPtr<StyleValue> basis)
|
|
: StyleValue(Type::Flex)
|
|
, m_grow(grow)
|
|
, m_shrink(shrink)
|
|
, m_basis(basis)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_grow;
|
|
NonnullRefPtr<StyleValue> m_shrink;
|
|
NonnullRefPtr<StyleValue> m_basis;
|
|
};
|
|
|
|
class FlexFlowStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<FlexFlowStyleValue> create(NonnullRefPtr<StyleValue> flex_direction, NonnullRefPtr<StyleValue> flex_wrap)
|
|
{
|
|
return adopt_ref(*new FlexFlowStyleValue(flex_direction, flex_wrap));
|
|
}
|
|
virtual ~FlexFlowStyleValue() override { }
|
|
|
|
NonnullRefPtr<StyleValue> flex_direction() const { return m_flex_direction; }
|
|
NonnullRefPtr<StyleValue> flex_wrap() const { return m_flex_wrap; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
FlexFlowStyleValue(NonnullRefPtr<StyleValue> flex_direction, NonnullRefPtr<StyleValue> flex_wrap)
|
|
: StyleValue(Type::FlexFlow)
|
|
, m_flex_direction(flex_direction)
|
|
, m_flex_wrap(flex_wrap)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_flex_direction;
|
|
NonnullRefPtr<StyleValue> m_flex_wrap;
|
|
};
|
|
|
|
class FontStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<FontStyleValue> create(NonnullRefPtr<StyleValue> font_style, NonnullRefPtr<StyleValue> font_weight, NonnullRefPtr<StyleValue> font_size, NonnullRefPtr<StyleValue> line_height, NonnullRefPtr<StyleValue> font_families) { return adopt_ref(*new FontStyleValue(font_style, font_weight, font_size, line_height, font_families)); }
|
|
virtual ~FontStyleValue() override { }
|
|
|
|
NonnullRefPtr<StyleValue> font_style() const { return m_font_style; }
|
|
NonnullRefPtr<StyleValue> font_weight() const { return m_font_weight; }
|
|
NonnullRefPtr<StyleValue> font_size() const { return m_font_size; }
|
|
NonnullRefPtr<StyleValue> line_height() const { return m_line_height; }
|
|
NonnullRefPtr<StyleValue> font_families() const { return m_font_families; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
FontStyleValue(NonnullRefPtr<StyleValue> font_style, NonnullRefPtr<StyleValue> font_weight, NonnullRefPtr<StyleValue> font_size, NonnullRefPtr<StyleValue> line_height, NonnullRefPtr<StyleValue> font_families)
|
|
: StyleValue(Type::Font)
|
|
, m_font_style(font_style)
|
|
, m_font_weight(font_weight)
|
|
, m_font_size(font_size)
|
|
, m_line_height(line_height)
|
|
, m_font_families(font_families)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_font_style;
|
|
NonnullRefPtr<StyleValue> m_font_weight;
|
|
NonnullRefPtr<StyleValue> m_font_size;
|
|
NonnullRefPtr<StyleValue> m_line_height;
|
|
NonnullRefPtr<StyleValue> m_font_families;
|
|
// FIXME: Implement font-stretch and font-variant.
|
|
};
|
|
|
|
class IdentifierStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<IdentifierStyleValue> create(CSS::ValueID id)
|
|
{
|
|
return adopt_ref(*new IdentifierStyleValue(id));
|
|
}
|
|
virtual ~IdentifierStyleValue() override { }
|
|
|
|
CSS::ValueID id() const { return m_id; }
|
|
|
|
virtual bool has_auto() const override { return m_id == ValueID::Auto; }
|
|
virtual bool has_identifier() const override { return true; }
|
|
virtual CSS::ValueID to_identifier() const override { return m_id; }
|
|
virtual bool has_color() const override;
|
|
virtual Color to_color(Layout::NodeWithStyle const& node) const override;
|
|
virtual String to_string() const override;
|
|
|
|
virtual bool equals(StyleValue const& other) const override
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
return m_id == static_cast<const IdentifierStyleValue&>(other).m_id;
|
|
}
|
|
|
|
private:
|
|
explicit IdentifierStyleValue(CSS::ValueID id)
|
|
: StyleValue(Type::Identifier)
|
|
, m_id(id)
|
|
{
|
|
}
|
|
|
|
CSS::ValueID m_id { CSS::ValueID::Invalid };
|
|
};
|
|
|
|
class ImageStyleValue final
|
|
: public StyleValue
|
|
, public ImageResourceClient {
|
|
public:
|
|
static NonnullRefPtr<ImageStyleValue> create(AK::URL const& url) { return adopt_ref(*new ImageStyleValue(url)); }
|
|
virtual ~ImageStyleValue() override { }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
void load_bitmap(DOM::Document& document);
|
|
Gfx::Bitmap const* bitmap() const { return m_bitmap; }
|
|
|
|
private:
|
|
ImageStyleValue(AK::URL const&);
|
|
|
|
// ^ResourceClient
|
|
virtual void resource_did_load() override;
|
|
|
|
AK::URL m_url;
|
|
WeakPtr<DOM::Document> m_document;
|
|
RefPtr<Gfx::Bitmap> m_bitmap;
|
|
};
|
|
|
|
class InheritStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<InheritStyleValue> the()
|
|
{
|
|
static NonnullRefPtr<InheritStyleValue> instance = adopt_ref(*new InheritStyleValue);
|
|
return instance;
|
|
}
|
|
virtual ~InheritStyleValue() override { }
|
|
|
|
String to_string() const override { return "inherit"; }
|
|
|
|
private:
|
|
InheritStyleValue()
|
|
: StyleValue(Type::Inherit)
|
|
{
|
|
}
|
|
};
|
|
|
|
class InitialStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<InitialStyleValue> the()
|
|
{
|
|
static NonnullRefPtr<InitialStyleValue> instance = adopt_ref(*new InitialStyleValue);
|
|
return instance;
|
|
}
|
|
virtual ~InitialStyleValue() override { }
|
|
|
|
String to_string() const override { return "initial"; }
|
|
|
|
private:
|
|
InitialStyleValue()
|
|
: StyleValue(Type::Initial)
|
|
{
|
|
}
|
|
};
|
|
|
|
class LengthStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<LengthStyleValue> create(Length const& length)
|
|
{
|
|
return adopt_ref(*new LengthStyleValue(length));
|
|
}
|
|
virtual ~LengthStyleValue() override { }
|
|
|
|
Length const& length() const { return m_length; }
|
|
|
|
virtual bool has_auto() const override { return m_length.is_auto(); }
|
|
virtual bool has_length() const override { return true; }
|
|
virtual bool has_identifier() const override { return has_auto(); }
|
|
virtual String to_string() const override { return m_length.to_string(); }
|
|
virtual Length to_length() const override { return m_length; }
|
|
virtual ValueID to_identifier() const override { return has_auto() ? ValueID::Auto : ValueID::Invalid; }
|
|
|
|
virtual bool equals(StyleValue const& other) const override
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
return m_length == static_cast<LengthStyleValue const&>(other).m_length;
|
|
}
|
|
|
|
private:
|
|
explicit LengthStyleValue(Length const& length)
|
|
: StyleValue(Type::Length)
|
|
, m_length(length)
|
|
{
|
|
}
|
|
|
|
virtual void visit_lengths(Function<void(CSS::Length&)> visitor) override
|
|
{
|
|
visitor(m_length);
|
|
}
|
|
|
|
Length m_length;
|
|
};
|
|
|
|
class ListStyleStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<ListStyleStyleValue> create(
|
|
NonnullRefPtr<StyleValue> position,
|
|
NonnullRefPtr<StyleValue> image,
|
|
NonnullRefPtr<StyleValue> style_type)
|
|
{
|
|
return adopt_ref(*new ListStyleStyleValue(position, image, style_type));
|
|
}
|
|
virtual ~ListStyleStyleValue() override { }
|
|
|
|
NonnullRefPtr<StyleValue> position() const { return m_position; }
|
|
NonnullRefPtr<StyleValue> image() const { return m_image; }
|
|
NonnullRefPtr<StyleValue> style_type() const { return m_style_type; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
ListStyleStyleValue(
|
|
NonnullRefPtr<StyleValue> position,
|
|
NonnullRefPtr<StyleValue> image,
|
|
NonnullRefPtr<StyleValue> style_type)
|
|
: StyleValue(Type::ListStyle)
|
|
, m_position(position)
|
|
, m_image(image)
|
|
, m_style_type(style_type)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_position;
|
|
NonnullRefPtr<StyleValue> m_image;
|
|
NonnullRefPtr<StyleValue> m_style_type;
|
|
};
|
|
|
|
class NumericStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<NumericStyleValue> create_float(float value)
|
|
{
|
|
return adopt_ref(*new NumericStyleValue(value));
|
|
}
|
|
|
|
static NonnullRefPtr<NumericStyleValue> create_integer(i64 value)
|
|
{
|
|
return adopt_ref(*new NumericStyleValue(value));
|
|
}
|
|
|
|
virtual bool has_length() const override { return to_number() == 0; }
|
|
virtual Length to_length() const override { return Length(0, Length::Type::Px); }
|
|
|
|
virtual bool has_number() const override { return true; }
|
|
virtual float to_number() const override
|
|
{
|
|
return m_value.visit(
|
|
[](float value) { return value; },
|
|
[](i64 value) { return (float)value; });
|
|
}
|
|
|
|
virtual bool has_integer() const override { return m_value.has<i64>(); }
|
|
virtual float to_integer() const override { return m_value.get<i64>(); }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
virtual bool equals(StyleValue const& other) const override
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
if (has_integer() != other.has_integer())
|
|
return false;
|
|
if (has_integer())
|
|
return m_value.get<i64>() == static_cast<NumericStyleValue const&>(other).m_value.get<i64>();
|
|
return m_value.get<float>() == static_cast<NumericStyleValue const&>(other).m_value.get<float>();
|
|
}
|
|
|
|
private:
|
|
explicit NumericStyleValue(Variant<float, i64> value)
|
|
: StyleValue(Type::Numeric)
|
|
, m_value(move(value))
|
|
{
|
|
}
|
|
|
|
Variant<float, i64> m_value { (i64)0 };
|
|
};
|
|
|
|
class OverflowStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<OverflowStyleValue> create(NonnullRefPtr<StyleValue> overflow_x, NonnullRefPtr<StyleValue> overflow_y)
|
|
{
|
|
return adopt_ref(*new OverflowStyleValue(overflow_x, overflow_y));
|
|
}
|
|
virtual ~OverflowStyleValue() override { }
|
|
|
|
NonnullRefPtr<StyleValue> overflow_x() const { return m_overflow_x; }
|
|
NonnullRefPtr<StyleValue> overflow_y() const { return m_overflow_y; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
OverflowStyleValue(NonnullRefPtr<StyleValue> overflow_x, NonnullRefPtr<StyleValue> overflow_y)
|
|
: StyleValue(Type::Overflow)
|
|
, m_overflow_x(overflow_x)
|
|
, m_overflow_y(overflow_y)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_overflow_x;
|
|
NonnullRefPtr<StyleValue> m_overflow_y;
|
|
};
|
|
|
|
class PercentageStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<PercentageStyleValue> create(Percentage percentage)
|
|
{
|
|
return adopt_ref(*new PercentageStyleValue(move(percentage)));
|
|
}
|
|
virtual ~PercentageStyleValue() override { }
|
|
|
|
Percentage const& percentage() const { return m_percentage; }
|
|
Percentage& percentage() { return m_percentage; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
PercentageStyleValue(Percentage&& percentage)
|
|
: StyleValue(Type::Percentage)
|
|
, m_percentage(percentage)
|
|
{
|
|
}
|
|
|
|
Percentage m_percentage;
|
|
};
|
|
|
|
class PositionStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<PositionStyleValue> create(PositionEdge edge_x, LengthPercentage const& offset_x, PositionEdge edge_y, LengthPercentage const& offset_y)
|
|
{
|
|
return adopt_ref(*new PositionStyleValue(edge_x, offset_x, edge_y, offset_y));
|
|
}
|
|
virtual ~PositionStyleValue() override { }
|
|
|
|
PositionEdge edge_x() const { return m_edge_x; }
|
|
LengthPercentage const& offset_x() const { return m_offset_x; }
|
|
PositionEdge edge_y() const { return m_edge_y; }
|
|
LengthPercentage const& offset_y() const { return m_offset_y; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
PositionStyleValue(PositionEdge edge_x, LengthPercentage const& offset_x, PositionEdge edge_y, LengthPercentage const& offset_y)
|
|
: StyleValue(Type::Position)
|
|
, m_edge_x(edge_x)
|
|
, m_offset_x(offset_x)
|
|
, m_edge_y(edge_y)
|
|
, m_offset_y(offset_y)
|
|
{
|
|
}
|
|
|
|
PositionEdge m_edge_x;
|
|
LengthPercentage m_offset_x;
|
|
PositionEdge m_edge_y;
|
|
LengthPercentage m_offset_y;
|
|
};
|
|
|
|
class StringStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<StringStyleValue> create(String const& string)
|
|
{
|
|
return adopt_ref(*new StringStyleValue(string));
|
|
}
|
|
virtual ~StringStyleValue() override { }
|
|
|
|
String to_string() const override { return m_string; }
|
|
|
|
private:
|
|
explicit StringStyleValue(String const& string)
|
|
: StyleValue(Type::String)
|
|
, m_string(string)
|
|
{
|
|
}
|
|
|
|
String m_string;
|
|
};
|
|
|
|
class TextDecorationStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<TextDecorationStyleValue> create(
|
|
NonnullRefPtr<StyleValue> line,
|
|
NonnullRefPtr<StyleValue> style,
|
|
NonnullRefPtr<StyleValue> color)
|
|
{
|
|
return adopt_ref(*new TextDecorationStyleValue(line, style, color));
|
|
}
|
|
virtual ~TextDecorationStyleValue() override { }
|
|
|
|
NonnullRefPtr<StyleValue> line() const { return m_line; }
|
|
NonnullRefPtr<StyleValue> style() const { return m_style; }
|
|
NonnullRefPtr<StyleValue> color() const { return m_color; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
TextDecorationStyleValue(
|
|
NonnullRefPtr<StyleValue> line,
|
|
NonnullRefPtr<StyleValue> style,
|
|
NonnullRefPtr<StyleValue> color)
|
|
: StyleValue(Type::TextDecoration)
|
|
, m_line(line)
|
|
, m_style(style)
|
|
, m_color(color)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_line;
|
|
NonnullRefPtr<StyleValue> m_style;
|
|
NonnullRefPtr<StyleValue> m_color;
|
|
};
|
|
|
|
class TransformationStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<TransformationStyleValue> create(CSS::TransformFunction transform_function, NonnullRefPtrVector<StyleValue>&& values)
|
|
{
|
|
return adopt_ref(*new TransformationStyleValue(transform_function, move(values)));
|
|
}
|
|
virtual ~TransformationStyleValue() override { }
|
|
|
|
CSS::TransformFunction transform_function() const { return m_transform_function; }
|
|
NonnullRefPtrVector<StyleValue> values() const { return m_values; }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
TransformationStyleValue(CSS::TransformFunction transform_function, NonnullRefPtrVector<StyleValue>&& values)
|
|
: StyleValue(Type::Transformation)
|
|
, m_transform_function(transform_function)
|
|
, m_values(move(values))
|
|
{
|
|
}
|
|
|
|
CSS::TransformFunction m_transform_function;
|
|
NonnullRefPtrVector<StyleValue> m_values;
|
|
};
|
|
|
|
class UnresolvedStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<UnresolvedStyleValue> create(Vector<StyleComponentValueRule>&& values, bool contains_var)
|
|
{
|
|
return adopt_ref(*new UnresolvedStyleValue(move(values), contains_var));
|
|
}
|
|
virtual ~UnresolvedStyleValue() override { }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
Vector<StyleComponentValueRule> const& values() const { return m_values; }
|
|
bool contains_var() const { return m_contains_var; }
|
|
|
|
private:
|
|
UnresolvedStyleValue(Vector<StyleComponentValueRule>&& values, bool contains_var)
|
|
: StyleValue(Type::Unresolved)
|
|
, m_values(move(values))
|
|
, m_contains_var(contains_var)
|
|
{
|
|
}
|
|
|
|
Vector<StyleComponentValueRule> m_values;
|
|
bool m_contains_var { false };
|
|
};
|
|
|
|
class UnsetStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<UnsetStyleValue> the()
|
|
{
|
|
static NonnullRefPtr<UnsetStyleValue> instance = adopt_ref(*new UnsetStyleValue);
|
|
return instance;
|
|
}
|
|
virtual ~UnsetStyleValue() override { }
|
|
|
|
String to_string() const override { return "unset"; }
|
|
|
|
private:
|
|
UnsetStyleValue()
|
|
: StyleValue(Type::Unset)
|
|
{
|
|
}
|
|
};
|
|
|
|
class StyleValueList final : public StyleValue {
|
|
public:
|
|
enum class Separator {
|
|
Space,
|
|
Comma,
|
|
};
|
|
static NonnullRefPtr<StyleValueList> create(NonnullRefPtrVector<StyleValue>&& values, Separator separator) { return adopt_ref(*new StyleValueList(move(values), separator)); }
|
|
|
|
size_t size() const { return m_values.size(); }
|
|
NonnullRefPtrVector<StyleValue> const& values() const { return m_values; }
|
|
NonnullRefPtr<StyleValue> value_at(size_t i, bool allow_loop) const
|
|
{
|
|
if (allow_loop)
|
|
return m_values[i % size()];
|
|
return m_values[i];
|
|
}
|
|
|
|
virtual String to_string() const override;
|
|
|
|
private:
|
|
StyleValueList(NonnullRefPtrVector<StyleValue>&& values, Separator separator)
|
|
: StyleValue(Type::ValueList)
|
|
, m_separator(separator)
|
|
, m_values(move(values))
|
|
{
|
|
}
|
|
|
|
Separator m_separator;
|
|
NonnullRefPtrVector<StyleValue> m_values;
|
|
};
|
|
|
|
}
|
|
|
|
template<>
|
|
struct AK::Formatter<Web::CSS::StyleValue> : Formatter<StringView> {
|
|
ErrorOr<void> format(FormatBuilder& builder, Web::CSS::StyleValue const& style_value)
|
|
{
|
|
return Formatter<StringView>::format(builder, style_value.to_string());
|
|
}
|
|
};
|