diff options
author | Sam Atkins <atkinssj@serenityos.org> | 2022-02-11 16:14:58 +0000 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2022-02-11 21:38:27 +0100 |
commit | 21bdcee3c3ee604742ea6524a320e0cf5ad65abb (patch) | |
tree | c319c012349fd18c49e1ffb932e38b4a2faa11ee /Userland/Libraries/LibWeb | |
parent | 1dde6a0a2bfef387c464a35de6d1b7332c4437b5 (diff) | |
download | serenity-21bdcee3c3ee604742ea6524a320e0cf5ad65abb.zip |
LibWeb: Add SVG `<circle>` element and test case :^)
Diffstat (limited to 'Userland/Libraries/LibWeb')
-rw-r--r-- | Userland/Libraries/LibWeb/Bindings/NodeWrapperFactory.cpp | 4 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/Bindings/WindowObjectHelper.h | 3 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/CMakeLists.txt | 2 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/DOM/ElementFactory.cpp | 3 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/Forward.h | 2 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/SVG/AttributeNames.h | 3 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/SVG/SVGCircleElement.cpp | 73 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/SVG/SVGCircleElement.h | 32 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/SVG/SVGCircleElement.idl | 6 | ||||
-rw-r--r-- | Userland/Libraries/LibWeb/SVG/TagNames.h | 1 |
10 files changed, 129 insertions, 0 deletions
diff --git a/Userland/Libraries/LibWeb/Bindings/NodeWrapperFactory.cpp b/Userland/Libraries/LibWeb/Bindings/NodeWrapperFactory.cpp index 2af9ccd4e3..34262a384a 100644 --- a/Userland/Libraries/LibWeb/Bindings/NodeWrapperFactory.cpp +++ b/Userland/Libraries/LibWeb/Bindings/NodeWrapperFactory.cpp @@ -82,6 +82,7 @@ #include <LibWeb/Bindings/HTMLVideoElementWrapper.h> #include <LibWeb/Bindings/NodeWrapper.h> #include <LibWeb/Bindings/NodeWrapperFactory.h> +#include <LibWeb/Bindings/SVGCircleElementWrapper.h> #include <LibWeb/Bindings/SVGPathElementWrapper.h> #include <LibWeb/Bindings/SVGRectElementWrapper.h> #include <LibWeb/Bindings/SVGSVGElementWrapper.h> @@ -157,6 +158,7 @@ #include <LibWeb/HTML/HTMLUListElement.h> #include <LibWeb/HTML/HTMLUnknownElement.h> #include <LibWeb/HTML/HTMLVideoElement.h> +#include <LibWeb/SVG/SVGCircleElement.h> #include <LibWeb/SVG/SVGPathElement.h> #include <LibWeb/SVG/SVGRectElement.h> #include <LibWeb/SVG/SVGSVGElement.h> @@ -311,6 +313,8 @@ NodeWrapper* wrap(JS::GlobalObject& global_object, DOM::Node& node) return static_cast<NodeWrapper*>(wrap_impl(global_object, verify_cast<HTML::HTMLElement>(node))); if (is<SVG::SVGSVGElement>(node)) return static_cast<NodeWrapper*>(wrap_impl(global_object, verify_cast<SVG::SVGSVGElement>(node))); + if (is<SVG::SVGCircleElement>(node)) + return static_cast<NodeWrapper*>(wrap_impl(global_object, verify_cast<SVG::SVGCircleElement>(node))); if (is<SVG::SVGPathElement>(node)) return static_cast<NodeWrapper*>(wrap_impl(global_object, verify_cast<SVG::SVGPathElement>(node))); if (is<SVG::SVGRectElement>(node)) diff --git a/Userland/Libraries/LibWeb/Bindings/WindowObjectHelper.h b/Userland/Libraries/LibWeb/Bindings/WindowObjectHelper.h index 945502d64e..6ebd37f30c 100644 --- a/Userland/Libraries/LibWeb/Bindings/WindowObjectHelper.h +++ b/Userland/Libraries/LibWeb/Bindings/WindowObjectHelper.h @@ -247,6 +247,8 @@ #include <LibWeb/Bindings/RangePrototype.h> #include <LibWeb/Bindings/ResizeObserverConstructor.h> #include <LibWeb/Bindings/ResizeObserverPrototype.h> +#include <LibWeb/Bindings/SVGCircleElementConstructor.h> +#include <LibWeb/Bindings/SVGCircleElementPrototype.h> #include <LibWeb/Bindings/SVGElementConstructor.h> #include <LibWeb/Bindings/SVGElementPrototype.h> #include <LibWeb/Bindings/SVGGeometryElementConstructor.h> @@ -434,6 +436,7 @@ ADD_WINDOW_OBJECT_INTERFACE(SubmitEvent) \ ADD_WINDOW_OBJECT_INTERFACE(SubtleCrypto) \ ADD_WINDOW_OBJECT_INTERFACE(SVGElement) \ + ADD_WINDOW_OBJECT_INTERFACE(SVGCircleElement) \ ADD_WINDOW_OBJECT_INTERFACE(SVGGeometryElement) \ ADD_WINDOW_OBJECT_INTERFACE(SVGGraphicsElement) \ ADD_WINDOW_OBJECT_INTERFACE(SVGPathElement) \ diff --git a/Userland/Libraries/LibWeb/CMakeLists.txt b/Userland/Libraries/LibWeb/CMakeLists.txt index ec8bc6edf9..5447a8068d 100644 --- a/Userland/Libraries/LibWeb/CMakeLists.txt +++ b/Userland/Libraries/LibWeb/CMakeLists.txt @@ -271,6 +271,7 @@ set(SOURCES SVG/SVGGeometryElement.cpp SVG/SVGGraphicsElement.cpp SVG/SVGPathElement.cpp + SVG/SVGCircleElement.cpp SVG/SVGRectElement.cpp SVG/SVGSVGElement.cpp SVG/TagNames.cpp @@ -513,6 +514,7 @@ libweb_js_wrapper(ResizeObserver/ResizeObserver) libweb_js_wrapper(SVG/SVGElement) libweb_js_wrapper(SVG/SVGGeometryElement) libweb_js_wrapper(SVG/SVGGraphicsElement) +libweb_js_wrapper(SVG/SVGCircleElement) libweb_js_wrapper(SVG/SVGPathElement) libweb_js_wrapper(SVG/SVGRectElement) libweb_js_wrapper(SVG/SVGSVGElement) diff --git a/Userland/Libraries/LibWeb/DOM/ElementFactory.cpp b/Userland/Libraries/LibWeb/DOM/ElementFactory.cpp index afde1ff1ed..02e587b4a1 100644 --- a/Userland/Libraries/LibWeb/DOM/ElementFactory.cpp +++ b/Userland/Libraries/LibWeb/DOM/ElementFactory.cpp @@ -76,6 +76,7 @@ #include <LibWeb/HTML/HTMLUListElement.h> #include <LibWeb/HTML/HTMLUnknownElement.h> #include <LibWeb/HTML/HTMLVideoElement.h> +#include <LibWeb/SVG/SVGCircleElement.h> #include <LibWeb/SVG/SVGGElement.h> #include <LibWeb/SVG/SVGPathElement.h> #include <LibWeb/SVG/SVGRectElement.h> @@ -235,6 +236,8 @@ NonnullRefPtr<Element> create_element(Document& document, const FlyString& tag_n return adopt_ref(*new HTML::HTMLElement(document, move(qualified_name))); if (lowercase_tag_name == SVG::TagNames::svg) return adopt_ref(*new SVG::SVGSVGElement(document, move(qualified_name))); + if (lowercase_tag_name == SVG::TagNames::circle) + return adopt_ref(*new SVG::SVGCircleElement(document, move(qualified_name))); if (lowercase_tag_name == SVG::TagNames::path) return adopt_ref(*new SVG::SVGPathElement(document, move(qualified_name))); if (lowercase_tag_name == SVG::TagNames::rect) diff --git a/Userland/Libraries/LibWeb/Forward.h b/Userland/Libraries/LibWeb/Forward.h index 8dead35fb5..4bc2bd98db 100644 --- a/Userland/Libraries/LibWeb/Forward.h +++ b/Userland/Libraries/LibWeb/Forward.h @@ -244,6 +244,7 @@ class ResizeObserver; } namespace Web::SVG { +class SVGCircleElement; class SVGElement; class SVGGeometryElement; class SVGGraphicsElement; @@ -450,6 +451,7 @@ class StyleSheetListWrapper; class StyleSheetWrapper; class SubmitEventWrapper; class SubtleCryptoWrapper; +class SVGCircleElementWrapper; class SVGElementWrapper; class SVGGeometryElementWrapper; class SVGGraphicsElementWrapper; diff --git a/Userland/Libraries/LibWeb/SVG/AttributeNames.h b/Userland/Libraries/LibWeb/SVG/AttributeNames.h index 0cc3af1844..4f8ad259d8 100644 --- a/Userland/Libraries/LibWeb/SVG/AttributeNames.h +++ b/Userland/Libraries/LibWeb/SVG/AttributeNames.h @@ -19,6 +19,8 @@ namespace Web::SVG::AttributeNames { E(clipPathUnits) \ E(contentScriptType) \ E(contentStyleType) \ + E(cx) \ + E(cy) \ E(diffuseConstant) \ E(edgeMode) \ E(filterUnits) \ @@ -49,6 +51,7 @@ namespace Web::SVG::AttributeNames { E(preserveAlpha) \ E(preserveAspectRatio) \ E(primitiveUnits) \ + E(r) \ E(refX) \ E(refY) \ E(repeatCount) \ 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(); +} + +} diff --git a/Userland/Libraries/LibWeb/SVG/SVGCircleElement.h b/Userland/Libraries/LibWeb/SVG/SVGCircleElement.h new file mode 100644 index 0000000000..cf945aef27 --- /dev/null +++ b/Userland/Libraries/LibWeb/SVG/SVGCircleElement.h @@ -0,0 +1,32 @@ +/* + * Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#pragma once + +#include <LibWeb/SVG/SVGGeometryElement.h> + +namespace Web::SVG { + +class SVGCircleElement final : public SVGGeometryElement { +public: + using WrapperType = Bindings::SVGCircleElementWrapper; + + SVGCircleElement(DOM::Document&, QualifiedName); + virtual ~SVGCircleElement() override = default; + + virtual void parse_attribute(FlyString const& name, String const& value) override; + + virtual Gfx::Path& get_path() override; + +private: + Optional<Gfx::Path> m_path; + + Optional<float> m_center_x; + Optional<float> m_center_y; + Optional<float> m_radius; +}; + +} diff --git a/Userland/Libraries/LibWeb/SVG/SVGCircleElement.idl b/Userland/Libraries/LibWeb/SVG/SVGCircleElement.idl new file mode 100644 index 0000000000..5866cbaeff --- /dev/null +++ b/Userland/Libraries/LibWeb/SVG/SVGCircleElement.idl @@ -0,0 +1,6 @@ +[Exposed=Window] +interface SVGCircleElement : SVGGeometryElement { + // [SameObject] readonly attribute SVGAnimatedLength cx; + // [SameObject] readonly attribute SVGAnimatedLength cy; + // [SameObject] readonly attribute SVGAnimatedLength r; +}; diff --git a/Userland/Libraries/LibWeb/SVG/TagNames.h b/Userland/Libraries/LibWeb/SVG/TagNames.h index f6791cdeed..ef4065ae71 100644 --- a/Userland/Libraries/LibWeb/SVG/TagNames.h +++ b/Userland/Libraries/LibWeb/SVG/TagNames.h @@ -11,6 +11,7 @@ namespace Web::SVG::TagNames { #define ENUMERATE_SVG_GRAPHICS_TAGS \ + __ENUMERATE_SVG_TAG(circle) \ __ENUMERATE_SVG_TAG(g) \ __ENUMERATE_SVG_TAG(path) \ __ENUMERATE_SVG_TAG(rect) \ |