e652493e26834a241cbc8e013890bdb315097d4d
[WebKit-https.git] / Source / WebCore / svg / graphics / filters / SVGFEImage.cpp
1 /*
2  * Copyright (C) 2004, 2005, 2006, 2007 Nikolas Zimmermann <zimmermann@kde.org>
3  * Copyright (C) 2004, 2005 Rob Buis <buis@kde.org>
4  * Copyright (C) 2005 Eric Seidel <eric@webkit.org>
5  * Copyright (C) 2010 Dirk Schulze <krit@webkit.org>
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Library General Public
9  * License as published by the Free Software Foundation; either
10  * version 2 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Library General Public License for more details.
16  *
17  * You should have received a copy of the GNU Library General Public License
18  * along with this library; see the file COPYING.LIB.  If not, write to
19  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
20  * Boston, MA 02110-1301, USA.
21  */
22
23 #include "config.h"
24
25 #if ENABLE(FILTERS)
26 #include "SVGFEImage.h"
27
28 #include "AffineTransform.h"
29 #include "Filter.h"
30 #include "GraphicsContext.h"
31 #include "RenderElement.h"
32 #include "RenderTreeAsText.h"
33 #include "SVGElement.h"
34 #include "SVGFilter.h"
35 #include "SVGPreserveAspectRatio.h"
36 #include "SVGRenderingContext.h"
37 #include "SVGURIReference.h"
38 #include "TextStream.h"
39
40 namespace WebCore {
41
42 FEImage::FEImage(Filter* filter, PassRefPtr<Image> image, const SVGPreserveAspectRatio& preserveAspectRatio)
43     : FilterEffect(filter)
44     , m_image(image)
45     , m_document(0)
46     , m_preserveAspectRatio(preserveAspectRatio)
47 {
48 }
49
50 FEImage::FEImage(Filter* filter, Document& document, const String& href, const SVGPreserveAspectRatio& preserveAspectRatio)
51     : FilterEffect(filter)
52     , m_document(&document)
53     , m_href(href)
54     , m_preserveAspectRatio(preserveAspectRatio)
55 {
56 }
57
58 PassRefPtr<FEImage> FEImage::createWithImage(Filter* filter, PassRefPtr<Image> image, const SVGPreserveAspectRatio& preserveAspectRatio)
59 {
60     return adoptRef(new FEImage(filter, image, preserveAspectRatio));
61 }
62
63 PassRefPtr<FEImage> FEImage::createWithIRIReference(Filter* filter, Document& document, const String& href, const SVGPreserveAspectRatio& preserveAspectRatio)
64 {
65     return adoptRef(new FEImage(filter, document, href, preserveAspectRatio));
66 }
67
68 void FEImage::determineAbsolutePaintRect()
69 {
70     SVGFilter* svgFilter = toSVGFilter(&(filter()));
71
72     FloatRect paintRect = svgFilter->absoluteTransform().mapRect(filterPrimitiveSubregion());
73     FloatRect srcRect;
74     if (m_image) {
75         srcRect.setSize(m_image->size());
76         m_preserveAspectRatio.transformRect(paintRect, srcRect);
77     } else if (RenderElement* renderer = referencedRenderer())
78         srcRect = svgFilter->absoluteTransform().mapRect(renderer->repaintRectInLocalCoordinates());
79
80     if (clipsToBounds())
81         paintRect.intersect(maxEffectRect());
82     else
83         paintRect.unite(maxEffectRect());
84     setAbsolutePaintRect(enclosingIntRect(paintRect));
85 }
86
87 RenderElement* FEImage::referencedRenderer() const
88 {
89     if (!m_document)
90         return 0;
91     Element* hrefElement = SVGURIReference::targetElementFromIRIString(m_href, *m_document);
92     if (!hrefElement || !hrefElement->isSVGElement())
93         return 0;
94     return hrefElement->renderer();
95 }
96
97 void FEImage::platformApplySoftware()
98 {
99     RenderElement* renderer = referencedRenderer();
100     if (!m_image && !renderer)
101         return;
102
103     ImageBuffer* resultImage = createImageBufferResult();
104     if (!resultImage)
105         return;
106
107     SVGFilter* svgFilter = toSVGFilter(&(filter()));
108     FloatRect destRect = svgFilter->absoluteTransform().mapRect(filterPrimitiveSubregion());
109
110     FloatRect srcRect;
111     if (renderer)
112         srcRect = svgFilter->absoluteTransform().mapRect(renderer->repaintRectInLocalCoordinates());
113     else {
114         srcRect = FloatRect(FloatPoint(), m_image->size());
115         m_preserveAspectRatio.transformRect(destRect, srcRect);
116     }
117
118     IntPoint paintLocation = absolutePaintRect().location();
119     destRect.move(-paintLocation.x(), -paintLocation.y());
120
121     // FEImage results are always in ColorSpaceDeviceRGB
122     setResultColorSpace(ColorSpaceDeviceRGB);
123
124     if (renderer) {
125         const AffineTransform& absoluteTransform = svgFilter->absoluteTransform();
126         resultImage->context()->concatCTM(absoluteTransform);
127
128         SVGElement* contextNode = toSVGElement(renderer->element());
129         if (contextNode->hasRelativeLengths()) {
130             SVGLengthContext lengthContext(contextNode);
131             FloatSize viewportSize;
132
133             // If we're referencing an element with percentage units, eg. <rect with="30%"> those values were resolved against the viewport.
134             // Build up a transformation that maps from the viewport space to the filter primitive subregion.
135             if (lengthContext.determineViewport(viewportSize))
136                 resultImage->context()->concatCTM(makeMapBetweenRects(FloatRect(FloatPoint(), viewportSize), destRect));
137         }
138
139         AffineTransform contentTransformation;
140         SVGRenderingContext::renderSubtreeToImageBuffer(resultImage, *renderer, contentTransformation);
141         return;
142     }
143
144     resultImage->context()->drawImage(m_image.get(), ColorSpaceDeviceRGB, destRect, srcRect);
145 }
146
147 void FEImage::dump()
148 {
149 }
150
151 TextStream& FEImage::externalRepresentation(TextStream& ts, int indent) const
152 {
153     FloatSize imageSize;
154     if (m_image)
155         imageSize = m_image->size();
156     else if (RenderObject* renderer = referencedRenderer())
157         imageSize = enclosingIntRect(renderer->repaintRectInLocalCoordinates()).size();
158     writeIndent(ts, indent);
159     ts << "[feImage";
160     FilterEffect::externalRepresentation(ts);
161     ts << " image-size=\"" << imageSize.width() << "x" << imageSize.height() << "\"]\n";
162     // FIXME: should this dump also object returned by SVGFEImage::image() ?
163     return ts;
164 }
165
166 } // namespace WebCore
167
168 #endif // ENABLE(FILTERS)