Remove unnecessary #include SVGResourcesCache.h in SVGDocumentExtensions.h; use forwa...
[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::computeRectForRepaint(RenderBoxModelObject* repaintContainer, LayoutRect& repaintRect, bool fixed) const
53 {
54     SVGRenderSupport::computeRectForRepaint(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     RenderObject::styleWillChange(diff, newStyle);
97 }
98
99 void RenderSVGModelObject::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
100 {
101     RenderObject::styleDidChange(diff, oldStyle);
102     SVGResourcesCache::clientStyleChanged(this, diff, style());
103 }
104
105 void RenderSVGModelObject::updateFromElement()
106 {
107     RenderObject::updateFromElement();
108     SVGResourcesCache::clientUpdatedFromElement(this, style());
109 }
110
111 bool RenderSVGModelObject::nodeAtPoint(const HitTestRequest&, HitTestResult&, const LayoutPoint&, const LayoutPoint&, HitTestAction)
112 {
113     ASSERT_NOT_REACHED();
114     return false;
115 }
116
117 static void getElementCTM(SVGElement* element, AffineTransform& transform)
118 {
119     ASSERT(element);
120     element->document()->updateLayoutIgnorePendingStylesheets();
121
122     SVGElement* stopAtElement = SVGLocatable::nearestViewportElement(element);
123     ASSERT(stopAtElement);
124
125     Node* current = element;
126     while (current && current->isSVGElement()) {
127         SVGElement* currentElement = static_cast<SVGElement*>(current);
128         if (currentElement->isStyled())
129             transform = const_cast<AffineTransform&>(currentElement->renderer()->localToParentTransform()).multiply(transform);
130
131         // For getCTM() computation, stop at the nearest viewport element
132         if (currentElement == stopAtElement)
133             break;
134
135         current = current->parentOrHostNode();
136     }
137 }
138
139 // FloatRect::intersects does not consider horizontal or vertical lines (because of isEmpty()).
140 // So special-case handling of such lines.
141 static bool intersectsAllowingEmpty(const FloatRect& r, const FloatRect& other)
142 {
143     if (r.isEmpty() && other.isEmpty())
144         return false;
145     if (r.isEmpty() && !other.isEmpty()) {
146         return (other.contains(r.x(), r.y()) && !other.contains(r.maxX(), r.maxY()))
147                || (!other.contains(r.x(), r.y()) && other.contains(r.maxX(), r.maxY()));
148     }
149     if (other.isEmpty() && !r.isEmpty())
150         return intersectsAllowingEmpty(other, r);
151     return r.intersects(other);
152 }
153
154 // One of the element types that can cause graphics to be drawn onto the target canvas. Specifically: circle, ellipse,
155 // image, line, path, polygon, polyline, rect, text and use.
156 static bool isGraphicsElement(RenderObject* renderer)
157 {
158     return renderer->isSVGPath() || renderer->isSVGText() || renderer->isSVGImage() || renderer->isSVGShadowTreeRootContainer();
159 }
160
161 bool RenderSVGModelObject::checkIntersection(RenderObject* renderer, const FloatRect& rect)
162 {
163     if (!renderer || renderer->style()->pointerEvents() == PE_NONE)
164         return false;
165     if (!isGraphicsElement(renderer))
166         return false;
167     AffineTransform ctm;
168     getElementCTM(static_cast<SVGElement*>(renderer->node()), ctm);
169     return intersectsAllowingEmpty(rect, ctm.mapRect(renderer->repaintRectInLocalCoordinates()));
170 }
171
172 bool RenderSVGModelObject::checkEnclosure(RenderObject* renderer, const FloatRect& rect)
173 {
174     if (!renderer || renderer->style()->pointerEvents() == PE_NONE)
175         return false;
176     if (!isGraphicsElement(renderer))
177         return false;
178     AffineTransform ctm;
179     getElementCTM(static_cast<SVGElement*>(renderer->node()), ctm);
180     return rect.contains(ctm.mapRect(renderer->repaintRectInLocalCoordinates()));
181 }
182
183 } // namespace WebCore
184
185 #endif // ENABLE(SVG)