⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 decodegif.cpp

📁 hl2 source code. Do not use it illegal.
💻 CPP
字号:
/*----------------------------------------------------------------------
Copyright (c) 1998,1999 Gipsysoft. All Rights Reserved.
Please see the file "licence.txt" for licencing details.
File:	DecodeMNG.cpp
Owner:	russf@gipsysoft.com
Purpose:	Decode an GIF file into a frame array.
					The only exported function is:
					bool DecodeGIF( CDataSourceABC &ds, CFrameArray &arrFrames, SIZE &size )

----------------------------------------------------------------------*/
#include "stdafx.h"
#include "DataSourceABC.h"
#include <ImgLib.h>
#include "ImgLibInt.h"
#include "Config.h"
#include "Gif.h"

#ifdef IMGLIB_GIF

typedef LONG RGBAPIXEL;

static short GIF_GetDataBlock( CDataSourceABC &ds, LPBYTE buf, short &bZeroDataBlock );
static bool GIF_ReadImage( const CDib *pDibPrevious, CDataSourceABC &ds, short len, short height, LPBYTE cmap, short interlace, CDib *& pDib, short bitPixel, short bZeroDataBlock );
static short GIF_GetCode(CDataSourceABC &ds, short code_size, short flag, short &bZeroDataBlock );
static short GIF_LZWReadByte( CDataSourceABC &ds, short flag, short input_code_size, short bZeroDataBlock );

#define RGBA_BLUE   0
#define RGBA_GREEN  1
#define RGBA_RED    2
#define RGBA_ALPHA  3


inline void SetRGBAPixel
    ( RGBAPIXEL * pPixel,
      BYTE r,
      BYTE g,
      BYTE b,
      BYTE a
    )
{
  ((BYTE *)pPixel)[RGBA_RED] = r;
  ((BYTE *)pPixel)[RGBA_GREEN] = g;
  ((BYTE *)pPixel)[RGBA_BLUE] = b;
  ((BYTE *)pPixel)[RGBA_ALPHA] = a;
}


// Read color map into buffer with number entries
static BOOL GIF_ReadColorMap( CDataSourceABC &ds , short number, LPBYTE buffer)
{
	for (int i = 0; i < number; i++)
	{
		buffer[GIF_CM_RED * GIF_MAXCOLORMAPSIZE + i] = ds.ReadByte();
		buffer[GIF_CM_GREEN * GIF_MAXCOLORMAPSIZE + i] = ds.ReadByte();
		buffer[GIF_CM_BLUE * GIF_MAXCOLORMAPSIZE + i] = ds.ReadByte();
	}

	return TRUE;
}

static BOOL GIF_DoExtension(CDataSourceABC &ds, BYTE& label, GIF89& Gif89, short &bZeroDataBlock )
{
	static BYTE buf[256];
	switch (label)
	{
	case 0X01:  // Plain Text Extension
		break;

	case 0XFF:  // Application Extension
		break;

	case 0XFE:  // Comment Extension
		while ( GIF_GetDataBlock( ds, &buf[0], bZeroDataBlock ) != 0)
		{
			TRACE( "GIF Comment: %s\n", buf);
		}
		return TRUE;

	case 0XF9:  // Graphic Control Extension
		GIF_GetDataBlock( ds, &buf[0], bZeroDataBlock );
		Gif89.disposal = (WORD)((buf[0] >> 2) & 0X07);
		Gif89.inputFlag = (WORD)((buf[0] >> 1) & 0X01);
		Gif89.delayTime = (WORD)(GIF_LM_to_unit(buf[1], buf[2]));
		if ((buf[0] & 0X01) != 0)
			Gif89.transparent = buf[3];
		while( GIF_GetDataBlock( ds, &buf[0], bZeroDataBlock ) != 0)
			;
		return TRUE;

	default:
		TRACE( "Unknown (0x%02x)\n", label );
		break;
	}
	while( GIF_GetDataBlock( ds, &buf[0], bZeroDataBlock ) != 0 )
		;
	return TRUE;
}


