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