2011-11-29 Nikolas Zimmermann <nzimmermann@rim.com>
[WebKit-https.git] / Source / WebCore / rendering / svg / RenderSVGModelObject.cpp
1 /*
2  * Copyright (c) 2009, Google Inc. All rights reserved.
3  * 
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  * 
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  * 
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30
31 #include "config.h"
32
33 #if ENABLE(SVG)
34 #include "RenderSVGModelObject.h"
35
36 #include "RenderSVGResource.h"
37 #include "SVGResourcesCache.h"
38 #include "SVGStyledElement.h"
39
40 namespace WebCore {
41
42 RenderSVGModelObject::RenderSVGModelObject(SVGStyledElement* node)
43     : RenderObject(node)
44 {
45 }
46
47 LayoutRect RenderSVGModelObject::clippedOverflowRectForRepaint(RenderBoxModelObject* repaintContainer) const
48 {
49     return SVGRenderSupport::clippedOverflowRectForRepaint(this, repaintContainer);
50 }
51
52 void RenderSVGModelObject::computeFloatRectForRepaint(RenderBoxModelObject* repaintContainer, FloatRect& repaintRect, bool fixed) const
53 {
54     SVGRenderSupport::computeFloatRectForRepaint(this, repaintContainer, repaintRect, fixed);
55 }
56
57 void RenderSVGModelObject::mapLocalToContainer(RenderBoxModelObject* repaintContainer, bool /* fixed */, bool /* useTransforms */, TransformState& transformState, bool* wasFixed) const
58 {
59     SVGRenderSupport::mapLocalToContainer(this, repaintContainer, transformState, wasFixed);
60 }
61
62 // Copied from RenderBox, this method likely requires further refactoring to work easily for both SVG and CSS Box Model content.
63 // FIXME: This may also need to move into SVGRenderSupport as the RenderBox version depends
64 // on borderBoundingBox() which SVG RenderBox subclases (like SVGRenderBlock) do not implement.
65 LayoutRect RenderSVGModelObject::outlineBoundsForRepaint(RenderBoxModelObject* repaintContainer, LayoutPoint*) const
66 {
67     LayoutRect box = enclosingLayoutRect(repaintRectInLocalCoordinates());
68     adjustRectForOutlineAndShadow(box);
69
70     FloatQuad containerRelativeQuad = localToContainerQuad(FloatRect(box), repaintContainer);
71     return containerRelativeQuad.enclosingBoundingBox();
72 }
73
74 void RenderSVGModelObject::absoluteRects(Vector<LayoutRect>& rects, const LayoutPoint& accumulatedOffset) const
75 {
76     LayoutRect rect = enclosingLayoutRect(strokeBoundingBox());
77     rect.moveBy(accumulatedOffset);
78     rects.append(rect);
79 }
80
81 void RenderSVGModelObject::absoluteQuads(Vector<FloatQuad>& quads, bool* wasFixed) const
82 {
83     quads.append(localToAbsoluteQuad(strokeBoundingBox(), false, wasFixed));
84 }
85
86 void RenderSVGModelObject::willBeDestroyed()
87 {
88     SVGResourcesCache::clientDestroyed(this);
89     RenderObject::willBeDestroyed();
90 }
91
92 void RenderSVGModelObject::styleWillChange(StyleDifference diff, const RenderStyle* newStyle)
93 {
94     if (diff == StyleDifferenceLayout) {
95         setNeedsBoundariesUpdate();
96         if (newStyle->hasTransform())
97             setNeedsTransformUpdate();
98     }
99     RenderObject::styleWillChange(diff, newStyle);
100 }
101
102 void RenderSVGModelObject::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
103 {
104     RenderObject::styleDidChange(diff, oldStyle);
105     SVGResourcesCache::clientStyleChanged(this, diff, style());
106 }
107
108 void RenderSVGModelObject::updateFromElement()
109 {
110     RenderObject::updateFromElement();
111     SVGResourcesCache::clientUpdatedFromElement(this, style());
112 }
113
114 bool RenderSVGModelObject::nodeAtPoint(const HitTestRequest&, HitTestResult&, const LayoutPoint&, const LayoutPoint&, HitTestAction)
115 {
116     ASSERT_NOT_REACHED();
117     return false;
118 }
119
120 static void getElementCTM(SVGElement* element, AffineTransform& transform)
121 {
122     ASSERT(element);
123     element->document()->updateLayoutIgnorePendingStylesheets();
124
125     SVGElement* stopAtElement = SVGLocatable::nearestViewportElement(element);
126     ASSERT(stopAtElement);
127
128     Node* current = element;
129     while (current && current->isSVGElement()) {
130         SVGElement* currentElement = static_cast<SVGElement*>(current);
131         if (currentElement->isStyled())
132             transform = const_cast<AffineTransform&>(currentElement->renderer()->localToParentTransform()).multiply(transform);
133
134         // For getCTM() computation, stop at the nearest viewport element
135         if (currentElement == stopAtElement)
136             break;
137
138         current = current->parentOrHostNode();
139     }
140 }
141
142 // FloatRect::intersects does not consider horizontal or vertical lines (because of isEmpty()).
143 // So special-case handling of such lines.
144 static bool intersectsAllowingEmpty(const FloatRect& r, const FloatRect& other)
145 {
146     if (r.isEmpty() && other.isEmpty())
147         return false;
148     if (r.isEmpty() && !other.isEmpty()) {
149         return (other.contains(r.x(), r.y()) && !other.contains(r.maxX(), r.maxY()))
150                || (!other.contains(r.x(), r.y()) && other.contains(r.maxX(), r.maxY()));
151     }
152     if (other.isEmpty() && !r.isEmpty())
153         return intersectsAllowingEmpty(other, r);
154     return r.intersects(other);
155 }
156
157 // One of the element types that can cause graphics to be drawn onto the target canvas. Specifically: circle, ellipse,
158 // image, line, path, polygon, polyline, rect, text and use.
159 static bool isGraphicsElement(RenderObject* renderer)
160 {
161     return renderer->isSVGPath() || renderer->isSVGText() || renderer->isSVGImage() || renderer->isSVGShadowTreeRootContainer();
162 }
163
164 bool RenderSVGModelObject::checkIntersection(RenderObject* renderer, const FloatRect& rect)
165 {
166     if (!renderer || renderer->style()->pointerEvents() == PE_NONE)
167         return false;
168     if (!isGraphicsElement(renderer))
169         return false;
170     AffineTransform ctm;
171     getElementCTM(static_cast<SVGElement*>(renderer->node()), ctm);
172     return intersectsAllowingEmpty(rect, ctm.mapRect(renderer->repaintRectInLocalCoordinates()));
173 }
174
175 bool RenderSVGModelObject::checkEnclosure(RenderObject* renderer, const FloatRect& rect)
176 {
177     if (!renderer || renderer->style()->pointerEvents() == PE_NONE)
178         return false;
179     if (!isGraphicsElement(renderer))
180         return false;
181     AffineTransform ctm;
182     getElementCTM(static_cast<SVGElement*>(renderer->node()), ctm);
183     return rect.contains(ctm.mapRect(renderer->repaintRectInLocalCoordinates()));
184 }
185
186 } // namespace WebCore
187
188 #endif // ENABLE(SVG)