/* * Copyright (c) 2022, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include namespace Web::SVG { JS_DEFINE_ALLOCATOR(SVGCircleElement); SVGCircleElement::SVGCircleElement(DOM::Document& document, DOM::QualifiedName qualified_name) : SVGGeometryElement(document, qualified_name) { } void SVGCircleElement::initialize(JS::Realm& realm) { Base::initialize(realm); set_prototype(&Bindings::ensure_web_prototype(realm, "SVGCircleElement"_fly_string)); } void SVGCircleElement::attribute_changed(FlyString const& name, Optional const& value) { SVGGeometryElement::attribute_changed(name, value); if (name == SVG::AttributeNames::cx) { m_center_x = AttributeParser::parse_coordinate(value.value_or(String {})); m_path.clear(); } else if (name == SVG::AttributeNames::cy) { m_center_y = AttributeParser::parse_coordinate(value.value_or(String {})); m_path.clear(); } else if (name == SVG::AttributeNames::r) { m_radius = AttributeParser::parse_positive_length(value.value_or(String {})); m_path.clear(); } } Gfx::Path SVGCircleElement::get_path(CSSPixelSize viewport_size) { float cx = m_center_x.value_or(0); float cy = m_center_y.value_or(0); float r = m_radius.value_or(0); Gfx::Path path; // A zero radius disables rendering. if (r == 0) return {}; bool large_arc = false; bool sweep = true; // 1. A move-to command to the point cx+r,cy; path.move_to({ cx + r, cy }); // 2. arc to cx,cy+r; path.arc_to({ cx, cy + r }, r, large_arc, sweep); // 3. arc to cx-r,cy; path.arc_to({ cx - r, cy }, r, large_arc, sweep); // 4. arc to cx,cy-r; path.arc_to({ cx, cy - r }, r, large_arc, sweep); // 5. arc with a segment-completing close path operation. path.arc_to({ cx + r, cy }, r, large_arc, sweep); return path; } // https://www.w3.org/TR/SVG11/shapes.html#CircleElementCXAttribute JS::NonnullGCPtr SVGCircleElement::cx() const { // FIXME: Populate the unit type when it is parsed (0 here is "unknown"). // FIXME: Create a proper animated value when animations are supported. auto base_length = SVGLength::create(realm(), 0, m_center_x.value_or(0)); auto anim_length = SVGLength::create(realm(), 0, m_center_x.value_or(0)); return SVGAnimatedLength::create(realm(), move(base_length), move(anim_length)); } // https://www.w3.org/TR/SVG11/shapes.html#CircleElementCYAttribute JS::NonnullGCPtr SVGCircleElement::cy() const { // FIXME: Populate the unit type when it is parsed (0 here is "unknown"). // FIXME: Create a proper animated value when animations are supported. auto base_length = SVGLength::create(realm(), 0, m_center_y.value_or(0)); auto anim_length = SVGLength::create(realm(), 0, m_center_y.value_or(0)); return SVGAnimatedLength::create(realm(), move(base_length), move(anim_length)); } // https://www.w3.org/TR/SVG11/shapes.html#CircleElementRAttribute JS::NonnullGCPtr SVGCircleElement::r() const { // FIXME: Populate the unit type when it is parsed (0 here is "unknown"). // FIXME: Create a proper animated value when animations are supported. auto base_length = SVGLength::create(realm(), 0, m_radius.value_or(0)); auto anim_length = SVGLength::create(realm(), 0, m_radius.value_or(0)); return SVGAnimatedLength::create(realm(), move(base_length), move(anim_length)); } }