1 /*
2 * Copyright (C) 2004, 2005, 2006, 2008 Nikolas Zimmermann <zimmermann@kde.org>
3 * Copyright (C) 2004, 2005, 2006, 2007 Rob Buis <buis@kde.org>
4 * Copyright (C) 2008 Eric Seidel <eric@webkit.org>
5 * Copyright (C) 2008 Dirk Schulze <krit@webkit.org>
6 * Copyright (C) Research In Motion Limited 2010. All rights reserved.
7 *
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Library General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
12 *
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Library General Public License for more details.
17 *
18 * You should have received a copy of the GNU Library General Public License
19 * along with this library; see the file COPYING.LIB. If not, write to
20 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
21 * Boston, MA 02110-1301, USA.
22 */
23
24 #include "config.h"
25
26 #if ENABLE(SVG)
27 #include "SVGLinearGradientElement.h"
28
29 #include "Attribute.h"
30 #include "Document.h"
31 #include "FloatPoint.h"
32 #include "LinearGradientAttributes.h"
33 #include "RenderSVGResourceLinearGradient.h"
34 #include "SVGLength.h"
35 #include "SVGNames.h"
36 #include "SVGTransform.h"
37 #include "SVGTransformList.h"
38 #include "SVGUnitTypes.h"
39
40 namespace WebCore {
41
42 // Animated property definitions
DEFINE_ANIMATED_LENGTH(SVGLinearGradientElement,SVGNames::x1Attr,X1,x1)43 DEFINE_ANIMATED_LENGTH(SVGLinearGradientElement, SVGNames::x1Attr, X1, x1)
44 DEFINE_ANIMATED_LENGTH(SVGLinearGradientElement, SVGNames::y1Attr, Y1, y1)
45 DEFINE_ANIMATED_LENGTH(SVGLinearGradientElement, SVGNames::x2Attr, X2, x2)
46 DEFINE_ANIMATED_LENGTH(SVGLinearGradientElement, SVGNames::y2Attr, Y2, y2)
47
48 inline SVGLinearGradientElement::SVGLinearGradientElement(const QualifiedName& tagName, Document* document)
49 : SVGGradientElement(tagName, document)
50 , m_x1(LengthModeWidth)
51 , m_y1(LengthModeHeight)
52 , m_x2(LengthModeWidth, "100%")
53 , m_y2(LengthModeHeight)
54 {
55 // Spec: If the x2 attribute is not specified, the effect is as if a value of "100%" were specified.
56 }
57
create(const QualifiedName & tagName,Document * document)58 PassRefPtr<SVGLinearGradientElement> SVGLinearGradientElement::create(const QualifiedName& tagName, Document* document)
59 {
60 return adoptRef(new SVGLinearGradientElement(tagName, document));
61 }
62
parseMappedAttribute(Attribute * attr)63 void SVGLinearGradientElement::parseMappedAttribute(Attribute* attr)
64 {
65 if (attr->name() == SVGNames::x1Attr)
66 setX1BaseValue(SVGLength(LengthModeWidth, attr->value()));
67 else if (attr->name() == SVGNames::y1Attr)
68 setY1BaseValue(SVGLength(LengthModeHeight, attr->value()));
69 else if (attr->name() == SVGNames::x2Attr)
70 setX2BaseValue(SVGLength(LengthModeWidth, attr->value()));
71 else if (attr->name() == SVGNames::y2Attr)
72 setY2BaseValue(SVGLength(LengthModeHeight, attr->value()));
73 else
74 SVGGradientElement::parseMappedAttribute(attr);
75 }
76
svgAttributeChanged(const QualifiedName & attrName)77 void SVGLinearGradientElement::svgAttributeChanged(const QualifiedName& attrName)
78 {
79 SVGGradientElement::svgAttributeChanged(attrName);
80
81 if (attrName == SVGNames::x1Attr
82 || attrName == SVGNames::y1Attr
83 || attrName == SVGNames::x2Attr
84 || attrName == SVGNames::y2Attr) {
85 updateRelativeLengthsInformation();
86
87 RenderObject* object = renderer();
88 if (!object)
89 return;
90
91 object->setNeedsLayout(true);
92 }
93 }
94
synchronizeProperty(const QualifiedName & attrName)95 void SVGLinearGradientElement::synchronizeProperty(const QualifiedName& attrName)
96 {
97 SVGGradientElement::synchronizeProperty(attrName);
98
99 if (attrName == anyQName()) {
100 synchronizeX1();
101 synchronizeY1();
102 synchronizeX2();
103 synchronizeY2();
104 return;
105 }
106
107 if (attrName == SVGNames::x1Attr)
108 synchronizeX1();
109 else if (attrName == SVGNames::y1Attr)
110 synchronizeY1();
111 else if (attrName == SVGNames::x2Attr)
112 synchronizeX2();
113 else if (attrName == SVGNames::y2Attr)
114 synchronizeY2();
115 }
116
attributeToPropertyTypeMap()117 AttributeToPropertyTypeMap& SVGLinearGradientElement::attributeToPropertyTypeMap()
118 {
119 DEFINE_STATIC_LOCAL(AttributeToPropertyTypeMap, s_attributeToPropertyTypeMap, ());
120 return s_attributeToPropertyTypeMap;
121 }
122
fillAttributeToPropertyTypeMap()123 void SVGLinearGradientElement::fillAttributeToPropertyTypeMap()
124 {
125 AttributeToPropertyTypeMap& attributeToPropertyTypeMap = this->attributeToPropertyTypeMap();
126
127 SVGGradientElement::fillPassedAttributeToPropertyTypeMap(attributeToPropertyTypeMap);
128 attributeToPropertyTypeMap.set(SVGNames::x1Attr, AnimatedLength);
129 attributeToPropertyTypeMap.set(SVGNames::y1Attr, AnimatedLength);
130 attributeToPropertyTypeMap.set(SVGNames::x2Attr, AnimatedLength);
131 attributeToPropertyTypeMap.set(SVGNames::y2Attr, AnimatedLength);
132 }
133
createRenderer(RenderArena * arena,RenderStyle *)134 RenderObject* SVGLinearGradientElement::createRenderer(RenderArena* arena, RenderStyle*)
135 {
136 return new (arena) RenderSVGResourceLinearGradient(this);
137 }
138
collectGradientAttributes(LinearGradientAttributes & attributes)139 void SVGLinearGradientElement::collectGradientAttributes(LinearGradientAttributes& attributes)
140 {
141 HashSet<SVGGradientElement*> processedGradients;
142
143 bool isLinear = true;
144 SVGGradientElement* current = this;
145
146 while (current) {
147 if (!attributes.hasSpreadMethod() && current->hasAttribute(SVGNames::spreadMethodAttr))
148 attributes.setSpreadMethod((GradientSpreadMethod) current->spreadMethod());
149
150 if (!attributes.hasBoundingBoxMode() && current->hasAttribute(SVGNames::gradientUnitsAttr))
151 attributes.setBoundingBoxMode(current->gradientUnits() == SVGUnitTypes::SVG_UNIT_TYPE_OBJECTBOUNDINGBOX);
152
153 if (!attributes.hasGradientTransform() && current->hasAttribute(SVGNames::gradientTransformAttr)) {
154 AffineTransform transform;
155 current->gradientTransform().concatenate(transform);
156 attributes.setGradientTransform(transform);
157 }
158
159 if (!attributes.hasStops()) {
160 const Vector<Gradient::ColorStop>& stops(current->buildStops());
161 if (!stops.isEmpty())
162 attributes.setStops(stops);
163 }
164
165 if (isLinear) {
166 SVGLinearGradientElement* linear = static_cast<SVGLinearGradientElement*>(current);
167
168 if (!attributes.hasX1() && current->hasAttribute(SVGNames::x1Attr))
169 attributes.setX1(linear->x1());
170
171 if (!attributes.hasY1() && current->hasAttribute(SVGNames::y1Attr))
172 attributes.setY1(linear->y1());
173
174 if (!attributes.hasX2() && current->hasAttribute(SVGNames::x2Attr))
175 attributes.setX2(linear->x2());
176
177 if (!attributes.hasY2() && current->hasAttribute(SVGNames::y2Attr))
178 attributes.setY2(linear->y2());
179 }
180
181 processedGradients.add(current);
182
183 // Respect xlink:href, take attributes from referenced element
184 Node* refNode = ownerDocument()->getElementById(SVGURIReference::getTarget(current->href()));
185 if (refNode && (refNode->hasTagName(SVGNames::linearGradientTag) || refNode->hasTagName(SVGNames::radialGradientTag))) {
186 current = static_cast<SVGGradientElement*>(refNode);
187
188 // Cycle detection
189 if (processedGradients.contains(current)) {
190 current = 0;
191 break;
192 }
193
194 isLinear = current->hasTagName(SVGNames::linearGradientTag);
195 } else
196 current = 0;
197 }
198 }
199
calculateStartEndPoints(const LinearGradientAttributes & attributes,FloatPoint & startPoint,FloatPoint & endPoint)200 void SVGLinearGradientElement::calculateStartEndPoints(const LinearGradientAttributes& attributes, FloatPoint& startPoint, FloatPoint& endPoint)
201 {
202 // Determine gradient start/end points
203 if (attributes.boundingBoxMode()) {
204 startPoint = FloatPoint(attributes.x1().valueAsPercentage(), attributes.y1().valueAsPercentage());
205 endPoint = FloatPoint(attributes.x2().valueAsPercentage(), attributes.y2().valueAsPercentage());
206 } else {
207 startPoint = FloatPoint(attributes.x1().value(this), attributes.y1().value(this));
208 endPoint = FloatPoint(attributes.x2().value(this), attributes.y2().value(this));
209 }
210 }
211
selfHasRelativeLengths() const212 bool SVGLinearGradientElement::selfHasRelativeLengths() const
213 {
214 return x1().isRelative()
215 || y1().isRelative()
216 || x2().isRelative()
217 || y2().isRelative();
218 }
219
220 }
221
222 #endif // ENABLE(SVG)
223