StyledElement: Rename parseMappedAttribute() to parseAttribute().
[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
32 namespace WebCore {
33
34 // Animated property definitions
35 DEFINE_ANIMATED_LENGTH(SVGTextPathElement, SVGNames::startOffsetAttr, StartOffset, startOffset)
36 DEFINE_ANIMATED_ENUMERATION(SVGTextPathElement, SVGNames::methodAttr, Method, method, SVGTextPathMethodType)
37 DEFINE_ANIMATED_ENUMERATION(SVGTextPathElement, SVGNames::spacingAttr, Spacing, spacing, SVGTextPathSpacingType)
38 DEFINE_ANIMATED_STRING(SVGTextPathElement, XLinkNames::hrefAttr, Href, href)
39
40 BEGIN_REGISTER_ANIMATED_PROPERTIES(SVGTextPathElement)
41     REGISTER_LOCAL_ANIMATED_PROPERTY(startOffset)
42     REGISTER_LOCAL_ANIMATED_PROPERTY(method)
43     REGISTER_LOCAL_ANIMATED_PROPERTY(spacing)
44     REGISTER_LOCAL_ANIMATED_PROPERTY(href)
45     REGISTER_PARENT_ANIMATED_PROPERTIES(SVGTextContentElement)
46 END_REGISTER_ANIMATED_PROPERTIES
47
48 inline SVGTextPathElement::SVGTextPathElement(const QualifiedName& tagName, Document* document)
49     : SVGTextContentElement(tagName, document)
50     , m_startOffset(LengthModeOther)
51     , m_method(SVGTextPathMethodAlign)
52     , m_spacing(SVGTextPathSpacingExact)
53 {
54     ASSERT(hasTagName(SVGNames::textPathTag));
55     registerAnimatedPropertiesForSVGTextPathElement();
56 }
57
58 PassRefPtr<SVGTextPathElement> SVGTextPathElement::create(const QualifiedName& tagName, Document* document)
59 {
60     return adoptRef(new SVGTextPathElement(tagName, document));
61 }
62
63 bool SVGTextPathElement::isSupportedAttribute(const QualifiedName& attrName)
64 {
65     DEFINE_STATIC_LOCAL(HashSet<QualifiedName>, supportedAttributes, ());
66     if (supportedAttributes.isEmpty()) {
67         SVGURIReference::addSupportedAttributes(supportedAttributes);
68         supportedAttributes.add(SVGNames::startOffsetAttr);
69         supportedAttributes.add(SVGNames::methodAttr);
70         supportedAttributes.add(SVGNames::spacingAttr);
71     }
72     return supportedAttributes.contains<QualifiedName, SVGAttributeHashTranslator>(attrName);
73 }
74
75 void SVGTextPathElement::parseAttribute(Attribute* attr)
76 {
77     SVGParsingError parseError = NoError;
78     const AtomicString& value = attr->value();
79
80     if (!isSupportedAttribute(attr->name()))
81         SVGTextContentElement::parseAttribute(attr);
82     else if (attr->name() == SVGNames::startOffsetAttr)
83         setStartOffsetBaseValue(SVGLength::construct(LengthModeOther, value, parseError));
84     else if (attr->name() == SVGNames::methodAttr) {
85         SVGTextPathMethodType propertyValue = SVGPropertyTraits<SVGTextPathMethodType>::fromString(value);
86         if (propertyValue > 0)
87             setMethodBaseValue(propertyValue);
88     } else if (attr->name() == SVGNames::spacingAttr) {
89         SVGTextPathSpacingType propertyValue = SVGPropertyTraits<SVGTextPathSpacingType>::fromString(value);
90         if (propertyValue > 0)
91             setSpacingBaseValue(propertyValue);
92     } else if (SVGURIReference::parseAttribute(attr)) {
93     } else
94         ASSERT_NOT_REACHED();
95
96     reportAttributeParsingError(parseError, attr);
97 }
98
99 void SVGTextPathElement::svgAttributeChanged(const QualifiedName& attrName)
100 {
101     if (!isSupportedAttribute(attrName)) {
102         SVGTextContentElement::svgAttributeChanged(attrName);
103         return;
104     }
105
106     SVGElementInstance::InvalidationGuard invalidationGuard(this);
107
108     if (attrName == SVGNames::startOffsetAttr)
109         updateRelativeLengthsInformation();
110
111     if (RenderObject* object = renderer())
112         RenderSVGResource::markForLayoutAndParentResourceInvalidation(object);
113 }
114
115 RenderObject* SVGTextPathElement::createRenderer(RenderArena* arena, RenderStyle*)
116 {
117     return new (arena) RenderSVGTextPath(this);
118 }
119
120 bool SVGTextPathElement::childShouldCreateRenderer(Node* child) const
121 {
122     if (child->isTextNode()
123         || child->hasTagName(SVGNames::aTag)
124         || child->hasTagName(SVGNames::trefTag)
125         || child->hasTagName(SVGNames::tspanTag))
126         return true;
127
128     return false;
129 }
130
131 bool SVGTextPathElement::rendererIsNeeded(const NodeRenderingContext& context)
132 {
133     if (parentNode()
134         && (parentNode()->hasTagName(SVGNames::aTag)
135             || parentNode()->hasTagName(SVGNames::textTag)))
136         return StyledElement::rendererIsNeeded(context);
137
138     return false;
139 }
140
141 void SVGTextPathElement::insertedIntoDocument()
142 {
143     SVGTextContentElement::insertedIntoDocument();
144
145     String id;
146     Element* targetElement = SVGURIReference::targetElementFromIRIString(href(), document(), &id);
147     if (!targetElement) {
148         if (hasPendingResources() || id.isEmpty())
149             return;
150
151         ASSERT(!hasPendingResources());
152         document()->accessSVGExtensions()->addPendingResource(id, this);
153         ASSERT(hasPendingResources());
154         return;
155     }
156 }
157
158 bool SVGTextPathElement::selfHasRelativeLengths() const
159 {
160     return startOffset().isRelative()
161         || SVGTextContentElement::selfHasRelativeLengths();
162 }
163
164 }
165
166 #endif // ENABLE(SVG)