
After working with the code for a while, it makes more sense to put all the parsing in Parser, instead of some of it living in StyleResolver. That means our current ValueListStyleValue needs to be replaced with specific StyleValue types for the properties that are shorthands or otherwise combine several values together. Here we implement FontStyleProperty, which represents a `font` CSS property. Also adjusted the fonts.html test page so that font-weights are featured in test cases without things we do not yet support.
724 lines
19 KiB
C++
724 lines
19 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
* Copyright (c) 2021, Tobias Christiansen <tobyase@serenityos.org>
|
|
* Copyright (c) 2021, Sam Atkins <atkinssj@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#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/Length.h>
|
|
#include <LibWeb/CSS/Parser/StyleComponentValueRule.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 Position {
|
|
Static,
|
|
Relative,
|
|
Absolute,
|
|
Fixed,
|
|
Sticky,
|
|
};
|
|
|
|
enum class TextAlign {
|
|
Left,
|
|
Center,
|
|
Right,
|
|
Justify,
|
|
LibwebCenter,
|
|
};
|
|
|
|
enum class TextDecorationLine {
|
|
None,
|
|
Underline,
|
|
Overline,
|
|
LineThrough,
|
|
Blink,
|
|
};
|
|
|
|
enum class TextTransform {
|
|
None,
|
|
Capitalize,
|
|
Uppercase,
|
|
Lowercase,
|
|
FullWidth,
|
|
FullSizeKana,
|
|
};
|
|
|
|
enum class Display {
|
|
None,
|
|
Block,
|
|
Inline,
|
|
InlineBlock,
|
|
ListItem,
|
|
Table,
|
|
TableRow,
|
|
TableCell,
|
|
TableHeaderGroup,
|
|
TableRowGroup,
|
|
TableFooterGroup,
|
|
TableColumn,
|
|
TableColumnGroup,
|
|
TableCaption,
|
|
Flex,
|
|
};
|
|
|
|
enum class FlexDirection {
|
|
Row,
|
|
RowReverse,
|
|
Column,
|
|
ColumnReverse,
|
|
};
|
|
|
|
enum class FlexWrap {
|
|
Nowrap,
|
|
Wrap,
|
|
WrapReverse
|
|
};
|
|
|
|
enum class FlexBasis {
|
|
Content,
|
|
Length,
|
|
Auto,
|
|
};
|
|
|
|
enum class WhiteSpace {
|
|
Normal,
|
|
Pre,
|
|
Nowrap,
|
|
PreLine,
|
|
PreWrap,
|
|
};
|
|
|
|
enum class Float {
|
|
None,
|
|
Left,
|
|
Right,
|
|
};
|
|
|
|
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 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 Repeat : u8 {
|
|
NoRepeat,
|
|
Repeat,
|
|
Round,
|
|
Space,
|
|
};
|
|
|
|
enum class JustifyContent {
|
|
FlexStart,
|
|
FlexEnd,
|
|
Center,
|
|
SpaceBetween,
|
|
SpaceAround,
|
|
};
|
|
|
|
class StyleValue : public RefCounted<StyleValue> {
|
|
public:
|
|
virtual ~StyleValue();
|
|
|
|
enum class Type {
|
|
Invalid,
|
|
Inherit,
|
|
Initial,
|
|
String,
|
|
Length,
|
|
Color,
|
|
Identifier,
|
|
Image,
|
|
CustomProperty,
|
|
Numeric,
|
|
ValueList,
|
|
ComponentValueList,
|
|
Calculated,
|
|
BoxShadow,
|
|
Font,
|
|
};
|
|
|
|
Type type() const { return m_type; }
|
|
|
|
bool is_inherit() const { return type() == Type::Inherit; }
|
|
bool is_initial() const { return type() == Type::Initial; }
|
|
bool is_color() const { return type() == Type::Color; }
|
|
bool is_identifier() const { return type() == Type::Identifier || is_auto(); }
|
|
bool is_image() const { return type() == Type::Image; }
|
|
bool is_string() const { return type() == Type::String; }
|
|
virtual bool is_length() const { return type() == Type::Length; }
|
|
bool is_custom_property() const { return type() == Type::CustomProperty; }
|
|
bool is_numeric() const { return type() == Type::Numeric; }
|
|
bool is_value_list() const { return type() == Type::ValueList; }
|
|
bool is_component_value_list() const { return type() == Type::ComponentValueList; }
|
|
bool is_calculated() const { return type() == Type::Calculated; }
|
|
bool is_box_shadow() const { return type() == Type::BoxShadow; }
|
|
bool is_font() const { return type() == Type::Font; }
|
|
|
|
bool is_builtin() const { return is_inherit() || is_initial(); }
|
|
|
|
bool is_builtin_or_dynamic() const
|
|
{
|
|
return is_inherit() || is_initial() || is_custom_property() || is_calculated();
|
|
}
|
|
|
|
virtual String to_string() const = 0;
|
|
virtual Length to_length() const { return Length::make_auto(); }
|
|
virtual Color to_color(const DOM::Document&) const { return {}; }
|
|
|
|
CSS::ValueID to_identifier() const;
|
|
|
|
virtual bool is_auto() const { return false; }
|
|
|
|
bool operator==(const StyleValue& other) const { return equals(other); }
|
|
bool operator!=(const StyleValue& other) const { return !(*this == other); }
|
|
|
|
virtual bool equals(const StyleValue& other) const
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
return to_string() == other.to_string();
|
|
}
|
|
|
|
protected:
|
|
explicit StyleValue(Type);
|
|
|
|
private:
|
|
Type m_type { Type::Invalid };
|
|
};
|
|
|
|
// FIXME: Allow for fallback
|
|
class CustomStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<CustomStyleValue> create(const String& custom_property_name)
|
|
{
|
|
return adopt_ref(*new CustomStyleValue(custom_property_name));
|
|
}
|
|
String custom_property_name() const { return m_custom_property_name; }
|
|
String to_string() const override { return m_custom_property_name; }
|
|
|
|
private:
|
|
explicit CustomStyleValue(const String& custom_property_name)
|
|
: StyleValue(Type::CustomProperty)
|
|
, m_custom_property_name(custom_property_name)
|
|
{
|
|
}
|
|
|
|
String m_custom_property_name {};
|
|
};
|
|
|
|
class NumericStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<NumericStyleValue> create(float value)
|
|
{
|
|
return adopt_ref(*new NumericStyleValue(value));
|
|
}
|
|
|
|
virtual bool is_length() const override { return m_value == 0; }
|
|
virtual Length to_length() const override { return Length(0, Length::Type::Px); }
|
|
|
|
float value() const { return m_value; }
|
|
String to_string() const override { return String::formatted("{}", m_value); }
|
|
|
|
private:
|
|
explicit NumericStyleValue(float value)
|
|
: StyleValue(Type::Numeric)
|
|
, m_value(value)
|
|
{
|
|
}
|
|
|
|
float m_value { 0 };
|
|
};
|
|
|
|
class StringStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<StringStyleValue> create(const String& string)
|
|
{
|
|
return adopt_ref(*new StringStyleValue(string));
|
|
}
|
|
virtual ~StringStyleValue() override { }
|
|
|
|
String to_string() const override { return m_string; }
|
|
|
|
private:
|
|
explicit StringStyleValue(const String& string)
|
|
: StyleValue(Type::String)
|
|
, m_string(string)
|
|
{
|
|
}
|
|
|
|
String m_string;
|
|
};
|
|
|
|
class BoxShadowStyleValue : 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 { }
|
|
|
|
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; }
|
|
|
|
String to_string() const override { return String::formatted("BoxShadow offset_x: {}, offset_y: {}, blur_radius: {}, color: {}",
|
|
m_offset_x.to_string(), m_offset_y.to_string(), m_blur_radius.to_string(), m_color.to_string()); }
|
|
|
|
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)
|
|
{
|
|
}
|
|
|
|
Length m_offset_x;
|
|
Length m_offset_y;
|
|
Length m_blur_radius;
|
|
Color m_color;
|
|
};
|
|
|
|
class LengthStyleValue : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<LengthStyleValue> create(const Length& length)
|
|
{
|
|
return adopt_ref(*new LengthStyleValue(length));
|
|
}
|
|
virtual ~LengthStyleValue() override { }
|
|
|
|
virtual String to_string() const override { return m_length.to_string(); }
|
|
virtual Length to_length() const override { return m_length; }
|
|
|
|
const Length& length() const { return m_length; }
|
|
|
|
virtual bool is_auto() const override { return m_length.is_auto(); }
|
|
|
|
virtual bool equals(const StyleValue& other) const override
|
|
{
|
|
if (type() != other.type())
|
|
return false;
|
|
return m_length == static_cast<const LengthStyleValue&>(other).m_length;
|
|
}
|
|
|
|
private:
|
|
explicit LengthStyleValue(const Length& length)
|
|
: StyleValue(Type::Length)
|
|
, m_length(length)
|
|
{
|
|
}
|
|
|
|
Length m_length;
|
|
};
|
|
|
|
class CalculatedStyleValue : public StyleValue {
|
|
public:
|
|
struct CalcSum;
|
|
struct CalcSumPartWithOperator;
|
|
struct CalcProduct;
|
|
struct CalcProductPartWithOperator;
|
|
struct CalcNumberSum;
|
|
struct CalcNumberSumPartWithOperator;
|
|
struct CalcNumberProduct;
|
|
struct CalcNumberProductPartWithOperator;
|
|
|
|
using CalcNumberValue = Variant<float, NonnullOwnPtr<CalcNumberSum>>;
|
|
using CalcValue = Variant<float, CSS::Length, NonnullOwnPtr<CalcSum>>;
|
|
|
|
// This represents that: https://drafts.csswg.org/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;
|
|
};
|
|
|
|
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;
|
|
};
|
|
|
|
struct CalcProduct {
|
|
CalcValue first_calc_value;
|
|
NonnullOwnPtrVector<CalcProductPartWithOperator> zero_or_more_additional_calc_values;
|
|
};
|
|
|
|
struct CalcSumPartWithOperator {
|
|
enum Operation {
|
|
Add,
|
|
Subtract,
|
|
};
|
|
|
|
CalcSumPartWithOperator(Operation op, NonnullOwnPtr<CalcProduct> calc_product)
|
|
: op(op)
|
|
, calc_product(move(calc_product)) {};
|
|
|
|
Operation op;
|
|
NonnullOwnPtr<CalcProduct> calc_product;
|
|
};
|
|
|
|
struct CalcProductPartWithOperator {
|
|
enum {
|
|
Multiply,
|
|
Divide,
|
|
} op;
|
|
Variant<CalcValue, CalcNumberValue> value;
|
|
};
|
|
|
|
struct CalcNumberProduct {
|
|
CalcNumberValue first_calc_number_value;
|
|
NonnullOwnPtrVector<CalcNumberProductPartWithOperator> zero_or_more_additional_calc_number_values;
|
|
};
|
|
|
|
struct CalcNumberProductPartWithOperator {
|
|
enum {
|
|
Multiply,
|
|
Divide,
|
|
} op;
|
|
CalcNumberValue value;
|
|
};
|
|
|
|
struct CalcNumberSumPartWithOperator {
|
|
enum Operation {
|
|
Add,
|
|
Subtract,
|
|
};
|
|
|
|
CalcNumberSumPartWithOperator(Operation op, NonnullOwnPtr<CalcNumberProduct> calc_number_product)
|
|
: op(op)
|
|
, calc_number_product(move(calc_number_product)) {};
|
|
|
|
Operation op;
|
|
NonnullOwnPtr<CalcNumberProduct> calc_number_product;
|
|
};
|
|
|
|
static NonnullRefPtr<CalculatedStyleValue> create(String const& expression_string, NonnullOwnPtr<CalcSum> calc_sum)
|
|
{
|
|
return adopt_ref(*new CalculatedStyleValue(expression_string, move(calc_sum)));
|
|
}
|
|
|
|
String to_string() const override { return m_expression_string; }
|
|
NonnullOwnPtr<CalcSum> const& expression() const { return m_expression; }
|
|
|
|
private:
|
|
explicit CalculatedStyleValue(String const& expression_string, NonnullOwnPtr<CalcSum> calc_sum)
|
|
: StyleValue(Type::Calculated)
|
|
, m_expression_string(expression_string)
|
|
, m_expression(move(calc_sum))
|
|
{
|
|
}
|
|
|
|
String m_expression_string;
|
|
NonnullOwnPtr<CalcSum> m_expression;
|
|
};
|
|
|
|
class InitialStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<InitialStyleValue> create() { return adopt_ref(*new InitialStyleValue); }
|
|
virtual ~InitialStyleValue() override { }
|
|
|
|
String to_string() const override { return "initial"; }
|
|
|
|
private:
|
|
InitialStyleValue()
|
|
: StyleValue(Type::Initial)
|
|
{
|
|
}
|
|
};
|
|
|
|
class InheritStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<InheritStyleValue> create() { return adopt_ref(*new InheritStyleValue); }
|
|
virtual ~InheritStyleValue() override { }
|
|
|
|
String to_string() const override { return "inherit"; }
|
|
|
|
private:
|
|
InheritStyleValue()
|
|
: StyleValue(Type::Inherit)
|
|
{
|
|
}
|
|
};
|
|
|
|
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; }
|
|
String to_string() const override { return m_color.to_string(); }
|
|
Color to_color(const DOM::Document&) const override { return m_color; }
|
|
|
|
virtual bool equals(const StyleValue& 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 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 String to_string() const override;
|
|
virtual Color to_color(const DOM::Document&) const override;
|
|
|
|
virtual bool equals(const StyleValue& 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(const URL& url, DOM::Document& document) { return adopt_ref(*new ImageStyleValue(url, document)); }
|
|
virtual ~ImageStyleValue() override { }
|
|
|
|
String to_string() const override { return String::formatted("Image({})", m_url.to_string()); }
|
|
|
|
const Gfx::Bitmap* bitmap() const { return m_bitmap; }
|
|
|
|
private:
|
|
ImageStyleValue(const URL&, DOM::Document&);
|
|
|
|
// ^ResourceClient
|
|
virtual void resource_did_load() override;
|
|
|
|
URL m_url;
|
|
WeakPtr<DOM::Document> m_document;
|
|
RefPtr<Gfx::Bitmap> m_bitmap;
|
|
};
|
|
|
|
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, NonnullRefPtrVector<StyleValue>&& font_families) { return adopt_ref(*new FontStyleValue(font_style, font_weight, font_size, line_height, move(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; }
|
|
NonnullRefPtrVector<StyleValue> const& font_families() const { return m_font_families; }
|
|
|
|
virtual String to_string() const override
|
|
{
|
|
StringBuilder string_builder;
|
|
string_builder.appendff("Font style: {}, weight: {}, size: {}, line_height: {}, families: [",
|
|
m_font_style->to_string(), m_font_weight->to_string(), m_font_size->to_string(), m_line_height->to_string());
|
|
for (auto& family : m_font_families) {
|
|
string_builder.append(family.to_string());
|
|
string_builder.append(",");
|
|
}
|
|
string_builder.append("]");
|
|
|
|
return string_builder.to_string();
|
|
}
|
|
|
|
private:
|
|
FontStyleValue(NonnullRefPtr<StyleValue> font_style, NonnullRefPtr<StyleValue> font_weight, NonnullRefPtr<StyleValue> font_size, NonnullRefPtr<StyleValue> line_height, NonnullRefPtrVector<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(move(font_families))
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<StyleValue> m_font_style;
|
|
NonnullRefPtr<StyleValue> m_font_weight;
|
|
NonnullRefPtr<StyleValue> m_font_size;
|
|
NonnullRefPtr<StyleValue> m_line_height;
|
|
NonnullRefPtrVector<StyleValue> m_font_families;
|
|
// FIXME: Implement font-stretch and font-variant.
|
|
};
|
|
|
|
class StyleValueList final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<StyleValueList> create(NonnullRefPtrVector<StyleValue>&& values) { return adopt_ref(*new StyleValueList(move(values))); }
|
|
|
|
NonnullRefPtrVector<StyleValue> const& values() const { return m_values; }
|
|
|
|
virtual String to_string() const
|
|
{
|
|
StringBuilder builder;
|
|
builder.appendff("List[{}](", m_values.size());
|
|
for (size_t i = 0; i < m_values.size(); ++i) {
|
|
if (i)
|
|
builder.append(',');
|
|
builder.append(m_values[i].to_string());
|
|
}
|
|
builder.append(')');
|
|
return builder.to_string();
|
|
}
|
|
|
|
private:
|
|
StyleValueList(NonnullRefPtrVector<StyleValue>&& values)
|
|
: StyleValue(Type::ValueList)
|
|
, m_values(move(values))
|
|
{
|
|
}
|
|
|
|
NonnullRefPtrVector<StyleValue> m_values;
|
|
};
|
|
|
|
class ValueListStyleValue final : public StyleValue {
|
|
public:
|
|
static NonnullRefPtr<ValueListStyleValue> create(Vector<StyleComponentValueRule>&& values) { return adopt_ref(*new ValueListStyleValue(move(values))); }
|
|
virtual ~ValueListStyleValue() override { }
|
|
|
|
virtual String to_string() const override;
|
|
|
|
Vector<StyleComponentValueRule> const& values() const { return m_values; }
|
|
|
|
private:
|
|
ValueListStyleValue(Vector<StyleComponentValueRule>&&);
|
|
|
|
Vector<StyleComponentValueRule> m_values;
|
|
};
|
|
|
|
inline CSS::ValueID StyleValue::to_identifier() const
|
|
{
|
|
if (is_identifier())
|
|
return static_cast<const IdentifierStyleValue&>(*this).id();
|
|
if (is_auto())
|
|
return CSS::ValueID::Auto;
|
|
return CSS::ValueID::Invalid;
|
|
}
|
|
}
|