]>
git.saurik.com Git - wxWidgets.git/blob - src/common/gifdecod.cpp
   1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxGIFDecoder, GIF reader for wxImage and wxAnimation 
   4 // Author:      Guillermo Rodriguez Garcia <guille@iies.es> 
   7 // Copyright:   (c) Guillermo Rodriguez Garcia 
   8 // Licence:     wxWindows licence 
   9 ///////////////////////////////////////////////////////////////////////////// 
  12 #pragma implementation "gifdecod.h" 
  15 // For compilers that support precompilation, includes "wx.h". 
  16 #include "wx/wxprec.h" 
  26 #if wxUSE_STREAMS && wxUSE_GIF 
  30 #include "wx/gifdecod.h" 
  33 //--------------------------------------------------------------------------- 
  34 // wxGIFDecoder constructor and destructor 
  35 //--------------------------------------------------------------------------- 
  37 wxGIFDecoder::wxGIFDecoder(wxInputStream 
*s
, bool anim
) 
  53 wxGIFDecoder::~wxGIFDecoder() 
  58 void wxGIFDecoder::Destroy() 
  60     GIFImage 
*pimg
, *paux
; 
  81 //--------------------------------------------------------------------------- 
  82 // Convert this image to a wxImage object 
  83 //--------------------------------------------------------------------------- 
  85 // This function was designed by Vaclav Slavik 
  87 bool wxGIFDecoder::ConvertToImage(wxImage 
*image
) const 
  89     unsigned char *src
, *dst
, *pal
; 
  96     /* create the image */ 
  97     image
->Create(GetWidth(), GetHeight()); 
 104     dst 
= image
->GetData(); 
 105     transparent 
= GetTransparentColour(); 
 107     /* set transparent colour mask */ 
 108     if (transparent 
!= -1) 
 110         for (i 
= 0; i 
< 256; i
++) 
 112             if ((pal
[3 * i 
+ 0] == 255) && 
 113                 (pal
[3 * i 
+ 1] == 0) && 
 114                 (pal
[3 * i 
+ 2] == 255)) 
 116                 pal
[3 * i 
+ 2] = 254; 
 120         pal
[3 * transparent 
+ 0] = 255, 
 121         pal
[3 * transparent 
+ 1] = 0, 
 122         pal
[3 * transparent 
+ 2] = 255; 
 124         image
->SetMaskColour(255, 0, 255); 
 127         image
->SetMask(FALSE
); 
 132         unsigned char r
[256]; 
 133         unsigned char g
[256]; 
 134         unsigned char b
[256]; 
 136         for (i 
= 0; i 
< 256; i
++) 
 143         image
->SetPalette(wxPalette(256, r
, g
, b
)); 
 145 #endif // wxUSE_PALETTE 
 147     /* copy image data */ 
 148     for (i 
= 0; i 
< (GetWidth() * GetHeight()); i
++, src
++) 
 150         *(dst
++) = pal
[3 * (*src
) + 0]; 
 151         *(dst
++) = pal
[3 * (*src
) + 1]; 
 152         *(dst
++) = pal
[3 * (*src
) + 2]; 
 159 //--------------------------------------------------------------------------- 
 161 //--------------------------------------------------------------------------- 
 163 // Get data for current frame 
 165 int wxGIFDecoder::GetFrameIndex() const         { return m_image
; } 
 166 unsigned char* wxGIFDecoder::GetData() const    { return (m_pimage
->p
); } 
 167 unsigned char* wxGIFDecoder::GetPalette() const { return (m_pimage
->pal
); } 
 168 unsigned int wxGIFDecoder::GetWidth() const     { return (m_pimage
->w
); } 
 169 unsigned int wxGIFDecoder::GetHeight() const    { return (m_pimage
->h
); } 
 170 unsigned int wxGIFDecoder::GetTop() const       { return (m_pimage
->top
); } 
 171 unsigned int wxGIFDecoder::GetLeft() const      { return (m_pimage
->left
); } 
 172 int wxGIFDecoder::GetTransparentColour() const  { return (m_pimage
->transparent
); } 
 173 int wxGIFDecoder::GetDisposalMethod() const     { return (m_pimage
->disposal
); } 
 174 long wxGIFDecoder::GetDelay() const             { return (m_pimage
->delay
); } 
 178 unsigned int wxGIFDecoder::GetLogicalScreenWidth() const    { return m_screenw
; } 
 179 unsigned int wxGIFDecoder::GetLogicalScreenHeight() const   { return m_screenh
; } 
 180 int wxGIFDecoder::GetBackgroundColour() const   { return m_background
; } 
 181 int wxGIFDecoder::GetNumberOfFrames() const     { return m_nimages
; } 
 182 bool wxGIFDecoder::IsAnimation() const          { return (m_nimages 
> 1); } 
 185 //--------------------------------------------------------------------------- 
 186 // Functions to move through the animation 
 187 //--------------------------------------------------------------------------- 
 189 bool wxGIFDecoder::GoFirstFrame() 
 199 bool wxGIFDecoder::GoLastFrame() 
 209 bool wxGIFDecoder::GoNextFrame(bool cyclic
) 
 214     if ((m_image 
< m_nimages
) || (cyclic
)) 
 216         m_pimage 
