mirror of
https://github.com/LadybirdBrowser/ladybird
synced 2026-05-01 03:57:15 +02:00
Previously, the LibWeb bindings generator would output multiple per interface files like Prototype/Constructor/Namespace/GlobalMixin depending on the contents of that IDL file. This complicates the build system as it means that it does not know what files will be generated without knowledge of the contents of that IDL file. Instead, for each IDL file only generate a single Bindings/<IDLFile>.h and Bindings/<IDLFile>.cpp.
257 lines
11 KiB
C++
257 lines
11 KiB
C++
/*
|
||
* Copyright (c) 2023, MacDue <macdue@dueutil.tech>
|
||
*
|
||
* SPDX-License-Identifier: BSD-2-Clause
|
||
*/
|
||
|
||
#include <LibWeb/Bindings/Intrinsics.h>
|
||
#include <LibWeb/Bindings/SVGRadialGradientElement.h>
|
||
#include <LibWeb/Painting/PaintStyle.h>
|
||
#include <LibWeb/SVG/AttributeNames.h>
|
||
#include <LibWeb/SVG/SVGRadialGradientElement.h>
|
||
|
||
namespace Web::SVG {
|
||
|
||
GC_DEFINE_ALLOCATOR(SVGRadialGradientElement);
|
||
|
||
SVGRadialGradientElement::SVGRadialGradientElement(DOM::Document& document, DOM::QualifiedName qualified_name)
|
||
: SVGGradientElement(document, qualified_name)
|
||
{
|
||
}
|
||
|
||
void SVGRadialGradientElement::initialize(JS::Realm& realm)
|
||
{
|
||
WEB_SET_PROTOTYPE_FOR_INTERFACE(SVGRadialGradientElement);
|
||
Base::initialize(realm);
|
||
}
|
||
|
||
void SVGRadialGradientElement::attribute_changed(FlyString const& name, Optional<String> const& old_value, Optional<String> const& value, Optional<FlyString> const& namespace_)
|
||
{
|
||
Base::attribute_changed(name, old_value, value, namespace_);
|
||
|
||
// FIXME: These are <length> or <coordinate> in the spec, but all examples seem to allow percentages
|
||
// and unitless values.
|
||
if (name == SVG::AttributeNames::cx) {
|
||
m_cx = AttributeParser::parse_number_percentage(value.value_or(String {}));
|
||
m_paint_style = nullptr;
|
||
} else if (name == SVG::AttributeNames::cy) {
|
||
m_cy = AttributeParser::parse_number_percentage(value.value_or(String {}));
|
||
m_paint_style = nullptr;
|
||
} else if (name == SVG::AttributeNames::fx) {
|
||
m_fx = AttributeParser::parse_number_percentage(value.value_or(String {}));
|
||
m_paint_style = nullptr;
|
||
} else if (name == SVG::AttributeNames::fy) {
|
||
m_fy = AttributeParser::parse_number_percentage(value.value_or(String {}));
|
||
m_paint_style = nullptr;
|
||
} else if (name == SVG::AttributeNames::fr) {
|
||
m_fr = AttributeParser::parse_number_percentage(value.value_or(String {}));
|
||
m_paint_style = nullptr;
|
||
} else if (name == SVG::AttributeNames::r) {
|
||
m_r = AttributeParser::parse_number_percentage(value.value_or(String {}));
|
||
m_paint_style = nullptr;
|
||
}
|
||
}
|
||
|
||
// https://svgwg.org/svg2-draft/pservers.html#RadialGradientElementFXAttribute
|
||
NumberPercentage SVGRadialGradientElement::start_circle_x() const
|
||
{
|
||
HashTable<SVGGradientElement const*> seen_gradients;
|
||
return start_circle_x_impl(seen_gradients);
|
||
}
|
||
|
||
NumberPercentage SVGRadialGradientElement::start_circle_x_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
|
||
{
|
||
if (m_fx.has_value())
|
||
return *m_fx;
|
||
// If the element references an element that specifies a value for 'fx', then the value of 'fx' is
|
||
// inherited from the referenced element.
|
||
if (auto gradient = linked_radial_gradient(seen_gradients))
|
||
return gradient->start_circle_x_impl(seen_gradients);
|
||
// If attribute ‘fx’ is not specified, ‘fx’ will coincide with the presentational value of ‘cx’ for
|
||
// the element whether the value for 'cx' was inherited or not.
|
||
return end_circle_x();
|
||
}
|
||
|
||
// https://svgwg.org/svg2-draft/pservers.html#RadialGradientElementFYAttribute
|
||
NumberPercentage SVGRadialGradientElement::start_circle_y() const
|
||
{
|
||
HashTable<SVGGradientElement const*> seen_gradients;
|
||
return start_circle_y_impl(seen_gradients);
|
||
}
|
||
|
||
NumberPercentage SVGRadialGradientElement::start_circle_y_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
|
||
{
|
||
if (m_fy.has_value())
|
||
return *m_fy;
|
||
// If the element references an element that specifies a value for 'fy', then the value of 'fy' is
|
||
// inherited from the referenced element.
|
||
if (auto gradient = linked_radial_gradient(seen_gradients))
|
||
return gradient->start_circle_y_impl(seen_gradients);
|
||
// If attribute ‘fy’ is not specified, ‘fy’ will coincide with the presentational value of ‘cy’ for
|
||
// the element whether the value for 'cy' was inherited or not.
|
||
return end_circle_y();
|
||
}
|
||
|
||
// https://svgwg.org/svg2-draft/pservers.html#RadialGradientElementFRAttribute
|
||
NumberPercentage SVGRadialGradientElement::start_circle_radius() const
|
||
{
|
||
HashTable<SVGGradientElement const*> seen_gradients;
|
||
return start_circle_radius_impl(seen_gradients);
|
||
}
|
||
|
||
NumberPercentage SVGRadialGradientElement::start_circle_radius_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
|
||
{
|
||
// Note: A negative value is an error.
|
||
if (m_fr.has_value() && m_fr->value() >= 0)
|
||
return *m_fr;
|
||
// if the element references an element that specifies a value for 'fr', then the value of
|
||
// 'fr' is inherited from the referenced element.
|
||
if (auto gradient = linked_radial_gradient(seen_gradients))
|
||
return gradient->start_circle_radius_impl(seen_gradients);
|
||
// If the attribute is not specified, the effect is as if a value of '0%' were specified.
|
||
return NumberPercentage::create_percentage(0);
|
||
}
|
||
|
||
// https://svgwg.org/svg2-draft/pservers.html#RadialGradientElementCXAttribute
|
||
NumberPercentage SVGRadialGradientElement::end_circle_x() const
|
||
{
|
||
HashTable<SVGGradientElement const*> seen_gradients;
|
||
return end_circle_x_impl(seen_gradients);
|
||
}
|
||
|
||
NumberPercentage SVGRadialGradientElement::end_circle_x_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
|
||
{
|
||
if (m_cx.has_value())
|
||
return *m_cx;
|
||
if (auto gradient = linked_radial_gradient(seen_gradients))
|
||
return gradient->end_circle_x_impl(seen_gradients);
|
||
return NumberPercentage::create_percentage(50);
|
||
}
|
||
|
||
// https://svgwg.org/svg2-draft/pservers.html#RadialGradientElementCYAttribute
|
||
NumberPercentage SVGRadialGradientElement::end_circle_y() const
|
||
{
|
||
HashTable<SVGGradientElement const*> seen_gradients;
|
||
return end_circle_y_impl(seen_gradients);
|
||
}
|
||
|
||
NumberPercentage SVGRadialGradientElement::end_circle_y_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
|
||
{
|
||
if (m_cy.has_value())
|
||
return *m_cy;
|
||
if (auto gradient = linked_radial_gradient(seen_gradients))
|
||
return gradient->end_circle_y_impl(seen_gradients);
|
||
return NumberPercentage::create_percentage(50);
|
||
}
|
||
|
||
// https://svgwg.org/svg2-draft/pservers.html#RadialGradientElementRAttribute
|
||
NumberPercentage SVGRadialGradientElement::end_circle_radius() const
|
||
{
|
||
HashTable<SVGGradientElement const*> seen_gradients;
|
||
return end_circle_radius_impl(seen_gradients);
|
||
}
|
||
|
||
NumberPercentage SVGRadialGradientElement::end_circle_radius_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
|
||
{
|
||
// Note: A negative value is an error.
|
||
if (m_r.has_value() && m_r->value() >= 0)
|
||
return *m_r;
|
||
if (auto gradient = linked_radial_gradient(seen_gradients))
|
||
return gradient->end_circle_radius_impl(seen_gradients);
|
||
return NumberPercentage::create_percentage(50);
|
||
}
|
||
|
||
Optional<Painting::PaintStyle> SVGRadialGradientElement::to_gfx_paint_style(SVGPaintContext const& paint_context) const
|
||
{
|
||
auto units = gradient_units();
|
||
Gfx::FloatPoint start_center;
|
||
float start_radius = 0.0f;
|
||
Gfx::FloatPoint end_center;
|
||
float end_radius = 0.0f;
|
||
|
||
// FIXME: Where in the spec does it say what axis the radius is relative to?
|
||
if (units == GradientUnits::ObjectBoundingBox) {
|
||
// If gradientUnits="objectBoundingBox", the user coordinate system for attributes ‘cx’, ‘cy’, ‘r’, ‘fx’, ‘fy’, and ‘fr’
|
||
// is established using the bounding box of the element to which the gradient is applied (see Object bounding box units)
|
||
// and then applying the transform specified by attribute ‘gradientTransform’. Percentages represent values relative
|
||
// to the bounding box for the object.
|
||
auto const& bounding_box = paint_context.path_bounding_box;
|
||
start_center = {
|
||
bounding_box.location().x() + start_circle_x().value() * bounding_box.width(),
|
||
bounding_box.location().y() + start_circle_y().value() * bounding_box.height(),
|
||
};
|
||
start_radius = start_circle_radius().value() * bounding_box.width();
|
||
end_center = {
|
||
bounding_box.location().x() + end_circle_x().value() * bounding_box.width(),
|
||
bounding_box.location().y() + end_circle_y().value() * bounding_box.height(),
|
||
};
|
||
end_radius = end_circle_radius().value() * bounding_box.width();
|
||
} else {
|
||
// GradientUnits::UserSpaceOnUse
|
||
// If gradientUnits="userSpaceOnUse", ‘cx’, ‘cy’, ‘r’, ‘fx’, ‘fy’, and ‘fr’ represent values in the coordinate system
|
||
// that results from taking the current user coordinate system in place at the time when the gradient element is
|
||
// referenced (i.e., the user coordinate system for the element referencing the gradient element via a fill or stroke property)
|
||
// and then applying the transform specified by attribute ‘gradientTransform’.
|
||
// Percentages represent values relative to the current SVG viewport.
|
||
// Note: The start/end centers will be in relative units here.
|
||
// They will be resolved at paint time using the gradient paint transform.
|
||
start_center = Gfx::FloatPoint {
|
||
start_circle_x().resolve_relative_to(paint_context.viewport.width()),
|
||
start_circle_y().resolve_relative_to(paint_context.viewport.height()),
|
||
};
|
||
start_radius = start_circle_radius().resolve_relative_to(paint_context.viewport.width());
|
||
end_center = Gfx::FloatPoint {
|
||
end_circle_x().resolve_relative_to(paint_context.viewport.width()),
|
||
end_circle_y().resolve_relative_to(paint_context.viewport.height()),
|
||
};
|
||
end_radius = end_circle_radius().resolve_relative_to(paint_context.viewport.width());
|
||
}
|
||
|
||
if (!m_paint_style) {
|
||
m_paint_style = Painting::SVGRadialGradientPaintStyle::create(start_center, start_radius, end_center, end_radius);
|
||
// FIXME: Update stops in DOM changes:
|
||
add_color_stops(*m_paint_style);
|
||
} else {
|
||
m_paint_style->set_start_center(start_center);
|
||
m_paint_style->set_start_radius(start_radius);
|
||
m_paint_style->set_end_center(end_center);
|
||
m_paint_style->set_end_radius(end_radius);
|
||
}
|
||
m_paint_style->set_gradient_transform(gradient_paint_transform(paint_context));
|
||
m_paint_style->set_spread_method(to_painting_spread_method(spread_method()));
|
||
m_paint_style->set_color_space(color_space());
|
||
return *m_paint_style;
|
||
}
|
||
|
||
GC::Ref<SVGAnimatedLength> SVGRadialGradientElement::cx() const
|
||
{
|
||
return fake_animated_length_fixme();
|
||
}
|
||
|
||
GC::Ref<SVGAnimatedLength> SVGRadialGradientElement::cy() const
|
||
{
|
||
return fake_animated_length_fixme();
|
||
}
|
||
|
||
GC::Ref<SVGAnimatedLength> SVGRadialGradientElement::fx() const
|
||
{
|
||
return fake_animated_length_fixme();
|
||
}
|
||
|
||
GC::Ref<SVGAnimatedLength> SVGRadialGradientElement::fy() const
|
||
{
|
||
return fake_animated_length_fixme();
|
||
}
|
||
|
||
GC::Ref<SVGAnimatedLength> SVGRadialGradientElement::fr() const
|
||
{
|
||
return fake_animated_length_fixme();
|
||
}
|
||
|
||
GC::Ref<SVGAnimatedLength> SVGRadialGradientElement::r() const
|
||
{
|
||
return fake_animated_length_fixme();
|
||
}
|
||
|
||
}
|