ladybird/Libraries/LibWeb/CSS/StyleValues/BasicShapeStyleValue.h

187 lines
6.1 KiB
C
Raw Normal View History

/*
* Copyright (c) 2024, MacDue <macdue@dueutil.tech>
* Copyright (c) 2025, Sam Atkins <sam@ladybird.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#pragma once
#include <AK/Variant.h>
#include <LibGfx/WindingRule.h>
2024-10-27 11:58:52 +11:00
#include <LibWeb/CSS/StyleValues/PositionStyleValue.h>
#include <LibWeb/CSS/StyleValues/StyleValue.h>
#include <LibWeb/SVG/AttributeParser.h>
namespace Web::CSS {
2024-10-27 11:58:52 +11:00
struct Inset {
Gfx::Path to_path(CSSPixelRect reference_box, Layout::Node const&) const;
void serialize(StringBuilder&, SerializationMode) const;
2024-10-27 11:58:52 +11:00
bool operator==(Inset const&) const = default;
bool is_computationally_independent() const
{
return top->is_computationally_independent()
&& right->is_computationally_independent()
&& bottom->is_computationally_independent()
&& left->is_computationally_independent()
&& border_radius->is_computationally_independent();
}
ValueComparingNonnullRefPtr<StyleValue const> top;
ValueComparingNonnullRefPtr<StyleValue const> right;
ValueComparingNonnullRefPtr<StyleValue const> bottom;
ValueComparingNonnullRefPtr<StyleValue const> left;
ValueComparingNonnullRefPtr<StyleValue const> border_radius;
2024-10-27 11:58:52 +11:00
};
struct Xywh {
void serialize(StringBuilder&, SerializationMode) const;
2024-10-27 11:58:52 +11:00
bool operator==(Xywh const&) const = default;
bool is_computationally_independent() const
{
return x->is_computationally_independent()
&& y->is_computationally_independent()
&& width->is_computationally_independent()
&& height->is_computationally_independent()
&& border_radius->is_computationally_independent();
}
ValueComparingNonnullRefPtr<StyleValue const> x;
ValueComparingNonnullRefPtr<StyleValue const> y;
ValueComparingNonnullRefPtr<StyleValue const> width;
ValueComparingNonnullRefPtr<StyleValue const> height;
ValueComparingNonnullRefPtr<StyleValue const> border_radius;
2024-10-27 11:58:52 +11:00
};
struct Rect {
void serialize(StringBuilder&, SerializationMode) const;
2024-10-27 11:58:52 +11:00
bool operator==(Rect const&) const = default;
bool is_computationally_independent() const
{
return top->is_computationally_independent()
&& right->is_computationally_independent()
&& bottom->is_computationally_independent()
&& left->is_computationally_independent()
&& border_radius->is_computationally_independent();
}
ValueComparingNonnullRefPtr<StyleValue const> top;
ValueComparingNonnullRefPtr<StyleValue const> right;
ValueComparingNonnullRefPtr<StyleValue const> bottom;
ValueComparingNonnullRefPtr<StyleValue const> left;
ValueComparingNonnullRefPtr<StyleValue const> border_radius;
2024-10-27 11:58:52 +11:00
};
struct Circle {
Gfx::Path to_path(CSSPixelRect reference_box, Layout::Node const&) const;
void serialize(StringBuilder&, SerializationMode) const;
2024-10-27 11:58:52 +11:00
bool operator==(Circle const&) const = default;
bool is_computationally_independent() const
{
return radius->is_computationally_independent()
&& (!position || position->is_computationally_independent());
}
ValueComparingNonnullRefPtr<StyleValue const> radius;
ValueComparingRefPtr<StyleValue const> position;
2024-10-27 11:58:52 +11:00
};
struct Ellipse {
Gfx::Path to_path(CSSPixelRect reference_box, Layout::Node const&) const;
void serialize(StringBuilder&, SerializationMode) const;
2024-10-27 11:58:52 +11:00
bool operator==(Ellipse const&) const = default;
bool is_computationally_independent() const
{
return radius->is_computationally_independent()
&& (!position || position->is_computationally_independent());
}
ValueComparingNonnullRefPtr<StyleValue const> radius;
ValueComparingRefPtr<StyleValue const> position;
2024-10-27 11:58:52 +11:00
};
struct Polygon {
struct Point {
bool operator==(Point const&) const = default;
ValueComparingNonnullRefPtr<StyleValue const> x;
ValueComparingNonnullRefPtr<StyleValue const> y;
};
Gfx::Path to_path(CSSPixelRect reference_box, Layout::Node const&) const;
void serialize(StringBuilder&, SerializationMode) const;
bool operator==(Polygon const&) const = default;
bool is_computationally_independent() const
{
return all_of(points, [](Point const& point) { return point.x->is_computationally_independent() && point.y->is_computationally_independent(); });
}
Gfx::WindingRule fill_rule;
Vector<Point> points;
};
// https://drafts.csswg.org/css-shapes/#funcdef-basic-shape-path
struct Path {
Gfx::Path to_path(CSSPixelRect reference_box, Layout::Node const&) const;
void serialize(StringBuilder&, SerializationMode) const;
bool operator==(Path const&) const = default;
bool is_computationally_independent() const { return true; }
Gfx::WindingRule fill_rule;
SVG::Path path_instructions;
};
// https://www.w3.org/TR/css-shapes-1/#basic-shape-functions
using BasicShape = Variant<Inset, Xywh, Rect, Circle, Ellipse, Polygon, Path>;
class BasicShapeStyleValue : public StyleValueWithDefaultOperators<BasicShapeStyleValue> {
public:
static ValueComparingNonnullRefPtr<BasicShapeStyleValue const> create(BasicShape basic_shape)
{
return adopt_ref(*new (nothrow) BasicShapeStyleValue(move(basic_shape)));
}
virtual ~BasicShapeStyleValue() override;
BasicShape const& basic_shape() const { return m_basic_shape; }
virtual void serialize(StringBuilder&, SerializationMode) const override;
virtual ValueComparingNonnullRefPtr<StyleValue const> absolutized(ComputationContext const&) const override;
bool properties_equal(BasicShapeStyleValue const& other) const { return m_basic_shape == other.m_basic_shape; }
virtual bool is_computationally_independent() const override
{
return m_basic_shape.visit([](auto const& shape) { return shape.is_computationally_independent(); });
}
Gfx::Path to_path(CSSPixelRect reference_box, Layout::Node const&) const;
private:
BasicShapeStyleValue(BasicShape basic_shape)
: StyleValueWithDefaultOperators(Type::BasicShape)
, m_basic_shape(move(basic_shape))
{
}
BasicShape m_basic_shape;
};
}