= m_pimage
->next
; 
 231 bool wxGIFDecoder::GoPrevFrame(bool cyclic
) 
 236     if ((m_image 
> 1) || (cyclic
)) 
 238         m_pimage 
= m_pimage
->prev
; 
 253 bool wxGIFDecoder::GoFrame(int which
) 
 260     if ((which 
>= 1) && (which 
<= m_nimages
)) 
 264         for (i 
= 1; i 
< which
; i
++) 
 265             m_pimage 
= m_pimage
->next
; 
 274 //--------------------------------------------------------------------------- 
 275 // GIF reading and decoding 
 276 //--------------------------------------------------------------------------- 
 279 //  Reads the next code from the file stream, with size 'bits' 
 281 int wxGIFDecoder::getcode(int bits
, int ab_fin
) 
 283     unsigned int mask
;          /* bit mask */ 
 284     unsigned int code
;          /* code (result) */ 
 287     /* get remaining bits from last byte read */ 
 288     mask 
= (1 << bits
) - 1; 
 289     code 
= (m_lastbyte 
>> (8 - m_restbits
)) & mask
; 
 291     /* keep reading new bytes while needed */ 
 292     while (bits 
> m_restbits
) 
 294         /* if no bytes left in this block, read the next block */ 
 297             m_restbyte 
= (unsigned char)m_f
->GetC(); 
 299             /* Some encoders are a bit broken: instead of issuing 
 300              * an end-of-image symbol (ab_fin) they come up with 
 301              * a zero-length subblock!! We catch this here so 
 302              * that the decoder sees an ab_fin code. 
 311             m_f
->Read((void *) m_buffer
, m_restbyte
); 
 315         /* read next byte and isolate the bits we need */ 
 316         m_lastbyte 
= (unsigned char) (*m_bufp
++); 
 317         mask       
= (1 << (bits 
- m_restbits
)) - 1; 
 318         code       
= code 
+ ((m_lastbyte 
& mask
) << m_restbits
); 
 321         /* adjust total number of bits extracted from the buffer */ 
 322         m_restbits 
= m_restbits 
+ 8; 
 325     /* find number of bits remaining for next code */ 
 326     m_restbits 
= (m_restbits 
- bits
); 
 333 //  GIF decoding function. The initial code size (aka root size) 
 334 //  is 'bits'. Supports interlaced images (interl == 1). 
 336 int wxGIFDecoder::dgif(GIFImage 
*img
, int interl
, int bits
) 
 338     int *ab_prefix 
= new int[4096]; /* alphabet (prefixes) */ 
 339     int *ab_tail 
= new int[4096];   /* alphabet (tails) */ 
 340     int *stack 
= new int[4096];     /* decompression stack */ 
 341     int ab_clr
;                     /* clear code */ 
 342     int ab_fin
;                     /* end of info code */ 
 343     int ab_bits
;                    /* actual symbol width, in bits */ 
 344     int ab_free
;                    /* first free position in alphabet */ 
 345     int ab_max
