Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.
Articles
(untagged)

Decompress and display an animated GIF on PC and PPC

0.00/5 (No votes)
23 Mar 2008 1  
A class to allow animated GIFs to be displayed, manualy scanning and decompressing.

Introduction

The code provided here will allow animated GIF files to be displayed on Windows within your projects. It does not rely on any library to do this but rather manually scans the GIF to determine its contents, decompresses the image data, and displays it on the window of your choice. By doing this manually, the code has some degree of portability on Win32 systems, allowing the same code to work in both desktop and Pocket PC applications.

Background

Often while coding a project, I have had the need to show some kind of animation, either as a status message to the user or to spice up an attribution dialog. After doing this manually several times using timers and and icons to switch images, I needed a better, more flexible way. The popularity of the GIF image format and its ability to display animations made it a prime candidate to store the animations for my code.

Using the Code

The class that you will interact with is gifDisplayer; it has four methods to interact with the desired GIF file.

//Calls the load from reader then prepairs bitmaps for display,
//a gif must be loaded before it can be displayed. 
BOOL loadGif(CString filePath,
        HDC hDCin,
        BOOL animate,
        HWND hWnd,
        int multiplySizeBy,
        int Xoffset,
        int Yoffset);

BOOL loadGif(unsigned char* bufferIn,
        unsigned long lengthIn,
        HDC hDCin,
        BOOL animate,
        HWND hWnd,
        int multiplySizeBy,
        int Xoffset,
        int Yoffset); 

//An image must be loaded by calling the load method from the gifReader first.
BOOL displayGif();

//clean up.
void unloadGif();

Multiple objects of this class can be used simultaneously. Multiple animations can be drawn to the same window simultaneously. What is not supported are multiple animations being stacked in the same place. If that is done, the transparency will be lost.

To use the class, some static members will need to be set up:

#include "stdafx.h"
#include "gifDisplayer.h"
HANDLE gifDisplayer::hAccessMutex=0;
HANDLE gifDisplayer::hTimingThread=0;
int gifDisplayer::countOfObjects=0;
gifDisplayer* gifDisplayer::gifDisplayerObjectArray[MAX_SURFACES];

This is done from the main .cpp file or is included in a file from there.

You can instantiate objects from the class after that, as normally would be done:

//The parameter is the starting size for the decompressed gif.
//This can be set small but a small size will make decompression take longer
//because the output will have to be grown many times. 
gifDisplayer* gr=new gifDisplayer(100000);
gifDisplayer* gr2=new gifDisplayer(100000);
gifDisplayer* gr3=new gifDisplayer(100000);

It is important to be sure your window has been drawn properly before calling loadGif(...) or displayGif(). This is because in order to allow the GIF to seem transparent on your window, the contents of the window will be copied to form the base over which the GIF will show. To do this, I post a message to the window after the window has been created, allowing a delay for the initial drawing to occur.

switch (message) 
{
    case SETUP_MESSAGE:
    {
          //on a mobile device this can be a bit slow
          //so its a good idea to show a wait cursor.
       SetCursor(LoadCursor (NULL,IDC_WAIT));

          //load and display three gifs.
          gr->loadGif(gifPath,//the path of the file
          surfaceDC,//the device context of the window
          TRUE,//do you want to animate?
          hSurfaceWindow,//the handle of the window
          1,//scale the gif by a factor of 1 (possible values 1,2,3,4...)
          50,//x position
          100);//y position

        gr->displayGif();

        gr2->loadGif(gifPath2,
            surfaceDC,
                 TRUE,
           hSurfaceWindow,1,150,200);

        gr2->displayGif();

        gr3->loadGif(gifPath3,
            surfaceDC,
                 TRUE,
           hSurfaceWindow,2,0,0);

        gr3->displayGif();

        SetCursor(0);
    }
    break;

Remember to call unloadGif() on each object before ending your program.

The gifDisplayer class inherits from a number of other classes I wrote. You only need to include them in your project as shown in the demos. I split the functionality up this way because the scanning, decompressing, and displaying are distinctly different parts that are better suited to being independent. Note that I use MFC CStrings and CObjectArrays in my code. The projects themselves are Win32, and not MFC projects. Some changes need to be made in the project settings and included files to let this work in Win32 code. It would be easy enough to replace these classes with your own if you want to get rid of MFC all together.

Points of Interest

All the code in this project is based on the document "LZWand GIF explained" by Steve Blackstock and on the CompuServe GIF standard. Although these documents do not contain any C++ code, they are an accurate description of what a GIF is.

History

  • Submitted: 24 March 2008.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here