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