<rdar://problem/9973489> REGRESSION (r66599): -[DOMNode boundingBox] returns the...
[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 "SVGStyledElement.h"
38
39 namespace WebCore {
40
41 RenderSVGModelObject::RenderSVGModelObject(SVGStyledElement* node)
42     : RenderObject(node)
43 {
44 }
45
46 LayoutRect RenderSVGModelObject::clippedOverflowRectForRepaint(RenderBoxModelObject* repaintContainer) const
47 {
48     return SVGRenderSupport::clippedOverflowRectForRepaint(this, repaintContainer);
49 }
50
51 void RenderSVGModelObject::computeRectForRepaint(RenderBoxModelObject* repaintContainer, LayoutRect& repaintRect, bool fixed) const
52 {
53     SVGRenderSupport::computeRectForRepaint(this, repaintContainer, repaintRect, fixed);
54 }
55
56 void RenderSVGModelObject::mapLocalToContainer(RenderBoxModelObject* repaintContainer, bool /* fixed */, bool /* useTransforms */, TransformState& transformState, bool* wasFixed) const
57 {
58     SVGRenderSupport::mapLocalToContainer(this, repaintContainer, transformState, wasFixed);
59 }
60
61 // Copied from RenderBox, this method likely requires further refactoring to work easily for both SVG and CSS Box Model content.
62 // FIXME: This may also need to move into SVGRenderSupport as the RenderBox version depends
63 // on borderBoundingBox() which SVG RenderBox subclases (like SVGRenderBlock) do not implement.
64 LayoutRect RenderSVGModelObject::outlineBoundsForRepaint(RenderBoxModelObject* repaintContainer, LayoutPoint*) const
65 {
66     LayoutRect box = enclosingLayoutRect(repaintRectInLocalCoordinates());
67     adjustRectForOutlineAndShadow(box);
68
69     FloatQuad containerRelativeQuad = localToContainerQuad(FloatRect(box), repaintContainer);
70     return containerRelativeQuad.enclosingBoundingBox();
71 }
72
73 void RenderSVGModelObject::absoluteRects(Vector<LayoutRect>& rects, const LayoutPoint& accumulatedOffset)
74 {
75     LayoutRect rect = enclosingLayoutRect(strokeBoundingBox());
76     rect.moveBy(accumulatedOffset);
77     rects.append(rect);
78 }
79
80 void RenderSVGModelObject::absoluteQuads(Vector<FloatQuad>& quads, bool* wasFixed)
81 {
82     quads.append(localToAbsoluteQuad(strokeBoundingBox(), false, wasFixed));
83 }
84
85 void RenderSVGModelObject::willBeDestroyed()
86 {
87     SVGResourcesCache::clientDestroyed(this);
88     RenderObject::willBeDestroyed();
89 }
90
91 void RenderSVGModelObject::styleWillChange(StyleDifference diff, const RenderStyle* newStyle)
92 {
93     if (diff == StyleDifferenceLayout)
94         setNeedsBoundariesUpdate();
95     RenderObject::styleWillChange(diff, newStyle);
96 }
97
98 void RenderSVGModelObject::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
99 {
100     RenderObject::styleDidChange(diff, oldStyle);
101     SVGResourcesCache::clientStyleChanged(this, diff, style());
102 }
103
104 void RenderSVGModelObject::updateFromElement()
105 {
106     RenderObject::updateFromElement();
107     SVGResourcesCache::clientUpdatedFromElement(this, style());
108 }
109
110 bool RenderSVGModelObject::nodeAtPoint(const HitTestRequest&, HitTestResult&, const LayoutPoint&, const LayoutPoint&, HitTestAction)
111 {
112     ASSERT_NOT_REACHED();
113     return false;
114 }
115
116 static void getElementCTM(SVGElement* element, AffineTransform& transform)
117 {
118     ASSERT(element);
119     element->document()->updateLayoutIgnorePendingStylesheets();
120
121     SVGElement* stopAtElement = SVGLocatable::nearestViewportElement(element);
122     ASSERT(stopAtElement);
123
124     Node* current = element;
125     while (current && current->isSVGElement()) {
126         SVGElement* currentElement = static_cast<SVGElement*>(current);
127         if (currentElement->isStyled())
128             transform = const_cast<AffineTransform&>(currentElement->renderer()->localToParentTransform()).multiply(transform);
129
130         // For getCTM() computation, stop at the nearest viewport element
131         if (currentElement == stopAtElement)
132             break;
133
134         current = current->parentOrHostNode();
135     }
136 }
137
138 // FloatRect::intersects does not consider horizontal or vertical lines (because of isEmpty()).
139 // So special-case handling of such lines.
140 static bool intersectsAllowingEmpty(const FloatRect& r, const FloatRect& other)
141 {
142     if (r.isEmpty() && other.isEmpty())
143         return false;
144     if (r.isEmpty() && !other.isEmpty()) {
145         return (other.contains(r.x(), r.y()) && !other.contains(r.maxX(), r.maxY()))
146                || (!other.contains(r.x(), r.y()) && other.contains(r.maxX(), r.maxY()));
147     }
148     if (other.isEmpty() && !r.isEmpty())
149         return intersectsAllowingEmpty(other, r);
150     return r.intersects(other);
151 }
152
153 // One of the element types that can cause graphics to be drawn onto the target canvas. Specifically: circle, ellipse,
154 // image, line, path, polygon, polyline, rect, text and use.
155 static bool isGraphicsElement(RenderObject* renderer)
156 {
157     return renderer->isSVGPath() || renderer->isSVGText() || renderer->isSVGImage() || renderer->isSVGShadowTreeRootContainer();
158 }
159
160 bool RenderSVGModelObject::checkIntersection(RenderObject* renderer, const FloatRect& rect)
161 {
162     if (!renderer || renderer->style()->pointerEvents() == PE_NONE)
163         return false;
164     if (!isGraphicsElement(renderer))
165         return false;
166     AffineTransform ctm;
167     getElementCTM(static_cast<SVGElement*>(renderer->node()), ctm);
168     return intersectsAllowingEmpty(rect, ctm.mapRect(renderer->repaintRectInLocalCoordinates()));
169 }
170
171 bool RenderSVGModelObject::checkEnclosure(RenderObject* renderer, const FloatRect& rect)
172 {
173     if (!renderer || renderer->style()->pointerEvents() == PE_NONE)
174         return false;
175     if (!isGraphicsElement(renderer))
176         return false;
177     AffineTransform ctm;
178     getElementCTM(static_cast<SVGElement*>(renderer->node()), ctm);
179     return rect.contains(ctm.mapRect(renderer->repaintRectInLocalCoordinates()));
180 }
181
182 } // namespace WebCore
183
184 #endif // ENABLE(SVG)