Improve use of NeverDestroyed
[WebKit-https.git] / Source / WebCore / svg / SVGForeignObjectElement.cpp
1 /*
2  * Copyright (C) 2006 Apple Inc. All rights reserved.
3  * Copyright (C) 2008 Nikolas Zimmermann <zimmermann@kde.org>
4  * Copyright (C) 2014 Adobe Systems Incorporated. All rights reserved.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public License
17  * along with this library; see the file COPYING.LIB.  If not, write to
18  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21
22 #include "config.h"
23 #include "SVGForeignObjectElement.h"
24
25 #include "CSSPropertyNames.h"
26 #include "RenderSVGForeignObject.h"
27 #include "RenderSVGResource.h"
28 #include "SVGLengthValue.h"
29 #include "SVGNames.h"
30 #include "XLinkNames.h"
31 #include <wtf/Assertions.h>
32 #include <wtf/NeverDestroyed.h>
33
34 namespace WebCore {
35
36 // Animated property definitions
37 DEFINE_ANIMATED_LENGTH(SVGForeignObjectElement, SVGNames::xAttr, X, x)
38 DEFINE_ANIMATED_LENGTH(SVGForeignObjectElement, SVGNames::yAttr, Y, y)
39 DEFINE_ANIMATED_LENGTH(SVGForeignObjectElement, SVGNames::widthAttr, Width, width)
40 DEFINE_ANIMATED_LENGTH(SVGForeignObjectElement, SVGNames::heightAttr, Height, height)
41 DEFINE_ANIMATED_STRING(SVGForeignObjectElement, XLinkNames::hrefAttr, Href, href)
42 DEFINE_ANIMATED_BOOLEAN(SVGForeignObjectElement, SVGNames::externalResourcesRequiredAttr, ExternalResourcesRequired, externalResourcesRequired)
43
44 BEGIN_REGISTER_ANIMATED_PROPERTIES(SVGForeignObjectElement)
45     REGISTER_LOCAL_ANIMATED_PROPERTY(x)
46     REGISTER_LOCAL_ANIMATED_PROPERTY(y)
47     REGISTER_LOCAL_ANIMATED_PROPERTY(width)
48     REGISTER_LOCAL_ANIMATED_PROPERTY(height)
49     REGISTER_LOCAL_ANIMATED_PROPERTY(href)
50     REGISTER_LOCAL_ANIMATED_PROPERTY(externalResourcesRequired)
51     REGISTER_PARENT_ANIMATED_PROPERTIES(SVGGraphicsElement)
52 END_REGISTER_ANIMATED_PROPERTIES
53
54 inline SVGForeignObjectElement::SVGForeignObjectElement(const QualifiedName& tagName, Document& document)
55     : SVGGraphicsElement(tagName, document)
56     , m_x(LengthModeWidth)
57     , m_y(LengthModeHeight)
58     , m_width(LengthModeWidth)
59     , m_height(LengthModeHeight)
60 {
61     ASSERT(hasTagName(SVGNames::foreignObjectTag));
62     registerAnimatedPropertiesForSVGForeignObjectElement();
63 }
64
65 Ref<SVGForeignObjectElement> SVGForeignObjectElement::create(const QualifiedName& tagName, Document& document)
66 {
67     return adoptRef(*new SVGForeignObjectElement(tagName, document));
68 }
69
70 bool SVGForeignObjectElement::isSupportedAttribute(const QualifiedName& attrName)
71 {
72     static const auto supportedAttributes = makeNeverDestroyed([] {
73         HashSet<QualifiedName> set;
74         SVGLangSpace::addSupportedAttributes(set);
75         SVGExternalResourcesRequired::addSupportedAttributes(set);
76         set.add({ SVGNames::xAttr, SVGNames::yAttr, SVGNames::widthAttr, SVGNames::heightAttr });
77         return set;
78     }());
79     return supportedAttributes.get().contains<SVGAttributeHashTranslator>(attrName);
80 }
81
82 void SVGForeignObjectElement::parseAttribute(const QualifiedName& name, const AtomicString& value)
83 {
84     SVGParsingError parseError = NoError;
85
86     if (name == SVGNames::xAttr)
87         setXBaseValue(SVGLengthValue::construct(LengthModeWidth, value, parseError));
88     else if (name == SVGNames::yAttr)
89         setYBaseValue(SVGLengthValue::construct(LengthModeHeight, value, parseError));
90     else if (name == SVGNames::widthAttr)
91         setWidthBaseValue(SVGLengthValue::construct(LengthModeWidth, value, parseError));
92     else if (name == SVGNames::heightAttr)
93         setHeightBaseValue(SVGLengthValue::construct(LengthModeHeight, value, parseError));
94
95     reportAttributeParsingError(parseError, name, value);
96
97     SVGGraphicsElement::parseAttribute(name, value);
98     SVGExternalResourcesRequired::parseAttribute(name, value);
99 }
100
101 void SVGForeignObjectElement::svgAttributeChanged(const QualifiedName& attrName)
102 {
103     if (!isSupportedAttribute(attrName)) {
104         SVGGraphicsElement::svgAttributeChanged(attrName);
105         return;
106     }
107
108     InstanceInvalidationGuard guard(*this);
109
110     if (attrName == SVGNames::widthAttr
111         || attrName == SVGNames::heightAttr) {
112         invalidateSVGPresentationAttributeStyle();
113         return;
114     }
115
116     if (attrName == SVGNames::xAttr || attrName == SVGNames::yAttr)
117         updateRelativeLengthsInformation();
118
119     if (auto renderer = this->renderer())
120         RenderSVGResource::markForLayoutAndParentResourceInvalidation(*renderer);
121 }
122
123 RenderPtr<RenderElement> SVGForeignObjectElement::createElementRenderer(RenderStyle&& style, const RenderTreePosition&)
124 {
125     return createRenderer<RenderSVGForeignObject>(*this, WTFMove(style));
126 }
127
128 bool SVGForeignObjectElement::childShouldCreateRenderer(const Node& child) const
129 {
130     // Disallow arbitary SVG content. Only allow proper <svg xmlns="svgNS"> subdocuments.
131     if (child.isSVGElement())
132         return child.hasTagName(SVGNames::svgTag);
133
134     // Skip over SVG rules which disallow non-SVG kids
135     return StyledElement::childShouldCreateRenderer(child);
136 }
137
138 bool SVGForeignObjectElement::rendererIsNeeded(const RenderStyle& style)
139 {
140     // Suppress foreignObject renderers in SVG hidden containers.
141     // (https://bugs.webkit.org/show_bug.cgi?id=87297)
142     // Note that we currently do not support foreignObject instantiation via <use>, hence it is safe
143     // to use parentElement() here. If that changes, this method should be updated to use
144     // parentOrShadowHostElement() instead.
145     Element* ancestor = parentElement();
146     while (ancestor && ancestor->isSVGElement()) {
147         if (ancestor->renderer() && ancestor->renderer()->isSVGHiddenContainer())
148             return false;
149
150         ancestor = ancestor->parentElement();
151     }
152
153     return SVGGraphicsElement::rendererIsNeeded(style);
154 }
155
156 }