summaryrefslogtreecommitdiff
path: root/Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp
diff options
context:
space:
mode:
authorSam Atkins <atkinssj@serenityos.org>2022-02-11 16:14:58 +0000
committerAndreas Kling <kling@serenityos.org>2022-02-11 21:38:27 +0100
commit21bdcee3c3ee604742ea6524a320e0cf5ad65abb (patch)
treec319c012349fd18c49e1ffb932e38b4a2faa11ee /Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp
parent1dde6a0a2bfef387c464a35de6d1b7332c4437b5 (diff)
downloadserenity-21bdcee3c3ee604742ea6524a320e0cf5ad65abb.zip
LibWeb: Add SVG `<circle>` element and test case :^)
Diffstat (limited to 'Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp')
-rw-r--r--Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp73
1 files changed, 73 insertions, 0 deletions
diff --git a/Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp b/Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp
new file mode 100644
index 0000000000..e524859579
--- /dev/null
+++ b/Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp
@@ -0,0 +1,73 @@
+/*
+ * Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org>
+ *
+ * SPDX-License-Identifier: BSD-2-Clause
+ */
+
+#include "SVGCircleElement.h"
+#include <LibWeb/SVG/AttributeNames.h>
+#include <LibWeb/SVG/AttributeParser.h>
+
+namespace Web::SVG {
+
+SVGCircleElement::SVGCircleElement(DOM::Document& document, QualifiedName qualified_name)
+ : SVGGeometryElement(document, qualified_name)
+{
+}
+
+void SVGCircleElement::parse_attribute(FlyString const& name, String const& value)
+{
+ SVGGeometryElement::parse_attribute(name, value);
+
+ if (name == SVG::AttributeNames::cx) {
+ m_center_x = AttributeParser::parse_coordinate(value);
+ m_path.clear();
+ } else if (name == SVG::AttributeNames::cy) {
+ m_center_y = AttributeParser::parse_coordinate(value);
+ m_path.clear();
+ } else if (name == SVG::AttributeNames::r) {
+ m_radius = AttributeParser::parse_positive_length(value);
+ m_path.clear();
+ }
+}
+
+Gfx::Path& SVGCircleElement::get_path()
+{
+ if (m_path.has_value())
+ return m_path.value();
+
+ 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) {
+ m_path = move(path);
+ return m_path.value();
+ }
+
+ 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);
+
+ m_path = move(path);
+ return m_path.value();
+}
+
+}