2007-12-23 Alp Toker <alp@atoker.com>
[WebKit-https.git] / WebCore / html / CanvasPattern.cpp
1 /*
2  * Copyright (C) 2006 Apple Computer, 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
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "CanvasPattern.h"
28
29 #include "CachedImage.h"
30 #include "ExceptionCode.h"
31 #include "FloatRect.h"
32 #include "GraphicsContext.h"
33 #include "Image.h"
34
35 namespace WebCore {
36
37 void CanvasPattern::parseRepetitionType(const String& type, bool& repeatX, bool& repeatY, ExceptionCode& ec)
38 {
39     if (type.isEmpty() || type == "repeat") {
40         repeatX = true;
41         repeatY = true;
42         ec = 0;
43         return;
44     }
45     if (type == "no-repeat") {
46         repeatX = false;
47         repeatY = false;
48         ec = 0;
49         return;
50     }
51     if (type == "repeat-x") {
52         repeatX = true;
53         repeatY = false;
54         ec = 0;
55         return;
56     }
57     if (type == "repeat-y") {
58         repeatX = false;
59         repeatY = true;
60         ec = 0;
61         return;
62     }
63     ec = SYNTAX_ERR;
64 }
65
66 #if PLATFORM(CG)
67
68 CanvasPattern::CanvasPattern(CGImageRef image, bool repeatX, bool repeatY)
69     : m_platformImage(image)
70     , m_cachedImage(0)
71     , m_repeatX(repeatX)
72     , m_repeatY(repeatY)
73 {
74 }
75
76 #elif PLATFORM(CAIRO)
77
78 CanvasPattern::CanvasPattern(cairo_surface_t* surface, bool repeatX, bool repeatY)
79     : m_platformImage(surface)
80     , m_cachedImage(0)
81     , m_repeatX(repeatX)
82     , m_repeatY(repeatY)
83 {
84 }
85
86 #endif
87
88 CanvasPattern::CanvasPattern(CachedImage* cachedImage, bool repeatX, bool repeatY)
89     :
90 #if PLATFORM(CG) || PLATFORM(CAIRO)
91       m_platformImage(0)
92     ,
93 #endif
94       m_cachedImage(cachedImage)
95     , m_repeatX(repeatX)
96     , m_repeatY(repeatY)
97 {
98     if (cachedImage)
99         cachedImage->ref(this);
100 }
101
102 CanvasPattern::~CanvasPattern()
103 {
104 #if PLATFORM(CAIRO)
105     if (m_platformImage)
106         cairo_surface_destroy(m_platformImage);
107 #endif
108     if (m_cachedImage)
109         m_cachedImage->deref(this);
110 }
111
112 #if PLATFORM(CG)
113
114 static void patternCallback(void* info, CGContextRef context)
115 {
116     CGImageRef platformImage = static_cast<CanvasPattern*>(info)->platformImage();
117     if (platformImage) {
118         CGRect rect = GraphicsContext(context).roundToDevicePixels(
119             FloatRect(0, 0, CGImageGetWidth(platformImage), CGImageGetHeight(platformImage)));
120         CGContextDrawImage(context, rect, platformImage);
121         return;
122     }
123
124     CachedImage* cachedImage = static_cast<CanvasPattern*>(info)->cachedImage();
125     if (!cachedImage)
126         return;
127     Image* image = cachedImage->image();
128     if (!image)
129         return;
130
131     FloatRect rect = GraphicsContext(context).roundToDevicePixels(image->rect());
132
133     if (image->getCGImageRef()) {
134         CGContextDrawImage(context, rect, image->getCGImageRef());
135         // FIXME: We should refactor this code to use the platform-independent 
136         // drawing API in all cases. Then, this didDraw call will happen 
137         // automatically, and we can remove it.
138         cachedImage->didDraw(image);
139         return;
140     }
141
142     GraphicsContext(context).drawImage(image, rect);
143 }
144
145 static void patternReleaseCallback(void* info)
146 {
147     static_cast<CanvasPattern*>(info)->deref();
148 }
149
150 CGPatternRef CanvasPattern::createPattern(const CGAffineTransform& transform)
151 {
152     CGRect rect;
153     rect.origin.x = 0;
154     rect.origin.y = 0;
155     if (m_platformImage) {
156         rect.size.width = CGImageGetWidth(m_platformImage.get());
157         rect.size.height = CGImageGetHeight(m_platformImage.get());
158     } else {
159         if (!m_cachedImage)
160             return 0;
161         Image* image = m_cachedImage->image();
162         if (!image)
163             return 0;
164         rect.size.width = image->width();
165         rect.size.height = image->height();
166     }
167
168     CGAffineTransform patternTransform =
169         CGAffineTransformTranslate(CGAffineTransformScale(transform, 1, -1), 0, -rect.size.height);
170
171     float xStep = m_repeatX ? rect.size.width : FLT_MAX;
172     // If FLT_MAX should also be used for yStep, nothing is rendered. Using fractions of FLT_MAX also
173     // result in nothing being rendered. This is not a problem with xStep.
174     // INT_MAX is almost correct, but there seems to be some number wrapping occuring making the fill
175     // pattern is not filled correctly. 
176     // So, just pick a really large number that works. 
177     float yStep = m_repeatY ? rect.size.height : (100000000.0f);
178
179     const CGPatternCallbacks patternCallbacks = { 0, patternCallback, patternReleaseCallback };
180     ref();
181     return CGPatternCreate(this, rect, patternTransform, xStep, yStep,
182         kCGPatternTilingConstantSpacing, TRUE, &patternCallbacks);
183 }
184
185 #elif PLATFORM(CAIRO)
186
187 cairo_pattern_t* CanvasPattern::createPattern(const cairo_matrix_t& m)
188 {
189     cairo_surface_t* surface = 0;
190     if (m_platformImage) {
191         surface = m_platformImage;
192     } else {
193         if (!m_cachedImage)
194             return 0;
195         Image* image = m_cachedImage->image();
196         if (!image)
197             return 0;
198         surface = image->nativeImageForCurrentFrame();
199     }
200
201     if (!surface)
202         return 0;
203
204     cairo_pattern_t* pattern = cairo_pattern_create_for_surface(surface);
205     cairo_pattern_set_matrix(pattern, &m);
206     if (m_repeatX || m_repeatY)
207         cairo_pattern_set_extend(pattern, CAIRO_EXTEND_REPEAT);
208     return pattern;
209 }
210
211 #endif
212
213 }