;                     /* last possible character in alphabet */ 
 346     int pass
;                       /* pass number in interlaced images */ 
 347     int pos
;                        /* index into decompresion stack */ 
 348     unsigned int x
, y
;              /* position in image buffer */ 
 350     int code
, readcode
, lastcode
, abcabca
; 
 352     /* these won't change */ 
 353     ab_clr 
= (1 << bits
); 
 354     ab_fin 
= (1 << bits
) + 1; 
 356     /* these will change through the decompression proccess */ 
 358     ab_free  
= (1 << bits
) + 2; 
 359     ab_max   
= (1 << ab_bits
) - 1; 
 365     /* reset decoder vars */ 
 373         readcode 
= code 
= getcode(ab_bits
, ab_fin
); 
 376         if (code 
== ab_fin
) break; 
 378         /* reset alphabet? */ 
 381             /* reset main variables */ 
 383             ab_free  
= (1 << bits
) + 2; 
 384             ab_max   
= (1 << ab_bits
) - 1; 
 388             /* skip to next code */ 
 392         /* unknown code: special case (like in ABCABCA) */ 
 395             code 
= lastcode
;            /* take last string */ 
 396             stack
[pos
++] = abcabca
;     /* add first character */ 
 399         /* build the string for this code in the stack */ 
 400         while (code 
> ab_clr
) 
 402             stack
[pos
++] = ab_tail
[code
]; 
 403             code         
= ab_prefix
[code
]; 
 405         stack
[pos
] = code
;              /* push last code into the stack */ 
 406         abcabca    
= code
;              /* save for special case */ 
 408         /* make new entry in alphabet (only if NOT just cleared) */ 
 411             ab_prefix
[ab_free
] = lastcode
; 
 412             ab_tail
[ab_free
]   = code
; 
 415             if ((ab_free 
> ab_max
) && (ab_bits 
< 12)) 
 418                 ab_max 
= (1 << ab_bits
) - 1; 
 422         /* dump stack data to the buffer */ 
 425             (img
->p
)[x 
+ (y 
* (img
->w
))] = (char)stack
[pos
--]; 
 433                     /* support for interlaced images */ 
 436                         case 1: y 
+= 8; break; 
 437                         case 2: y 
+= 8; break; 
 438                         case 3: y 
+= 4; break; 
 439                         case 4: y 
+= 2; break; 
 445                             case 2: y 
= 4; break; 
 446                             case 3: y 
= 2; break; 
 447                             case 4: y 
= 1; break; 
 462     while (code 
!= ab_fin
); 
 464     delete [] ab_prefix 
; 
 473 //  Returns TRUE if the file looks like a valid GIF, FALSE otherwise. 
 475 bool wxGIFDecoder::CanRead() 
 477     unsigned char buf
[3]; 
 480     m_f
->SeekI(-3, wxFromCurrent
); 
 482     return (memcmp(buf
, "GIF", 3) == 0); 
 487 //  Reads and decodes one or more GIF images, depending on whether 
 488 //  animated GIF support is enabled. Can read GIFs with any bit 
 489 //  size (color depth), but the output images are always expanded 
 490 //  to 8 bits per pixel. Also, the image palettes always contain 
 491 //  256 colors, although some of them may be unused. Returns wxGIF_OK 
 492 //  (== 0) on success, or an error code if something fails (see 
 493 //  header file for details) 
 495 int wxGIFDecoder::ReadGIF() 
 497     int           ncolors
, bits
, interl
, transparent
, disposal
, i
; 
 500     unsigned char type 
= 0; 
 501     unsigned char pal
[768]; 
 502     unsigned char buf
[16]; 
 504         GIFImage      
*pimg
, *pprev
; 
 506     /* check GIF signature */ 
 508         return wxGIF_INVFORMAT
; 
 510     /* check for animated GIF support (ver. >= 89a) */ 
 513     if (memcmp(buf 
+ 3, "89a", 3) < 0) 
 516     /* read logical screen descriptor block (LSDB) */ 
 518     m_screenw 
= buf
[0] + 256 * buf
[1]; 
 519     m_screenh 
