Rename functions in SVGDocumentExtension
[WebKit-https.git] / Source / WebCore / svg / SVGTextPathElement.cpp
1 /*
2  * Copyright (C) 2007 Nikolas Zimmermann <zimmermann@kde.org>
3  * Copyright (C) 2010 Rob Buis <rwlbuis@gmail.com>
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
23 #if ENABLE(SVG)
24 #include "SVGTextPathElement.h"
25
26 #include "Attribute.h"
27 #include "RenderSVGResource.h"
28 #include "RenderSVGTextPath.h"
29 #include "SVGElementInstance.h"
30 #include "SVGNames.h"
31 #include "XLinkNames.h"
32
33 namespace WebCore {
34
35 // Animated property definitions
36 DEFINE_ANIMATED_LENGTH(SVGTextPathElement, SVGNames::startOffsetAttr, StartOffset, startOffset)
37 DEFINE_ANIMATED_ENUMERATION(SVGTextPathElement, SVGNames::methodAttr, Method, method, SVGTextPathMethodType)
38 DEFINE_ANIMATED_ENUMERATION(SVGTextPathElement, SVGNames::spacingAttr, Spacing, spacing, SVGTextPathSpacingType)
39 DEFINE_ANIMATED_STRING(SVGTextPathElement, XLinkNames::hrefAttr, Href, href)
40
41 BEGIN_REGISTER_ANIMATED_PROPERTIES(SVGTextPathElement)
42     REGISTER_LOCAL_ANIMATED_PROPERTY(startOffset)
43     REGISTER_LOCAL_ANIMATED_PROPERTY(method)
44     REGISTER_LOCAL_ANIMATED_PROPERTY(spacing)
45     REGISTER_LOCAL_ANIMATED_PROPERTY(href)
46     REGISTER_PARENT_ANIMATED_PROPERTIES(SVGTextContentElement)
47 END_REGISTER_ANIMATED_PROPERTIES
48
49 inline SVGTextPathElement::SVGTextPathElement(const QualifiedName& tagName, Document& document)
50     : SVGTextContentElement(tagName, document)
51     , m_startOffset(LengthModeOther)
52     , m_method(SVGTextPathMethodAlign)
53     , m_spacing(SVGTextPathSpacingExact)
54 {
55     ASSERT(hasTagName(SVGNames::textPathTag));
56     registerAnimatedPropertiesForSVGTextPathElement();
57 }
58
59 PassRefPtr<SVGTextPathElement> SVGTextPathElement::create(const QualifiedName& tagName, Document& document)
60 {
61     return adoptRef(new SVGTextPathElement(tagName, document));
62 }
63
64 SVGTextPathElement::~SVGTextPathElement()
65 {
66     clearResourceReferences();
67 }
68
69 void SVGTextPathElement::clearResourceReferences()
70 {
71     document().accessSVGExtensions()->removeAllTargetReferencesForElement(this);
72 }
73
74 bool SVGTextPathElement::isSupportedAttribute(const QualifiedName& attrName)
75 {
76     DEFINE_STATIC_LOCAL(HashSet<QualifiedName>, supportedAttributes, ());
77     if (supportedAttributes.isEmpty()) {
78         SVGURIReference::addSupportedAttributes(supportedAttributes);
79         supportedAttributes.add(SVGNames::startOffsetAttr);
80         supportedAttributes.add(SVGNames::methodAttr);
81         supportedAttributes.add(SVGNames::spacingAttr);
82     }
83     return supportedAttributes.contains<SVGAttributeHashTranslator>(attrName);
84 }
85
86 void SVGTextPathElement::parseAttribute(const QualifiedName& name, const AtomicString& value)
87 {
88     SVGParsingError parseError = NoError;
89
90     if (!isSupportedAttribute(name))
91         SVGTextContentElement::parseAttribute(name, value);
92     else if (name == SVGNames::startOffsetAttr)
93         setStartOffsetBaseValue(SVGLength::construct(LengthModeOther, value, parseError));
94     else if (name == SVGNames::methodAttr) {
95         SVGTextPathMethodType propertyValue = SVGPropertyTraits<SVGTextPathMethodType>::fromString(value);
96         if (propertyValue > 0)
97             setMethodBaseValue(propertyValue);
98     } else if (name == SVGNames::spacingAttr) {
99         SVGTextPathSpacingType propertyValue = SVGPropertyTraits<SVGTextPathSpacingType>::fromString(value);
100         if (propertyValue > 0)
101             setSpacingBaseValue(propertyValue);
102     } else if (SVGURIReference::parseAttribute(name, value)) {
103     } else
104         ASSERT_NOT_REACHED();
105
106     reportAttributeParsingError(parseError, name, value);
107 }
108
109 void SVGTextPathElement::svgAttributeChanged(const QualifiedName& attrName)
110 {
111     if (!isSupportedAttribute(attrName)) {
112         SVGTextContentElement::svgAttributeChanged(attrName);
113         return;
114     }
115
116     SVGElementInstance::InvalidationGuard invalidationGuard(this);
117
118     if (SVGURIReference::isKnownAttribute(attrName)) {
119         buildPendingResource();
120         return;
121     }
122
123     if (attrName == SVGNames::startOffsetAttr)
124         updateRelativeLengthsInformation();
125
126     if (auto renderer = this->renderer())
127         RenderSVGResource::markForLayoutAndParentResourceInvalidation(*renderer);
128 }
129
130 RenderPtr<RenderElement> SVGTextPathElement::createElementRenderer(PassRef<RenderStyle> style)
131 {
132     return createRenderer<RenderSVGTextPath>(*this, std::move(style));
133 }
134
135 bool SVGTextPathElement::childShouldCreateRenderer(const Node& child) const
136 {
137     if (child.isTextNode()
138         || child.hasTagName(SVGNames::aTag)
139         || child.hasTagName(SVGNames::trefTag)
140         || child.hasTagName(SVGNames::tspanTag))
141         return true;
142
143     return false;
144 }
145
146 bool SVGTextPathElement::rendererIsNeeded(const RenderStyle& style)
147 {
148     if (parentNode()
149         && (parentNode()->hasTagName(SVGNames::aTag)
150             || parentNode()->hasTagName(SVGNames::textTag)))
151         return StyledElement::rendererIsNeeded(style);
152
153     return false;
154 }
155
156 void SVGTextPathElement::buildPendingResource()
157 {
158     clearResourceReferences();
159     if (!inDocument())
160         return;
161
162     String id;
163     Element* target = SVGURIReference::targetElementFromIRIString(href(), document(), &id);
164     if (!target) {
165         // Do not register as pending if we are already pending this resource.
166         if (document().accessSVGExtensions()->isPendingResource(this, id))
167             return;
168
169         if (!id.isEmpty()) {
170             document().accessSVGExtensions()->addPendingResource(id, this);
171             ASSERT(hasPendingResources());
172         }
173     } else if (target->hasTagName(SVGNames::pathTag)) {
174         // Register us with the target in the dependencies map. Any change of hrefElement
175         // that leads to relayout/repainting now informs us, so we can react to it.
176         document().accessSVGExtensions()->addElementReferencingTarget(this, toSVGElement(target));
177     }
178 }
179
180 Node::InsertionNotificationRequest SVGTextPathElement::insertedInto(ContainerNode& rootParent)
181 {
182     SVGTextContentElement::insertedInto(rootParent);
183     buildPendingResource();
184     return InsertionDone;
185 }
186
187 void SVGTextPathElement::removedFrom(ContainerNode& rootParent)
188 {
189     SVGTextContentElement::removedFrom(rootParent);
190     if (rootParent.inDocument())
191         clearResourceReferences();
192 }
193
194 bool SVGTextPathElement::selfHasRelativeLengths() const
195 {
196     return startOffset().isRelative()
197         || SVGTextContentElement::selfHasRelativeLengths();
198 }
199
200 }
201
202 #endif // ENABLE(SVG)