bool DecodeGIF( CDataSourceABC &ds, CFrameArray &arrFrames, SIZE &size )
{
	GIF89 Gif89 = {-1, -1, -1, 0};
	BYTE buf[16], c;
	short useGlobalColorMap;
	short bitPixel;
	short imageCount = 0;
	char version[4];
	short i;

	short bZeroDataBlock = FALSE;

	BYTE localColorMap[3][GIF_MAXCOLORMAPSIZE];
	GIFSCREEN GifScreen;

	TRACE(_T("Decoding GIF.\n"));

	for (i = 0; i < 6; i++)
		buf[i] = ds.ReadByte();

	if (strncmp((const char*)buf, "GIF", 3) != 0)
	{
		return false;
	}
	strncpy(version, (const char*)buf + 3, 3);
	version[3] = '\0';
	if( ( strcmp( version, "87a" ) != 0 ) && ( strcmp( version, "89a" ) != 0 ) )
	{
		return false;
	}

	for (i = 0; i < 7; i++)
		buf[i] = ds.ReadByte();
	size.cx = GifScreen.Width = (WORD)GIF_LM_to_unit(buf[0], buf[1]);
	size.cy = GifScreen.Height = (WORD)GIF_LM_to_unit(buf[2], buf[3]);
	GifScreen.BitPixel = (WORD)(2 << (buf[4] & 0X07));
	GifScreen.ColorResolution = (WORD)(((buf[4] & 0X70) >> 3) + 1);
	GifScreen.Background = buf[5];
	GifScreen.AspectRatio = buf[6];

	// Read the color map data
	if (GIF_BitSet(buf[4], GIF_LOCALCOLORMAP))
		GIF_ReadColorMap(ds, GifScreen.BitPixel, &GifScreen.ColorMap[0][0]);

	if (GifScreen.AspectRatio != 0 && GifScreen.AspectRatio != 49)
	{
		double r;
		r = ((double)GifScreen.AspectRatio + 15.0) / 64.0;
		TRACE(_T("Warning - non-square pixels.\n"));
	}

	CDib * pDib = NULL, *pDibPrevious = NULL ;
	for(; ;)
	{
		c = ds.ReadByte();
		if (c == ';')  // GIF terminator
		{
			TRACE(_T("Decoding finished.\n"));
			return true;
		}

		if (c == '!')  // Extension
		{
			c = ds.ReadByte();
			GIF_DoExtension( ds, c, Gif89, bZeroDataBlock );
			continue;
		}

		if (c != ',')
		{
			TRACE(_T("Bogus character 0X%02x, ignoring\n"), (int)c);
			continue;
		}

		imageCount++;

		GIFIMAGEDESCRIPTOR gi;
		ds.ReadBytes( (LPBYTE)&gi, sizeof( gi ) );
		Gif89.nLeft = gi.nLeft;
		Gif89.nTop = gi.nTop;
		Gif89.nWidth = gi.nWidth;
		Gif89.nHeight = gi.nHeight;

		useGlobalColorMap = !GIF_BitSet(gi.cPacked, GIF_LOCALCOLORMAP);
		bitPixel = (short)(1 << ((gi.cPacked & 0X07) + 1));

		bool bGotImage = false;
		if (!useGlobalColorMap)
		{
			if (GIF_ReadColorMap(ds, bitPixel, &localColorMap[0][0]))
			{
				return false;
			}

			bGotImage = GIF_ReadImage(pDibPrevious, ds, Gif89.nWidth, Gif89.nHeight, &localColorMap[0][0], (short)GIF_BitSet(gi.cPacked, GIF_INTERLACE), pDib, GifScreen.BitPixel, bZeroDataBlock );
		}
		else
		{
			bGotImage = GIF_ReadImage(pDibPrevious, ds, Gif89.nWidth, Gif89.nHeight, &GifScreen.ColorMap[0][0], (short)GIF_BitSet(gi.cPacked, GIF_INTERLACE), pDib, GifScreen.BitPixel, bZeroDataBlock );
		}

	
		if( bGotImage )
		{
			//
			//	If the images are different sizes then we need to create the next image.
			if( Gif89.nWidth != GifScreen.Width || Gif89.nHeight != GifScreen.Height )
			{
				CDib *pdibNew = pDibPrevious->CreateCopy();
				pdibNew->AddImage( Gif89.nLeft, Gif89.nTop, pDib );
				delete pDib;
				pDib = pdibNew;
			}
			int nDelay = 0;
			if( Gif89.delayTime != -1 )
			{
				nDelay = Gif89.delayTime * 10;
			}
			CFrame *pFrame = new CFrame( pDib, nDelay );
			arrFrames.Add( pFrame );
			pDibPrevious = pDib;
		}
	}

 	return false;
}


