Use the Qt image loading framework instead of the Mozilla
[WebKit-https.git] / WebCore / platform / graphics / qt / ImageSourceQt.cpp
1 /*
2  * Copyright (C) 2006 Apple Computer, Inc.  All rights reserved. 
3  * Copyright (C) 2006 Trolltech ASA
4  *
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
17  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
20  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
21  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
22  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
23  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
24  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28
29 #include "config.h"
30 #include "ImageSource.h"
31 #include "ImageDecoderQt.h"
32
33
34 #include <QImage>
35 #include <qdebug.h>
36
37
38 namespace WebCore {
39     enum ImageFormat { ImageFormat_None, ImageFormat_GIF, ImageFormat_PNG, ImageFormat_JPEG,
40            ImageFormat_BMP,  ImageFormat_ICO,  ImageFormat_XBM };
41
42 ImageFormat  detectImageFormat(const Vector<char>& data)
43 {
44     // We need at least 4 bytes to figure out what kind of image we're dealing with.
45     int length = data.size();
46     if (length < 4)
47         return ImageFormat_None;
48
49     const unsigned char* uContents = (const unsigned char*) data.data();
50     const char* contents = data.data();
51
52     // GIFs begin with GIF8(7 or 9).
53     if (strncmp(contents, "GIF8", 4) == 0)
54         return ImageFormat_GIF;
55
56     // Test for PNG.
57     if (uContents[0] == 0x89 &&
58         uContents[1] == 0x50 &&
59         uContents[2] == 0x4E &&
60         uContents[3] == 0x47)
61         return ImageFormat_PNG;
62
63     // JPEG
64     if (uContents[0] == 0xFF &&
65         uContents[1] == 0xD8 &&
66         uContents[2] == 0xFF)
67         return ImageFormat_JPEG;
68
69     // BMP
70     if (strncmp(contents, "BM", 2) == 0)
71         return ImageFormat_BMP;
72
73     // ICOs always begin with a 2-byte 0 followed by a 2-byte 1.
74     // CURs begin with 2-byte 0 followed by 2-byte 2.
75     if (!memcmp(contents, "\000\000\001\000", 4) ||
76         !memcmp(contents, "\000\000\002\000", 4))
77         return ImageFormat_ICO;
78
79     // XBMs require 8 bytes of info.
80     if (length >= 8 && strncmp(contents, "#define ", 8) == 0)
81         return ImageFormat_XBM;
82
83     // Give up. We don't know what the heck this is.
84     return ImageFormat_None;
85 }
86     
87 ImageDecoderQt* createDecoder(const Vector<char>& data) {
88     if (detectImageFormat(data) != ImageFormat_None) 
89         return new ImageDecoderQt();
90     return 0;
91 }
92
93 ImageSource::ImageSource()
94     : m_decoder(0)
95 {
96 }
97
98 ImageSource::~ImageSource()
99 {
100     delete m_decoder;
101 }
102
103 bool ImageSource::initialized() const
104 {
105     return m_decoder;
106 }
107
108 void ImageSource::setData(const Vector<char>* data, bool allDataReceived)
109 {
110     if ( m_decoder) {
111         delete  m_decoder;
112         m_decoder = 0;
113     }
114     // Make the decoder by sniffing the bytes.
115     // This method will examine the data and instantiate an instance of the appropriate decoder plugin.
116     // If insufficient bytes are available to determine the image type, no decoder plugin will be
117     // made.
118     m_decoder = createDecoder(*data);
119
120     if (!m_decoder)
121         return;
122
123     m_decoder->setData(*data, allDataReceived);
124 }
125
126 bool ImageSource::isSizeAvailable()
127 {
128     if (!m_decoder)
129         return false;
130
131     return m_decoder->isSizeAvailable();
132 }
133
134 IntSize ImageSource::size() const
135 {
136     if (!m_decoder)
137         return IntSize();
138
139     return m_decoder->size();
140 }
141
142 int ImageSource::repetitionCount()
143 {
144     if (!m_decoder)
145         return cAnimationNone;
146
147     return m_decoder->repetitionCount();
148 }
149
150 size_t ImageSource::frameCount() const
151 {
152     if (!m_decoder)
153         return 0;
154
155     return m_decoder->frameCount();
156 }
157
158 NativeImagePtr ImageSource::createFrameAtIndex(size_t index)
159 {
160     if (!m_decoder)
161         return 0;
162
163     
164     const QImage* source = m_decoder->imageAtIndex( index);
165     if (!source)
166         return 0;
167
168     return new QImage(*source);
169 }
170
171 float ImageSource::frameDurationAtIndex(size_t index)
172 {
173     if (!m_decoder)
174         return 0;
175     
176     return m_decoder->duration(index) / 1000.0f;
177 }
178
179 bool ImageSource::frameHasAlphaAtIndex(size_t index)
180 {
181     if (!m_decoder || !m_decoder->supportsAlpha())
182         return false;
183     
184     const QImage* source = m_decoder->imageAtIndex( index);
185     if (!source)
186         return false;
187     
188     return source->hasAlphaChannel();
189 }
190
191 }
192
193 // vim: ts=4 sw=4 et