[SVG] Moving more special casing of SVG out of the bindings - SVG lists
[WebKit-https.git] / Source / WebCore / svg / SVGGraphicsElement.cpp
1 /*
2  * Copyright (C) 2004, 2005, 2008 Nikolas Zimmermann <zimmermann@kde.org>
3  * Copyright (C) 2004, 2005, 2006 Rob Buis <buis@kde.org>
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Library General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Library General Public License for more details.
14  *
15  * You should have received a copy of the GNU Library General Public License
16  * along with this library; see the file COPYING.LIB.  If not, write to
17  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
18  * Boston, MA 02110-1301, USA.
19  */
20
21 #include "config.h"
22 #include "SVGGraphicsElement.h"
23
24 #include "RenderSVGPath.h"
25 #include "RenderSVGResource.h"
26 #include "SVGMatrix.h"
27 #include "SVGNames.h"
28 #include "SVGPathData.h"
29 #include "SVGRect.h"
30 #include "SVGStringList.h"
31 #include <wtf/NeverDestroyed.h>
32
33 namespace WebCore {
34
35 // Animated property definitions
36 DEFINE_ANIMATED_TRANSFORM_LIST(SVGGraphicsElement, SVGNames::transformAttr, Transform, transform)
37
38 BEGIN_REGISTER_ANIMATED_PROPERTIES(SVGGraphicsElement)
39     REGISTER_LOCAL_ANIMATED_PROPERTY(transform)
40     REGISTER_PARENT_ANIMATED_PROPERTIES(SVGElement)
41     REGISTER_PARENT_ANIMATED_PROPERTIES(SVGTests)
42 END_REGISTER_ANIMATED_PROPERTIES
43
44 SVGGraphicsElement::SVGGraphicsElement(const QualifiedName& tagName, Document& document)
45     : SVGElement(tagName, document)
46     , m_shouldIsolateBlending(false)
47 {
48     registerAnimatedPropertiesForSVGGraphicsElement();
49 }
50
51 SVGGraphicsElement::~SVGGraphicsElement()
52 {
53 }
54
55 Ref<SVGMatrix> SVGGraphicsElement::getCTMForBindings()
56 {
57     return SVGMatrix::create(getCTM());
58 }
59
60 AffineTransform SVGGraphicsElement::getCTM(StyleUpdateStrategy styleUpdateStrategy)
61 {
62     return SVGLocatable::computeCTM(this, SVGLocatable::NearestViewportScope, styleUpdateStrategy);
63 }
64
65 Ref<SVGMatrix> SVGGraphicsElement::getScreenCTMForBindings()
66 {
67     return SVGMatrix::create(getScreenCTM());
68 }
69
70 AffineTransform SVGGraphicsElement::getScreenCTM(StyleUpdateStrategy styleUpdateStrategy)
71 {
72     return SVGLocatable::computeCTM(this, SVGLocatable::ScreenScope, styleUpdateStrategy);
73 }
74
75 AffineTransform SVGGraphicsElement::animatedLocalTransform() const
76 {
77     AffineTransform matrix;
78     auto* style = renderer() ? &renderer()->style() : nullptr;
79
80     // If CSS property was set, use that, otherwise fallback to attribute (if set).
81     if (style && style->hasTransform()) {
82         // Note: objectBoundingBox is an emptyRect for elements like pattern or clipPath.
83         // See the "Object bounding box units" section of http://dev.w3.org/csswg/css3-transforms/
84         TransformationMatrix transform;
85         style->applyTransform(transform, renderer()->objectBoundingBox());
86
87         // Flatten any 3D transform.
88         matrix = transform.toAffineTransform();
89         // CSS bakes the zoom factor into lengths, including translation components.
90         // In order to align CSS & SVG transforms, we need to invert this operation.
91         float zoom = style->effectiveZoom();
92         if (zoom != 1) {
93             matrix.setE(matrix.e() / zoom);
94             matrix.setF(matrix.f() / zoom);
95         }
96
97     } else
98         transform().concatenate(matrix);
99
100     if (m_supplementalTransform)
101         return *m_supplementalTransform * matrix;
102     return matrix;
103 }
104
105 AffineTransform* SVGGraphicsElement::supplementalTransform()
106 {
107     if (!m_supplementalTransform)
108         m_supplementalTransform = std::make_unique<AffineTransform>();
109     return m_supplementalTransform.get();
110 }
111
112 bool SVGGraphicsElement::isSupportedAttribute(const QualifiedName& attrName)
113 {
114     static NeverDestroyed<HashSet<QualifiedName>> supportedAttributes;
115     if (supportedAttributes.get().isEmpty()) {
116         SVGTests::addSupportedAttributes(supportedAttributes);
117         supportedAttributes.get().add(SVGNames::transformAttr);
118     }
119     return supportedAttributes.get().contains<SVGAttributeHashTranslator>(attrName);
120 }
121
122 void SVGGraphicsElement::parseAttribute(const QualifiedName& name, const AtomicString& value)
123 {
124     if (name == SVGNames::transformAttr) {
125         SVGTransformListValues newList;
126         newList.parse(value);
127         detachAnimatedTransformListWrappers(newList.size());
128         setTransformBaseValue(newList);
129         return;
130     }
131
132     SVGElement::parseAttribute(name, value);
133     SVGTests::parseAttribute(name, value);
134 }
135
136 void SVGGraphicsElement::svgAttributeChanged(const QualifiedName& attrName)
137 {
138     if (!isSupportedAttribute(attrName)) {
139         SVGElement::svgAttributeChanged(attrName);
140         return;
141     }
142
143     InstanceInvalidationGuard guard(*this);
144
145     if (SVGTests::handleAttributeChange(this, attrName))
146         return;
147
148     auto renderer = this->renderer();
149     if (!renderer)
150         return;
151
152     if (attrName == SVGNames::transformAttr) {
153         renderer->setNeedsTransformUpdate();
154         RenderSVGResource::markForLayoutAndParentResourceInvalidation(*renderer);
155         return;
156     }
157
158     ASSERT_NOT_REACHED();
159 }
160
161 SVGElement* SVGGraphicsElement::nearestViewportElement() const
162 {
163     return SVGTransformable::nearestViewportElement(this);
164 }
165
166 SVGElement* SVGGraphicsElement::farthestViewportElement() const
167 {
168     return SVGTransformable::farthestViewportElement(this);
169 }
170
171 Ref<SVGRect> SVGGraphicsElement::getBBoxForBindings()
172 {
173     return SVGRect::create(getBBox());
174 }
175
176 FloatRect SVGGraphicsElement::getBBox(StyleUpdateStrategy styleUpdateStrategy)
177 {
178     return SVGTransformable::getBBox(this, styleUpdateStrategy);
179 }
180
181 RenderPtr<RenderElement> SVGGraphicsElement::createElementRenderer(RenderStyle&& style, const RenderTreePosition&)
182 {
183     return createRenderer<RenderSVGPath>(*this, WTFMove(style));
184 }
185
186 void SVGGraphicsElement::toClipPath(Path& path)
187 {
188     updatePathFromGraphicsElement(this, path);
189     // FIXME: How do we know the element has done a layout?
190     path.transform(animatedLocalTransform());
191 }
192
193 Ref<SVGStringList> SVGGraphicsElement::requiredFeatures()
194 {
195     return SVGTests::requiredFeatures(*this);
196 }
197
198 Ref<SVGStringList> SVGGraphicsElement::requiredExtensions()
199
200     return SVGTests::requiredExtensions(*this);
201 }
202
203 Ref<SVGStringList> SVGGraphicsElement::systemLanguage()
204 {
205     return SVGTests::systemLanguage(*this);
206 }
207
208 }