= buf
[2] + 256 * buf
[3]; 
 521     /* load global color map if available */ 
 522     if ((buf
[4] & 0x80) == 0x80) 
 524         m_background 
= buf
[5]; 
 526         ncolors 
= 2 << (buf
[4] & 0x07); 
 527         m_f
->Read(pal
, 3 * ncolors
); 
 530     /* transparent colour, disposal method and delay default to unused */ 
 544         type 
= (unsigned char)m_f
->GetC(); 
 552         /* extension block? */ 
 555             if (((unsigned char)m_f
->GetC()) == 0xF9) 
 556             /* graphics control extension, parse it */ 
 560                 /* read delay and convert from 1/100 of a second to ms */ 
 561                 delay 
= 10 * (buf
[2] + 256 * buf
[3]); 
 563                 /* read transparent colour index, if used */ 
 565                     transparent 
= buf
[4]; 
 567                 /* read disposal method */ 
 568                 disposal 
= (buf
[1] & 0x1C) - 1; 
 571             /* other extension, skip */ 
 573                 while ((i 
= (unsigned char)m_f
->GetC()) != 0) 
 575                     m_f
->SeekI(i
, wxFromCurrent
); 
 580         /* image descriptor block? */ 
 583             /* allocate memory for IMAGEN struct */ 
 584             pimg 
= (*ppimg
) = new GIFImage(); 
 592             /* fill in the data */ 
 594             pimg
->left 
= buf
[0] + 256 * buf
[1]; 
 595             pimg
->top 
= buf
[2] + 256 * buf
[3]; 
 597             pimg->left = buf[4] + 256 * buf[5]; 
 598             pimg->top = buf[4] + 256 * buf[5]; 
 600             pimg
->w 
= buf
[4] + 256 * buf
[5]; 
 601             pimg
->h 
= buf
[6] + 256 * buf
[7]; 
 602             interl 
= ((buf
[8] & 0x40)? 1 : 0); 
 603             size 
= pimg
->w 
* pimg
->h
; 
 605             pimg
->transparent 
= transparent
; 
 606             pimg
->disposal 
= disposal
; 
 613             /* allocate memory for image and palette */ 
 614             pimg
->p   
= (unsigned char *) malloc((size_t)size
); 
 615             pimg
->pal 
= (unsigned char *) malloc(768); 
 617             if ((!pimg
->p
) || (!pimg
->pal
)) 
 623             /* load local color map if available, else use global map */ 
 624             if ((buf
[8] & 0x80) == 0x80) 
 626                 ncolors 
= 2 << (buf
[8] & 0x07); 
 627                 m_f
->Read(pimg
->pal
, 3 * ncolors
); 
 630                 memcpy(pimg
->pal
, pal
, 768); 
 632             /* get initial code size from first byte in raster data */ 
 633             bits 
= (unsigned char)m_f
->GetC(); 
 636             dgif(pimg
, interl
, bits
); 
 639             /* if this is not an animated GIF, exit after first image */ 
 645     /* setup image pointers */ 
 653     /* try to read to the end of the stream */ 
 657             return wxGIF_TRUNCATED
; 
 659         type 
= (unsigned char)m_f
->GetC(); 
 667             while ((i 
= (unsigned char)m_f
->GetC()) != 0) 
 669                 m_f
->SeekI(i
, wxFromCurrent
); 
 672         else if (type 
== 0x2C) 
 674             /* image descriptor block */ 
 677             /* local color map */ 
 678             if ((buf
[8] & 0x80) == 0x80) 
 680                 ncolors 
= 2 << (buf
[8] & 0x07); 
 681                 m_f
->SeekI(3 * ncolors
, wxFromCurrent
); 
 684             /* initial code size */ 
 688             while ((i 
= (unsigned char)m_f
->GetC()) != 0) 
 690                 m_f
->SeekI(i
, wxFromCurrent
); 
 693         else if ((type 
!= 0x3B) && (type 
!= 00)) /* testing */ 
 695             /* images are OK, but couldn't read to the end of the stream */ 
 696             return wxGIF_TRUNCATED
; 
 703 #endif // wxUSE_STREAMS && wxUSE_GIF