static short GIF_GetDataBlock( CDataSourceABC &ds, LPBYTE buf, short &bZeroDataBlock )
{
	BYTE count, i;
	count = ds.ReadByte();
	bZeroDataBlock = (count == 0);
	if (count != 0)
	{
		for (i = 0; i < count; i++)
			buf[i] = ds.ReadByte();
	}
	return count;
}



static bool GIF_ReadImage( const CDib *pDibPrevious, CDataSourceABC &ds, short len, short height, LPBYTE cmap, short interlace, CDib *& pDib, short bitPixel, short bZeroDataBlock )
{
	UNREFERENCED_PARAMETER( bitPixel );
	UNREFERENCED_PARAMETER( pDibPrevious );	

	BYTE c;
	short v;
	short xpos = 0, ypos = 0, pass = 0;

	c = ds.ReadByte();
	if( GIF_LZWReadByte( ds, TRUE, c, bZeroDataBlock ) < 0 )
	{
		return false;
	}
	pDib = new CDib( len, height, 32 );

	CLineArray arr;
	pDib->GetLineArray( arr );

	while( (v = GIF_LZWReadByte( ds, FALSE, c, bZeroDataBlock ) ) >= 0 )
	{
		// Look up pixel in color map and store it away
		RGBAPIXEL * pPixel = (RGBAPIXEL *)arr[ ypos];

		SetRGBAPixel( pPixel + xpos, cmap[GIF_CM_RED * GIF_MAXCOLORMAPSIZE + v], cmap[GIF_CM_GREEN * GIF_MAXCOLORMAPSIZE + v], cmap[GIF_CM_BLUE * GIF_MAXCOLORMAPSIZE + v], 0xFF );

		xpos++;
		if (xpos == len)
		{
			xpos = 0;
			if (interlace)
			{
				// handle interlaced row order
				switch (pass)
				{
				case 0:
				case 1:
					ypos += 8;
					break;
				case 2:
					ypos += 4;
					break;
				case 3:
					ypos += 2;
					break;
				}
				if (ypos >= height)
				{
					pass++;
					switch (pass)
					{
					case 1:
						ypos = 4;
						break;
					case 2:
						ypos = 2;
						break;
					case 3:
						ypos = 1;
						break;
					default:
						goto fini;
					}
				}
			}
			else  // no interlace, read in order
			{
				ypos++;
			}
		}
		if (ypos >= height)
			break;
	}

fini:
	if (GIF_LZWReadByte(ds, FALSE, c, bZeroDataBlock) >= 0)
	{
		TRACE(_T("Too much input data, ignoring extra...\n"));
	}
	return TRUE;
}

static short GIF_GetCode(CDataSourceABC &ds, short code_size, short flag, short &bZeroDataBlock)
{
	static BYTE buf[280];
	static short curbit;  // current bit number in buf[]
	static short lastbit;  // number of bits in buf[]
	static BOOL done;  // hit file EOF
	static short last_byte;  // last valid byte # in buffer
	short i, j, ret;
	BYTE count;

	if (flag)  // prepare to start reading
	{
		curbit = 0;
		lastbit = 0;
		done = FALSE;
		return 0;
	}
	if ((curbit + code_size) >= lastbit)  // need more bits?
	{
		if (done)
		{
			if ((curbit) >= lastbit)
			{
				TRACE(_T("Run off the end of my bits\n"));
				return -99;
			}
			return -1;
		}
		buf[0] = buf[last_byte - 2];
		buf[1] = buf[last_byte - 1];
		if ((count = (BYTE)GIF_GetDataBlock(ds, &buf[2], bZeroDataBlock)) == 0 )
			done = TRUE;
		last_byte = (short)(2 + count);  // update count of valid bytes
		curbit = (short)((curbit - lastbit) + 16);  // // account for slide
		lastbit = (short)((2 + count) * 8);
	}
	ret = 0;  // pick out bits one at a time
	for (i = curbit, j = 0; j < code_size; i++, j++)
		ret |= ((buf[i / 8] & (1 << (i % 8))) != 0) << j;
	curbit = (short)(curbit + code_size);  // update bit pointer
	return ret;
}

