mirror of
https://github.com/RGBCube/serenity
synced 2025-07-26 08:47:34 +00:00
LibWeb: Add ConicGradientStyleValue
This commit adds a simple style value (which is an abstract image) to represent conic-gradient()s. This commit also starts to factor out some reusable parts of the linear-gradient() style value for other gradient types.
This commit is contained in:
parent
5eeb6bbee3
commit
067759c0e9
3 changed files with 139 additions and 48 deletions
|
@ -90,6 +90,12 @@ ColorStyleValue const& StyleValue::as_color() const
|
||||||
return static_cast<ColorStyleValue const&>(*this);
|
return static_cast<ColorStyleValue const&>(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ConicGradientStyleValue const& StyleValue::as_conic_gradient() const
|
||||||
|
{
|
||||||
|
VERIFY(is_conic_gradient());
|
||||||
|
return static_cast<ConicGradientStyleValue const&>(*this);
|
||||||
|
}
|
||||||
|
|
||||||
ContentStyleValue const& StyleValue::as_content() const
|
ContentStyleValue const& StyleValue::as_content() const
|
||||||
{
|
{
|
||||||
VERIFY(is_content());
|
VERIFY(is_content());
|
||||||
|
@ -1697,6 +1703,26 @@ void ImageStyleValue::paint(PaintContext& context, Gfx::IntRect const& dest_rect
|
||||||
context.painter().draw_scaled_bitmap(dest_rect, *m_bitmap, m_bitmap->rect(), 1.0f, to_gfx_scaling_mode(image_rendering));
|
context.painter().draw_scaled_bitmap(dest_rect, *m_bitmap, m_bitmap->rect(), 1.0f, to_gfx_scaling_mode(image_rendering));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void serialize_color_stop_list(StringBuilder& builder, auto const& color_stop_list)
|
||||||
|
{
|
||||||
|
bool first = true;
|
||||||
|
for (auto const& element : color_stop_list) {
|
||||||
|
if (!first)
|
||||||
|
builder.append(", "sv);
|
||||||
|
|
||||||
|
if (element.transition_hint.has_value()) {
|
||||||
|
builder.appendff("{}, "sv, element.transition_hint->value.to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
serialize_a_srgb_value(builder, element.color_stop.color);
|
||||||
|
for (auto position : Array { &element.color_stop.position, &element.color_stop.second_position }) {
|
||||||
|
if (position->has_value())
|
||||||
|
builder.appendff(" {}"sv, (*position)->to_string());
|
||||||
|
}
|
||||||
|
first = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
String LinearGradientStyleValue::to_string() const
|
String LinearGradientStyleValue::to_string() const
|
||||||
{
|
{
|
||||||
StringBuilder builder;
|
StringBuilder builder;
|
||||||
|
@ -1736,22 +1762,7 @@ String LinearGradientStyleValue::to_string() const
|
||||||
builder.appendff("{}, "sv, angle.to_string());
|
builder.appendff("{}, "sv, angle.to_string());
|
||||||
});
|
});
|
||||||
|
|
||||||
bool first = true;
|
serialize_color_stop_list(builder, m_color_stop_list);
|
||||||
for (auto const& element : m_color_stop_list) {
|
|
||||||
if (!first)
|
|
||||||
builder.append(", "sv);
|
|
||||||
|
|
||||||
if (element.transition_hint.has_value()) {
|
|
||||||
builder.appendff("{}, "sv, element.transition_hint->value.to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
serialize_a_srgb_value(builder, element.color_stop.color);
|
|
||||||
for (auto position : Array { &element.color_stop.position, &element.color_stop.second_position }) {
|
|
||||||
if (position->has_value())
|
|
||||||
builder.appendff(" {}"sv, (*position)->to_string());
|
|
||||||
}
|
|
||||||
first = false;
|
|
||||||
}
|
|
||||||
builder.append(")"sv);
|
builder.append(")"sv);
|
||||||
return builder.to_string();
|
return builder.to_string();
|
||||||
}
|
}
|
||||||
|
@ -1765,21 +1776,6 @@ static bool operator==(LinearGradientStyleValue::GradientDirection const& a, Lin
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool operator==(GradientColorHint const& a, GradientColorHint const& b)
|
|
||||||
{
|
|
||||||
return a.value == b.value;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool operator==(GradientColorStop const& a, GradientColorStop const& b)
|
|
||||||
{
|
|
||||||
return a.color == b.color && a.position == b.position && a.second_position == b.second_position;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool operator==(ColorStopListElement const& a, ColorStopListElement const& b)
|
|
||||||
{
|
|
||||||
return a.transition_hint == b.transition_hint && a.color_stop == b.color_stop;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool operator==(EdgeRect const& a, EdgeRect const& b)
|
static bool operator==(EdgeRect const& a, EdgeRect const& b)
|
||||||
{
|
{
|
||||||
return a.top_edge == b.top_edge && a.right_edge == b.right_edge && a.bottom_edge == b.bottom_edge && a.left_edge == b.left_edge;
|
return a.top_edge == b.top_edge && a.right_edge == b.right_edge && a.bottom_edge == b.bottom_edge && a.left_edge == b.left_edge;
|
||||||
|
@ -1857,6 +1853,45 @@ void LinearGradientStyleValue::paint(PaintContext& context, Gfx::IntRect const&
|
||||||
Painting::paint_linear_gradient(context, dest_rect, m_resolved->data);
|
Painting::paint_linear_gradient(context, dest_rect, m_resolved->data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String ConicGradientStyleValue::to_string() const
|
||||||
|
{
|
||||||
|
StringBuilder builder;
|
||||||
|
builder.append("conic-gradient("sv);
|
||||||
|
bool has_from_angle = false;
|
||||||
|
bool has_at_position = false;
|
||||||
|
if ((has_from_angle = m_from_angle.to_degrees() != 0))
|
||||||
|
builder.appendff("from {}", m_from_angle.to_string());
|
||||||
|
if ((has_at_position = m_position != PositionValue::center())) {
|
||||||
|
if (has_from_angle)
|
||||||
|
builder.append(' ');
|
||||||
|
builder.appendff("at "sv);
|
||||||
|
m_position.serialize(builder);
|
||||||
|
}
|
||||||
|
if (has_from_angle || has_at_position)
|
||||||
|
builder.append(", "sv);
|
||||||
|
serialize_color_stop_list(builder, m_color_stop_list);
|
||||||
|
builder.append(')');
|
||||||
|
return builder.to_string();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConicGradientStyleValue::resolve_for_size(Layout::Node const&, Gfx::FloatSize const&) const
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConicGradientStyleValue::paint(PaintContext&, Gfx::IntRect const&, CSS::ImageRendering) const
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ConicGradientStyleValue::equals(StyleValue const&) const
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
float ConicGradientStyleValue::angle_degrees() const
|
||||||
|
{
|
||||||
|
return m_from_angle.to_degrees();
|
||||||
|
}
|
||||||
|
|
||||||
bool InheritStyleValue::equals(StyleValue const& other) const
|
bool InheritStyleValue::equals(StyleValue const& other) const
|
||||||
{
|
{
|
||||||
return type() == other.type();
|
return type() == other.type();
|
||||||
|
|
|
@ -74,21 +74,28 @@ enum class SideOrCorner {
|
||||||
BottomRight
|
BottomRight
|
||||||
};
|
};
|
||||||
|
|
||||||
struct GradientColorStop {
|
template<typename TPosition>
|
||||||
Color color;
|
|
||||||
Optional<LengthPercentage> position;
|
|
||||||
Optional<LengthPercentage> second_position = {};
|
|
||||||
};
|
|
||||||
|
|
||||||
struct GradientColorHint {
|
|
||||||
LengthPercentage value;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct ColorStopListElement {
|
struct ColorStopListElement {
|
||||||
Optional<GradientColorHint> transition_hint;
|
using PositionType = TPosition;
|
||||||
GradientColorStop color_stop;
|
struct ColorHint {
|
||||||
|
TPosition value;
|
||||||
|
inline bool operator==(ColorHint const&) const = default;
|
||||||
|
};
|
||||||
|
|
||||||
|
Optional<ColorHint> transition_hint;
|
||||||
|
struct ColorStop {
|
||||||
|
Color color;
|
||||||
|
Optional<TPosition> position;
|
||||||
|
Optional<TPosition> second_position = {};
|
||||||
|
inline bool operator==(ColorStop const&) const = default;
|
||||||
|
} color_stop;
|
||||||
|
|
||||||
|
inline bool operator==(ColorStopListElement const&) const = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
using LinearColorStopListElement = ColorStopListElement<LengthPercentage>;
|
||||||
|
using AngularColorStopListElement = ColorStopListElement<AnglePercentage>;
|
||||||
|
|
||||||
struct EdgeRect {
|
struct EdgeRect {
|
||||||
Length top_edge;
|
Length top_edge;
|
||||||
Length right_edge;
|
Length right_edge;
|
||||||
|
@ -173,6 +180,7 @@ public:
|
||||||
BorderRadiusShorthand,
|
BorderRadiusShorthand,
|
||||||
Calculated,
|
Calculated,
|
||||||
Color,
|
Color,
|
||||||
|
ConicGradient,
|
||||||
Content,
|
Content,
|
||||||
FilterValueList,
|
FilterValueList,
|
||||||
Flex,
|
Flex,
|
||||||
|
@ -208,7 +216,7 @@ public:
|
||||||
|
|
||||||
Type type() const { return m_type; }
|
Type type() const { return m_type; }
|
||||||
|
|
||||||
bool is_abstract_image() const { return AK::first_is_one_of(type(), Type::Image, Type::LinearGradient); }
|
bool is_abstract_image() const { return AK::first_is_one_of(type(), Type::Image, Type::LinearGradient, Type::ConicGradient); }
|
||||||
bool is_angle() const { return type() == Type::Angle; }
|
bool is_angle() const { return type() == Type::Angle; }
|
||||||
bool is_background() const { return type() == Type::Background; }
|
bool is_background() const { return type() == Type::Background; }
|
||||||
bool is_background_repeat() const { return type() == Type::BackgroundRepeat; }
|
bool is_background_repeat() const { return type() == Type::BackgroundRepeat; }
|
||||||
|
@ -218,6 +226,7 @@ public:
|
||||||
bool is_border_radius_shorthand() const { return type() == Type::BorderRadiusShorthand; }
|
bool is_border_radius_shorthand() const { return type() == Type::BorderRadiusShorthand; }
|
||||||
bool is_calculated() const { return type() == Type::Calculated; }
|
bool is_calculated() const { return type() == Type::Calculated; }
|
||||||
bool is_color() const { return type() == Type::Color; }
|
bool is_color() const { return type() == Type::Color; }
|
||||||
|
bool is_conic_gradient() const { return type() == Type::ConicGradient; }
|
||||||
bool is_content() const { return type() == Type::Content; }
|
bool is_content() const { return type() == Type::Content; }
|
||||||
bool is_filter_value_list() const { return type() == Type::FilterValueList; }
|
bool is_filter_value_list() const { return type() == Type::FilterValueList; }
|
||||||
bool is_flex() const { return type() == Type::Flex; }
|
bool is_flex() const { return type() == Type::Flex; }
|
||||||
|
@ -261,6 +270,7 @@ public:
|
||||||
BorderStyleValue const& as_border() const;
|
BorderStyleValue const& as_border() const;
|
||||||
CalculatedStyleValue const& as_calculated() const;
|
CalculatedStyleValue const& as_calculated() const;
|
||||||
ColorStyleValue const& as_color() const;
|
ColorStyleValue const& as_color() const;
|
||||||
|
ConicGradientStyleValue const& as_conic_gradient() const;
|
||||||
ContentStyleValue const& as_content() const;
|
ContentStyleValue const& as_content() const;
|
||||||
FilterValueListStyleValue const& as_filter_value_list() const;
|
FilterValueListStyleValue const& as_filter_value_list() const;
|
||||||
FlexFlowStyleValue const& as_flex_flow() const;
|
FlexFlowStyleValue const& as_flex_flow() const;
|
||||||
|
@ -302,6 +312,7 @@ public:
|
||||||
BorderStyleValue& as_border() { return const_cast<BorderStyleValue&>(const_cast<StyleValue const&>(*this).as_border()); }
|
BorderStyleValue& as_border() { return const_cast<BorderStyleValue&>(const_cast<StyleValue const&>(*this).as_border()); }
|
||||||
CalculatedStyleValue& as_calculated() { return const_cast<CalculatedStyleValue&>(const_cast<StyleValue const&>(*this).as_calculated()); }
|
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()); }
|
ColorStyleValue& as_color() { return const_cast<ColorStyleValue&>(const_cast<StyleValue const&>(*this).as_color()); }
|
||||||
|
ConicGradientStyleValue& as_conic_gradient() { return const_cast<ConicGradientStyleValue&>(const_cast<StyleValue const&>(*this).as_conic_gradient()); }
|
||||||
ContentStyleValue& as_content() { return const_cast<ContentStyleValue&>(const_cast<StyleValue const&>(*this).as_content()); }
|
ContentStyleValue& as_content() { return const_cast<ContentStyleValue&>(const_cast<StyleValue const&>(*this).as_content()); }
|
||||||
FilterValueListStyleValue& as_filter_value_list() { return const_cast<FilterValueListStyleValue&>(const_cast<StyleValue const&>(*this).as_filter_value_list()); }
|
FilterValueListStyleValue& as_filter_value_list() { return const_cast<FilterValueListStyleValue&>(const_cast<StyleValue const&>(*this).as_filter_value_list()); }
|
||||||
FlexFlowStyleValue& as_flex_flow() { return const_cast<FlexFlowStyleValue&>(const_cast<StyleValue const&>(*this).as_flex_flow()); }
|
FlexFlowStyleValue& as_flex_flow() { return const_cast<FlexFlowStyleValue&>(const_cast<StyleValue const&>(*this).as_flex_flow()); }
|
||||||
|
@ -1141,6 +1152,50 @@ private:
|
||||||
RefPtr<Gfx::Bitmap> m_bitmap;
|
RefPtr<Gfx::Bitmap> m_bitmap;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class ConicGradientStyleValue final : public AbstractImageStyleValue {
|
||||||
|
public:
|
||||||
|
static NonnullRefPtr<ConicGradientStyleValue> create(Angle from_angle, PositionValue position, Vector<AngularColorStopListElement> color_stop_list)
|
||||||
|
{
|
||||||
|
VERIFY(color_stop_list.size() >= 2);
|
||||||
|
return adopt_ref(*new ConicGradientStyleValue(from_angle, position, move(color_stop_list)));
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual String to_string() const override;
|
||||||
|
|
||||||
|
void paint(PaintContext&, Gfx::IntRect const& dest_rect, CSS::ImageRendering) const override;
|
||||||
|
|
||||||
|
virtual bool equals(StyleValue const& other) const override;
|
||||||
|
|
||||||
|
Vector<AngularColorStopListElement> const& color_stop_list() const
|
||||||
|
{
|
||||||
|
return m_color_stop_list;
|
||||||
|
}
|
||||||
|
|
||||||
|
float angle_degrees() const;
|
||||||
|
|
||||||
|
bool is_paintable() const override { return true; }
|
||||||
|
|
||||||
|
void resolve_for_size(Layout::Node const&, Gfx::FloatSize const&) const override;
|
||||||
|
|
||||||
|
virtual ~ConicGradientStyleValue() override = default;
|
||||||
|
|
||||||
|
Gfx::FloatPoint resolve_position(Layout::Node const&, Gfx::FloatRect const&) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
ConicGradientStyleValue(Angle from_angle, PositionValue position, Vector<AngularColorStopListElement> color_stop_list)
|
||||||
|
: AbstractImageStyleValue(Type::ConicGradient)
|
||||||
|
, m_from_angle(from_angle)
|
||||||
|
, m_position(position)
|
||||||
|
, m_color_stop_list(move(color_stop_list))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: Support <color-interpolation-method>
|
||||||
|
Angle m_from_angle;
|
||||||
|
PositionValue m_position;
|
||||||
|
Vector<AngularColorStopListElement> m_color_stop_list;
|
||||||
|
};
|
||||||
|
|
||||||
class LinearGradientStyleValue final : public AbstractImageStyleValue {
|
class LinearGradientStyleValue final : public AbstractImageStyleValue {
|
||||||
public:
|
public:
|
||||||
using GradientDirection = Variant<Angle, SideOrCorner>;
|
using GradientDirection = Variant<Angle, SideOrCorner>;
|
||||||
|
@ -1155,7 +1210,7 @@ public:
|
||||||
No
|
No
|
||||||
};
|
};
|
||||||
|
|
||||||
static NonnullRefPtr<LinearGradientStyleValue> create(GradientDirection direction, Vector<ColorStopListElement> color_stop_list, GradientType type, Repeating repeating)
|
static NonnullRefPtr<LinearGradientStyleValue> create(GradientDirection direction, Vector<LinearColorStopListElement> color_stop_list, GradientType type, Repeating repeating)
|
||||||
{
|
{
|
||||||
VERIFY(color_stop_list.size() >= 2);
|
VERIFY(color_stop_list.size() >= 2);
|
||||||
return adopt_ref(*new LinearGradientStyleValue(direction, move(color_stop_list), type, repeating));
|
return adopt_ref(*new LinearGradientStyleValue(direction, move(color_stop_list), type, repeating));
|
||||||
|
@ -1165,7 +1220,7 @@ public:
|
||||||
virtual ~LinearGradientStyleValue() override = default;
|
virtual ~LinearGradientStyleValue() override = default;
|
||||||
virtual bool equals(StyleValue const& other) const override;
|
virtual bool equals(StyleValue const& other) const override;
|
||||||
|
|
||||||
Vector<ColorStopListElement> const& color_stop_list() const
|
Vector<LinearColorStopListElement> const& color_stop_list() const
|
||||||
{
|
{
|
||||||
return m_color_stop_list;
|
return m_color_stop_list;
|
||||||
}
|
}
|
||||||
|
@ -1180,7 +1235,7 @@ public:
|
||||||
void paint(PaintContext& context, Gfx::IntRect const& dest_rect, CSS::ImageRendering image_rendering) const override;
|
void paint(PaintContext& context, Gfx::IntRect const& dest_rect, CSS::ImageRendering image_rendering) const override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
LinearGradientStyleValue(GradientDirection direction, Vector<ColorStopListElement> color_stop_list, GradientType type, Repeating repeating)
|
LinearGradientStyleValue(GradientDirection direction, Vector<LinearColorStopListElement> color_stop_list, GradientType type, Repeating repeating)
|
||||||
: AbstractImageStyleValue(Type::LinearGradient)
|
: AbstractImageStyleValue(Type::LinearGradient)
|
||||||
, m_direction(direction)
|
, m_direction(direction)
|
||||||
, m_color_stop_list(move(color_stop_list))
|
, m_color_stop_list(move(color_stop_list))
|
||||||
|
@ -1190,7 +1245,7 @@ private:
|
||||||
}
|
}
|
||||||
|
|
||||||
GradientDirection m_direction;
|
GradientDirection m_direction;
|
||||||
Vector<ColorStopListElement> m_color_stop_list;
|
Vector<LinearColorStopListElement> m_color_stop_list;
|
||||||
GradientType m_gradient_type;
|
GradientType m_gradient_type;
|
||||||
Repeating m_repeating;
|
Repeating m_repeating;
|
||||||
|
|
||||||
|
|
|
@ -36,6 +36,7 @@ class BorderStyleValue;
|
||||||
class Clip;
|
class Clip;
|
||||||
class CalculatedStyleValue;
|
class CalculatedStyleValue;
|
||||||
class ColorStyleValue;
|
class ColorStyleValue;
|
||||||
|
class ConicGradientStyleValue;
|
||||||
class ContentStyleValue;
|
class ContentStyleValue;
|
||||||
class CSSConditionRule;
|
class CSSConditionRule;
|
||||||
class CSSGroupingRule;
|
class CSSGroupingRule;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue