2 * Copyright 2009 Vincent Povirk for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
32 #include "wincodecs_private.h"
34 #include "wine/debug.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(wincodecs
);
39 const IWICBitmapDecoderVtbl
*lpVtbl
;
43 CRITICAL_SECTION lock
;
47 const IWICBitmapFrameDecodeVtbl
*lpVtbl
;
53 static HRESULT WINAPI
GifFrameDecode_QueryInterface(IWICBitmapFrameDecode
*iface
, REFIID iid
,
56 GifFrameDecode
*This
= (GifFrameDecode
*)iface
;
57 TRACE("(%p,%s,%p)\n", iface
, debugstr_guid(iid
), ppv
);
59 if (!ppv
) return E_INVALIDARG
;
61 if (IsEqualIID(&IID_IUnknown
, iid
) ||
62 IsEqualIID(&IID_IWICBitmapSource
, iid
) ||
63 IsEqualIID(&IID_IWICBitmapFrameDecode
, iid
))
73 IUnknown_AddRef((IUnknown
*)*ppv
);
77 static ULONG WINAPI
GifFrameDecode_AddRef(IWICBitmapFrameDecode
*iface
)
79 GifFrameDecode
*This
= (GifFrameDecode
*)iface
;
80 ULONG ref
= InterlockedIncrement(&This
->ref
);
82 TRACE("(%p) refcount=%u\n", iface
, ref
);
87 static ULONG WINAPI
GifFrameDecode_Release(IWICBitmapFrameDecode
*iface
)
89 GifFrameDecode
*This
= (GifFrameDecode
*)iface
;
90 ULONG ref
= InterlockedDecrement(&This
->ref
);
92 TRACE("(%p) refcount=%u\n", iface
, ref
);
96 IUnknown_Release((IUnknown
*)This
->parent
);
97 HeapFree(GetProcessHeap(), 0, This
);
103 static HRESULT WINAPI
GifFrameDecode_GetSize(IWICBitmapFrameDecode
*iface
,
104 UINT
*puiWidth
, UINT
*puiHeight
)
106 GifFrameDecode
*This
= (GifFrameDecode
*)iface
;
107 TRACE("(%p,%p,%p)\n", iface
, puiWidth
, puiHeight
);
109 *puiWidth
= This
->frame
->ImageDesc
.Width
;
110 *puiHeight
= This
->frame
->ImageDesc
.Height
;
115 static HRESULT WINAPI
GifFrameDecode_GetPixelFormat(IWICBitmapFrameDecode
*iface
,
116 WICPixelFormatGUID
*pPixelFormat
)
118 memcpy(pPixelFormat
, &GUID_WICPixelFormat8bppIndexed
, sizeof(GUID
));
123 static HRESULT WINAPI
GifFrameDecode_GetResolution(IWICBitmapFrameDecode
*iface
,
124 double *pDpiX
, double *pDpiY
)
126 FIXME("(%p,%p,%p): stub\n", iface
, pDpiX
, pDpiY
);
130 static HRESULT WINAPI
GifFrameDecode_CopyPalette(IWICBitmapFrameDecode
*iface
,
131 IWICPalette
*pIPalette
)
133 GifFrameDecode
*This
= (GifFrameDecode
*)iface
;
134 WICColor colors
[256];
135 ColorMapObject
*cm
= This
->frame
->ImageDesc
.ColorMap
;
138 TRACE("(%p,%p)\n", iface
, pIPalette
);
140 if (!cm
) cm
= This
->parent
->gif
->SColorMap
;
142 if (cm
->ColorCount
> 256)
144 ERR("GIF contains %i colors???\n", cm
->ColorCount
);
148 for (i
= 0; i
< cm
->ColorCount
; i
++) {
149 colors
[i
] = 0xff000000| /* alpha */
150 cm
->Colors
[i
].Red
<< 16|
151 cm
->Colors
[i
].Green
<< 8|
155 /* look for the transparent color extension */
156 for (i
= 0; i
< This
->frame
->ExtensionBlockCount
; ++i
) {
157 eb
= This
->frame
->ExtensionBlocks
+ i
;
158 if (eb
->Function
== 0xF9 && eb
->ByteCount
== 4) {
159 if ((eb
->Bytes
[0] & 1) == 1) {
160 trans
= (unsigned char)eb
->Bytes
[3];
161 colors
[trans
] &= 0xffffff; /* set alpha to 0 */
167 IWICPalette_InitializeCustom(pIPalette
, colors
, cm
->ColorCount
);
172 static HRESULT
copy_interlaced_pixels(const BYTE
*srcbuffer
,
173 UINT srcwidth
, UINT srcheight
, INT srcstride
, const WICRect
*rc
,
174 UINT dststride
, UINT dstbuffersize
, BYTE
*dstbuffer
)
176 UINT row_offset
; /* number of bytes into the source rows where the data starts */
181 if (rc
->X
< 0 || rc
->Y
< 0 || rc
->X
+rc
->Width
> srcwidth
|| rc
->Y
+rc
->Height
> srcheight
)
184 if (dststride
< rc
->Width
)
187 if ((dststride
* rc
->Height
) > dstbuffersize
)
193 for (y
=rc
->Y
; y
-rc
->Y
< rc
->Height
; y
++)
196 src
= srcbuffer
+ srcstride
* (y
/8);
198 src
= srcbuffer
+ srcstride
* ((srcheight
+7)/8 + y
/8);
200 src
= srcbuffer
+ srcstride
* ((srcheight
+3)/4 + y
/4);
202 src
= srcbuffer
+ srcstride
* ((srcheight
+1)/2 + y
/2);
204 memcpy(dst
, src
, rc
->Width
);
210 static HRESULT WINAPI
GifFrameDecode_CopyPixels(IWICBitmapFrameDecode
*iface
,
211 const WICRect
*prc
, UINT cbStride
, UINT cbBufferSize
, BYTE
*pbBuffer
)
213 GifFrameDecode
*This
= (GifFrameDecode
*)iface
;
214 TRACE("(%p,%p,%u,%u,%p)\n", iface
, prc
, cbStride
, cbBufferSize
, pbBuffer
);
216 if (This
->frame
->ImageDesc
.Interlace
)
218 return copy_interlaced_pixels(This
->frame
->RasterBits
, This
->frame
->ImageDesc
.Width
,
219 This
->frame
->ImageDesc
.Height
, This
->frame
->ImageDesc
.Width
,
220 prc
, cbStride
, cbBufferSize
, pbBuffer
);
224 return copy_pixels(8, This
->frame
->RasterBits
, This
->frame
->ImageDesc
.Width
,
225 This
->frame
->ImageDesc
.Height
, This
->frame
->ImageDesc
.Width
,
226 prc
, cbStride
, cbBufferSize
, pbBuffer
);
230 static HRESULT WINAPI
GifFrameDecode_GetMetadataQueryReader(IWICBitmapFrameDecode
*iface
,
231 IWICMetadataQueryReader
**ppIMetadataQueryReader
)
233 TRACE("(%p,%p)\n", iface
, ppIMetadataQueryReader
);
234 return WINCODEC_ERR_UNSUPPORTEDOPERATION
;
237 static HRESULT WINAPI
GifFrameDecode_GetColorContexts(IWICBitmapFrameDecode
*iface
,
238 UINT cCount
, IWICColorContext
**ppIColorContexts
, UINT
*pcActualCount
)
240 TRACE("(%p,%u,%p,%p)\n", iface
, cCount
, ppIColorContexts
, pcActualCount
);
241 return WINCODEC_ERR_UNSUPPORTEDOPERATION
;
244 static HRESULT WINAPI
GifFrameDecode_GetThumbnail(IWICBitmapFrameDecode
*iface
,
245 IWICBitmapSource
**ppIThumbnail
)
247 TRACE("(%p,%p)\n", iface
, ppIThumbnail
);
248 return WINCODEC_ERR_CODECNOTHUMBNAIL
;
251 static const IWICBitmapFrameDecodeVtbl GifFrameDecode_Vtbl
= {
252 GifFrameDecode_QueryInterface
,
253 GifFrameDecode_AddRef
,
254 GifFrameDecode_Release
,
255 GifFrameDecode_GetSize
,
256 GifFrameDecode_GetPixelFormat
,
257 GifFrameDecode_GetResolution
,
258 GifFrameDecode_CopyPalette
,
259 GifFrameDecode_CopyPixels
,
260 GifFrameDecode_GetMetadataQueryReader
,
261 GifFrameDecode_GetColorContexts
,
262 GifFrameDecode_GetThumbnail
265 static HRESULT WINAPI
GifDecoder_QueryInterface(IWICBitmapDecoder
*iface
, REFIID iid
,
268 GifDecoder
*This
= (GifDecoder
*)iface
;
269 TRACE("(%p,%s,%p)\n", iface
, debugstr_guid(iid
), ppv
);
271 if (!ppv
) return E_INVALIDARG
;
273 if (IsEqualIID(&IID_IUnknown
, iid
) || IsEqualIID(&IID_IWICBitmapDecoder
, iid
))
280 return E_NOINTERFACE
;
283 IUnknown_AddRef((IUnknown
*)*ppv
);
287 static ULONG WINAPI
GifDecoder_AddRef(IWICBitmapDecoder
*iface
)
289 GifDecoder
*This
= (GifDecoder
*)iface
;
290 ULONG ref
= InterlockedIncrement(&This
->ref
);
292 TRACE("(%p) refcount=%u\n", iface
, ref
);
297 static ULONG WINAPI
GifDecoder_Release(IWICBitmapDecoder
*iface
)
299 GifDecoder
*This
= (GifDecoder
*)iface
;
300 ULONG ref
= InterlockedDecrement(&This
->ref
);
302 TRACE("(%p) refcount=%u\n", iface
, ref
);
306 This
->lock
.DebugInfo
->Spare
[0] = 0;
307 DeleteCriticalSection(&This
->lock
);
308 DGifCloseFile(This
->gif
);
309 HeapFree(GetProcessHeap(), 0, This
);
315 static HRESULT WINAPI
GifDecoder_QueryCapability(IWICBitmapDecoder
*iface
, IStream
*pIStream
,
316 DWORD
*pdwCapability
)
318 FIXME("(%p,%p,%p): stub\n", iface
, pIStream
, pdwCapability
);
322 static int _gif_inputfunc(GifFileType
*gif
, GifByteType
*data
, int len
) {
323 IStream
*stream
= gif
->UserData
;
329 ERR("attempting to read file after initialization\n");
333 hr
= IStream_Read(stream
, data
, len
, &bytesread
);
334 if (hr
!= S_OK
) bytesread
= 0;
338 static HRESULT WINAPI
GifDecoder_Initialize(IWICBitmapDecoder
*iface
, IStream
*pIStream
,
339 WICDecodeOptions cacheOptions
)
341 GifDecoder
*This
= (GifDecoder
*)iface
;
345 TRACE("(%p,%p,%x)\n", iface
, pIStream
, cacheOptions
);
347 EnterCriticalSection(&This
->lock
);
349 if (This
->initialized
|| This
->gif
)
351 WARN("already initialized\n");
352 LeaveCriticalSection(&This
->lock
);
353 return WINCODEC_ERR_WRONGSTATE
;
356 /* seek to start of stream */
358 IStream_Seek(pIStream
, seek
, STREAM_SEEK_SET
, NULL
);
360 /* read all data from the stream */
361 This
->gif
= DGifOpen((void*)pIStream
, _gif_inputfunc
);
364 LeaveCriticalSection(&This
->lock
);
368 ret
= DGifSlurp(This
->gif
);
369 if (ret
== GIF_ERROR
)
371 LeaveCriticalSection(&This
->lock
);
375 /* make sure we don't use the stream after this method returns */
376 This
->gif
->UserData
= NULL
;
378 This
->initialized
= TRUE
;
380 LeaveCriticalSection(&This
->lock
);
385 static HRESULT WINAPI
GifDecoder_GetContainerFormat(IWICBitmapDecoder
*iface
,
386 GUID
*pguidContainerFormat
)
388 memcpy(pguidContainerFormat
, &GUID_ContainerFormatGif
, sizeof(GUID
));
392 static HRESULT WINAPI
GifDecoder_GetDecoderInfo(IWICBitmapDecoder
*iface
,
393 IWICBitmapDecoderInfo
**ppIDecoderInfo
)
396 IWICComponentInfo
*compinfo
;
398 TRACE("(%p,%p)\n", iface
, ppIDecoderInfo
);
400 hr
= CreateComponentInfo(&CLSID_WICGifDecoder
, &compinfo
);
401 if (FAILED(hr
)) return hr
;
403 hr
= IWICComponentInfo_QueryInterface(compinfo
, &IID_IWICBitmapDecoderInfo
,
404 (void**)ppIDecoderInfo
);
406 IWICComponentInfo_Release(compinfo
);
411 static HRESULT WINAPI
GifDecoder_CopyPalette(IWICBitmapDecoder
*iface
,
412 IWICPalette
*pIPalette
)
414 TRACE("(%p,%p)\n", iface
, pIPalette
);
415 return WINCODEC_ERR_PALETTEUNAVAILABLE
;
418 static HRESULT WINAPI
GifDecoder_GetMetadataQueryReader(IWICBitmapDecoder
*iface
,
419 IWICMetadataQueryReader
**ppIMetadataQueryReader
)
421 TRACE("(%p,%p)\n", iface
, ppIMetadataQueryReader
);
422 return WINCODEC_ERR_UNSUPPORTEDOPERATION
;
425 static HRESULT WINAPI
GifDecoder_GetPreview(IWICBitmapDecoder
*iface
,
426 IWICBitmapSource
**ppIBitmapSource
)
428 TRACE("(%p,%p)\n", iface
, ppIBitmapSource
);
429 return WINCODEC_ERR_UNSUPPORTEDOPERATION
;
432 static HRESULT WINAPI
GifDecoder_GetColorContexts(IWICBitmapDecoder
*iface
,
433 UINT cCount
, IWICColorContext
**ppIColorContexts
, UINT
*pcActualCount
)
435 TRACE("(%p,%u,%p,%p)\n", iface
, cCount
, ppIColorContexts
, pcActualCount
);
436 return WINCODEC_ERR_UNSUPPORTEDOPERATION
;
439 static HRESULT WINAPI
GifDecoder_GetThumbnail(IWICBitmapDecoder
*iface
,
440 IWICBitmapSource
**ppIThumbnail
)
442 TRACE("(%p,%p)\n", iface
, ppIThumbnail
);
443 return WINCODEC_ERR_CODECNOTHUMBNAIL
;
446 static HRESULT WINAPI
GifDecoder_GetFrameCount(IWICBitmapDecoder
*iface
,
449 GifDecoder
*This
= (GifDecoder
*)iface
;
450 TRACE("(%p,%p)\n", iface
, pCount
);
452 if (!This
->initialized
) return WINCODEC_ERR_NOTINITIALIZED
;
454 *pCount
= This
->gif
->ImageCount
;
456 TRACE("<- %u\n", *pCount
);
461 static HRESULT WINAPI
GifDecoder_GetFrame(IWICBitmapDecoder
*iface
,
462 UINT index
, IWICBitmapFrameDecode
**ppIBitmapFrame
)
464 GifDecoder
*This
= (GifDecoder
*)iface
;
465 GifFrameDecode
*result
;
466 TRACE("(%p,%u,%p)\n", iface
, index
, ppIBitmapFrame
);
468 if (!This
->initialized
) return WINCODEC_ERR_NOTINITIALIZED
;
470 if (index
>= This
->gif
->ImageCount
) return E_INVALIDARG
;
472 result
= HeapAlloc(GetProcessHeap(), 0, sizeof(GifFrameDecode
));
473 if (!result
) return E_OUTOFMEMORY
;
475 result
->lpVtbl
= &GifFrameDecode_Vtbl
;
477 result
->frame
= &This
->gif
->SavedImages
[index
];
478 IWICBitmapDecoder_AddRef(iface
);
479 result
->parent
= This
;
481 *ppIBitmapFrame
= (IWICBitmapFrameDecode
*)result
;
486 static const IWICBitmapDecoderVtbl GifDecoder_Vtbl
= {
487 GifDecoder_QueryInterface
,
490 GifDecoder_QueryCapability
,
491 GifDecoder_Initialize
,
492 GifDecoder_GetContainerFormat
,
493 GifDecoder_GetDecoderInfo
,
494 GifDecoder_CopyPalette
,
495 GifDecoder_GetMetadataQueryReader
,
496 GifDecoder_GetPreview
,
497 GifDecoder_GetColorContexts
,
498 GifDecoder_GetThumbnail
,
499 GifDecoder_GetFrameCount
,
503 HRESULT
GifDecoder_CreateInstance(IUnknown
*pUnkOuter
, REFIID iid
, void** ppv
)
508 TRACE("(%p,%s,%p)\n", pUnkOuter
, debugstr_guid(iid
), ppv
);
512 if (pUnkOuter
) return CLASS_E_NOAGGREGATION
;
514 This
= HeapAlloc(GetProcessHeap(), 0, sizeof(GifDecoder
));
515 if (!This
) return E_OUTOFMEMORY
;
517 This
->lpVtbl
= &GifDecoder_Vtbl
;
519 This
->initialized
= FALSE
;
521 InitializeCriticalSection(&This
->lock
);
522 This
->lock
.DebugInfo
->Spare
[0] = (DWORD_PTR
)(__FILE__
": GifDecoder.lock");
524 ret
= IUnknown_QueryInterface((IUnknown
*)This
, iid
, ppv
);
525 IUnknown_Release((IUnknown
*)This
);