static short GIF_LZWReadByte(CDataSourceABC &ds, short flag, short input_code_size, short bZeroDataBlock )
{
	static short fresh = FALSE;
	short code, incode;
	static short code_size, set_code_size;
	static short max_code, max_code_size;
	static short firstcode, oldcode;
	static short clear_code, end_code;
	static WORD next[1 << GIF_MAX_LZW_BITS];
	static BYTE vals[1 << GIF_MAX_LZW_BITS];
	static BYTE stack[1 << (GIF_MAX_LZW_BITS + 1)], *sp;
	register short i;

	if (flag)  // initialization call
	{
		set_code_size = input_code_size;
		code_size = (short)(set_code_size + 1);
		clear_code = (short)(1 << set_code_size);
		end_code = (short)(clear_code + 1);
		max_code = (short)(clear_code + 2);
		max_code_size = (short)(2 * clear_code);
		GIF_GetCode(ds, 0, TRUE, bZeroDataBlock);
		fresh = TRUE;  // note next is first decode call
		// single pixel strings
		for (i = 0; i < clear_code; i++)
		{
			next[i] = 0;
			vals[i] = (BYTE)i;
		}  // clear the rest of the dictionary, just in case
		for (; i < (1 << GIF_MAX_LZW_BITS); i++)
			next[i] = vals[i] = 0;
		sp = stack;
		return 0;
	}
	else if (fresh)  // first decode call
	{
		fresh = FALSE;
		do
		{
			firstcode = oldcode = GIF_GetCode(ds, code_size, FALSE, bZeroDataBlock);
		} while (firstcode == clear_code);
		return firstcode;
	}
	if (sp > stack)  // still working through a previously decoded string
		return *--sp;
	while ((code = GIF_GetCode(ds, code_size, FALSE, bZeroDataBlock)) >= 0)
	{
		if (code == clear_code)  // clear out the dictionary
		{
			for (i = 0; i < clear_code; i++)
			{
				next[i] = 0;
				vals[i] = (BYTE)i;
			}
			for (; i < (1 << GIF_MAX_LZW_BITS); i++)
				next[i] = vals[i] = 0;
			code_size = (short)(set_code_size + 1);
			max_code_size = (short)(2 * clear_code);
			max_code = (short)(clear_code + 2);
			sp = stack;
			firstcode = oldcode = GIF_GetCode(ds, code_size, FALSE, bZeroDataBlock);
			return firstcode;
		}
		else if (code == end_code)  // end of image
		{
			short count;
			BYTE buf[260];
			if (bZeroDataBlock)
				return -2;

			while ((count = GIF_GetDataBlock(ds, &buf[0], bZeroDataBlock)) > 0)
				;  // discard input data
			if (count != 0)
				TRACE(_T("Missing EOD in data stream (common occurence)\n"));
			return -2;
		}
		incode = code;
		if (code >= max_code)
		{
			// defensive special case for preview of about to be inserted code
			*sp++ = (BYTE)firstcode;
			code = oldcode;
		}
		while (code >= clear_code)
		{
			// Run through dict pushing
			*sp++ = vals[code];  // entire string on the stack
			ASSERT(code != next[code]);
			code = next[code];
		}
		firstcode = vals[code];  // first pixel in dict string
		*sp++ = (BYTE)firstcode;
		if ((code = max_code) < (1 << GIF_MAX_LZW_BITS))  // add to dictionary
		{
			next[code] = oldcode;
			vals[code] = (BYTE)firstcode;
			max_code++;
			if ((max_code >= max_code_size) &&  // increase code size
				(max_code_size < (1 << GIF_MAX_LZW_BITS)))
			{
				max_code_size *= 2;
				code_size++;
			}
		}
		oldcode = incode;
		if (sp > stack)
			return *--sp;
	}  // while GIF_GetCode
	return code;
}


#endif	//	IMGLIB_GIF

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -