2011-11-10 Nikolas Zimmermann <nzimmermann@rim.com>
[WebKit-https.git] / Source / WebCore / rendering / svg / RenderSVGResourceRadialGradient.cpp
1 /*
2  * Copyright (C) 2006 Nikolas Zimmermann <zimmermann@kde.org>
3  * Copyright (C) Research In Motion Limited 2010. All rights reserved.
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 "RenderSVGResourceRadialGradient.h"
25
26 #include "RadialGradientAttributes.h"
27 #include "SVGRadialGradientElement.h"
28
29 namespace WebCore {
30
31 RenderSVGResourceType RenderSVGResourceRadialGradient::s_resourceType = RadialGradientResourceType;
32
33 RenderSVGResourceRadialGradient::RenderSVGResourceRadialGradient(SVGRadialGradientElement* node)
34     : RenderSVGResourceGradient(node)
35 {
36 }
37
38 RenderSVGResourceRadialGradient::~RenderSVGResourceRadialGradient()
39 {
40 }
41
42 bool RenderSVGResourceRadialGradient::collectGradientAttributes(SVGGradientElement* gradientElement)
43 {
44     m_attributes = RadialGradientAttributes();
45     return static_cast<SVGRadialGradientElement*>(gradientElement)->collectGradientAttributes(m_attributes);
46 }
47
48 FloatPoint RenderSVGResourceRadialGradient::centerPoint(const RadialGradientAttributes& attributes) const
49 {
50     return SVGLengthContext::resolvePoint(static_cast<const SVGElement*>(node()), attributes.gradientUnits(), attributes.cx(), attributes.cy());
51 }
52
53 FloatPoint RenderSVGResourceRadialGradient::focalPoint(const RadialGradientAttributes& attributes) const
54 {
55     return SVGLengthContext::resolvePoint(static_cast<const SVGElement*>(node()), attributes.gradientUnits(), attributes.fx(), attributes.fy());
56 }
57
58 float RenderSVGResourceRadialGradient::radius(const RadialGradientAttributes& attributes) const
59 {
60     return SVGLengthContext::resolveLength(static_cast<const SVGElement*>(node()), attributes.gradientUnits(), attributes.r());
61 }
62
63 void RenderSVGResourceRadialGradient::adjustFocalPointIfNeeded(float radius, const FloatPoint& centerPoint, FloatPoint& focalPoint) const
64 {
65     // Eventually adjust focal points, as described below.
66     float deltaX = focalPoint.x() - centerPoint.x();
67     float deltaY = focalPoint.y() - centerPoint.y();
68     float radiusMax = 0.99f * radius;
69
70     // Spec: If (fx, fy) lies outside the circle defined by (cx, cy) and r, set
71     // (fx, fy) to the point of intersection of the line through (fx, fy) and the circle.
72     // We scale the radius by 0.99 to match the behavior of FireFox.
73     if (sqrt(deltaX * deltaX + deltaY * deltaY) <= radiusMax)
74         return;
75
76     float angle = atan2f(deltaY, deltaX);
77     deltaX = cosf(angle) * radiusMax;
78     deltaY = sinf(angle) * radiusMax;
79     focalPoint = FloatPoint(deltaX + centerPoint.x(), deltaY + centerPoint.y());
80 }
81
82 void RenderSVGResourceRadialGradient::buildGradient(GradientData* gradientData) const
83 {
84     // Determine gradient focal/center points and radius
85     float radius = this->radius(m_attributes);
86     FloatPoint centerPoint = this->centerPoint(m_attributes);
87     FloatPoint focalPoint = this->focalPoint(m_attributes);
88     adjustFocalPointIfNeeded(radius, centerPoint, focalPoint);
89
90     gradientData->gradient = Gradient::create(focalPoint,
91                                               0, // SVG does not support a "focus radius"
92                                               centerPoint,
93                                               radius);
94
95     gradientData->gradient->setSpreadMethod(platformSpreadMethodFromSVGType(m_attributes.spreadMethod()));
96
97     addStops(gradientData, m_attributes.stops());
98 }
99
100 }
101
102 #endif