📄 gif.c
字号:
//// $Id: gif.c,v 1.3 2000/09/29 09:05:30 kevin Exp $// // gif.c: Low-level GIF file read/save routines.// // Copyright (C) 2000, BluePoint Software.// Copyright (C) 2000, Wei Yongming//// Current maintainer: Wei Yongming//// Create date: 2000/08/29/*** This library is free software; you can redistribute it and/or** modify it under the terms of the GNU Library General Public** License as published by the Free Software Foundation; either** version 2 of the License, or (at your option) any later version.**** This library is distributed in the hope that it will be useful,** but WITHOUT ANY WARRANTY; without even the implied warranty of** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU** Library General Public License for more details.**** You should have received a copy of the GNU Library General Public** License along with this library; if not, write to the Free** Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,** MA 02111-1307, USA*//* Code for GIF decoding has been adapted from XPaint: *//* +-------------------------------------------------------------------+ *//* | Copyright 1990, 1991, 1993 David Koblas. | *//* | Copyright 1996 Torsten Martinsen. | *//* | Permission to use, copy, modify, and distribute this software | *//* | and its documentation for any purpose and without fee is hereby | *//* | granted, provided that the above copyright notice appear in all | *//* | copies and that both that copyright notice and this permission | *//* | notice appear in supporting documentation. This software is | *//* | provided "as is" without express or implied warranty. | *//* +-------------------------------------------------------------------+ *//* Portions Copyright (C) 1999 Sam Lantinga*//* Adapted for use in SDL by Sam Lantinga -- 7/20/1998 *//* Adapted for use in MiniGUI by Wei Yongming -- 8/30/2000 */#include <stdio.h>#include <stdlib.h>#include <string.h>#include "common.h"#include "gdi.h"#include "readbmp.h"/* * GIF decoding routine */#define MAXCOLORMAPSIZE 256#define MAX_LWZ_BITS 12#define INTERLACE 0x40#define LOCALCOLORMAP 0x80#define CM_RED 0#define CM_GREEN 1#define CM_BLUE 2#define BitSet(byte, bit) (((byte) & (bit)) == (bit))#define ReadOK(file,buffer,len) fread(buffer, len, 1, file)#define LM_to_uint(a,b) (((b)<<8)|(a))struct { unsigned int Width; unsigned int Height; unsigned char ColorMap[3][MAXCOLORMAPSIZE]; unsigned int BitPixel; unsigned int ColorResolution; unsigned int Background; unsigned int AspectRatio; int GrayScale;} GifScreen;static struct { int transparent; int delayTime; int inputFlag; int disposal;} Gif89;static int ReadColorMap(FILE* src, int number, unsigned char buffer[3][MAXCOLORMAPSIZE], int *flag);static int DoExtension(FILE* src, int label);static int GetDataBlock(FILE* src, unsigned char *buf);static int GetCode(FILE* src, int code_size, int flag);static int LWZReadByte(FILE* src, int flag, int input_code_size);static int ReadImage(FILE* src, MYBITMAP* bmp, RGB* pal, int len, int height, int, unsigned char cmap[3][MAXCOLORMAPSIZE], int gray, int interlace, int ignore);int load_gif (FILE *src, MYBITMAP* bmp, RGB* pal){ unsigned char buf[16]; unsigned char c; unsigned char localColorMap[3][MAXCOLORMAPSIZE]; int grayScale; int useGlobalColormap; int bitPixel; int imageCount = 0; char version[4]; int imageNumber = 1; int ok = 0; fseek(src, 0L, 0); if (!ReadOK (src, buf, 6)) return ERR_BMP_IMAGE_TYPE; /* not gif image*/ if (strncmp((char *) buf, "GIF", 3) != 0) return ERR_BMP_IMAGE_TYPE; strncpy (version, (char *) buf + 3, 3); version [3] = '\0'; if (strcmp(version, "87a") != 0 && strcmp(version, "89a") != 0) {#ifdef _DEBUG fprintf (stderr, "load_gif: GIF version number not 87a or 89a\n");#endif return ERR_BMP_NOT_SUPPORTED; /* image loading error*/ } Gif89.transparent = -1; Gif89.delayTime = -1; Gif89.inputFlag = -1; Gif89.disposal = 0; if (!ReadOK (src, buf, 7)) {#ifdef _DEBUG fprintf (stderr, "load_gif: bad screen descriptor\n");#endif return ERR_BMP_LOAD; /* image loading error*/ } GifScreen.Width = LM_to_uint (buf[0], buf[1]); GifScreen.Height = LM_to_uint (buf[2], buf[3]); GifScreen.BitPixel = 2 << (buf[4] & 0x07); GifScreen.ColorResolution = (((buf[4] & 0x70) >> 3) + 1); GifScreen.Background = buf[5]; GifScreen.AspectRatio = buf[6]; if (BitSet(buf[4], LOCALCOLORMAP)) { /* Global Colormap */ if (ReadColorMap(src, GifScreen.BitPixel, GifScreen.ColorMap, &GifScreen.GrayScale)) {#ifdef _DEBUG fprintf (stderr, "load_gif: bad global colormap\n");#endif return ERR_BMP_LOAD; /* image loading error*/ } } do { if (!ReadOK (src, &c, 1)) {#ifdef _DEBUG fprintf (stderr, "load_gif: EOF on image data\n");#endif goto done; } if (c == ';') { /* GIF terminator */ if (imageCount < imageNumber) {#ifdef _DEBUG fprintf (stderr, "load_gif: no image %d of %d\n", imageNumber,imageCount);#endif goto done; } } if (c == '!') { /* Extension */ if (!ReadOK (src, &c, 1)) {#ifdef _DEBUG fprintf (stderr, "load_gif: EOF on extension function code\n");#endif goto done; } DoExtension (src, c); continue; } if (c != ',') { /* Not a valid start character */ continue; } ++imageCount; if (!ReadOK (src, buf, 9)) {#ifdef _DEBUG fprintf (stderr, "load_gif: bad image size\n");#endif goto done; } useGlobalColormap = !BitSet (buf[8], LOCALCOLORMAP); bitPixel = 1 << ((buf[8] & 0x07) + 1); if (!useGlobalColormap) { if (ReadColorMap(src, bitPixel, localColorMap, &grayScale)) {#ifdef _DEBUG fprintf (stderr, "load_gif: bad local colormap\n");#endif goto done; } ok = ReadImage (src, bmp, pal, LM_to_uint(buf[4], buf[5]), LM_to_uint(buf[6], buf[7]), bitPixel, localColorMap, grayScale, BitSet(buf[8], INTERLACE), imageCount != imageNumber); } else { ok = ReadImage(src, bmp, pal, LM_to_uint(buf[4], buf[5]), LM_to_uint(buf[6], buf[7]), GifScreen.BitPixel, GifScreen.ColorMap, GifScreen.GrayScale, BitSet(buf[8], INTERLACE), imageCount != imageNumber); } } while (ok == 0); if ( Gif89.transparent > 0 ) { bmp->transparent = Gif89.transparent; /* TODO: handle transparent bitmap */ pal [Gif89.transparent].r = 0; pal [Gif89.transparent].g = 0; pal [Gif89.transparent].b = 0; } if(ok) return 0; /* image load ok*/done: return ERR_BMP_LOAD; /* image load error*/}static intReadColorMap (FILE *src, int number, unsigned char buffer[3][MAXCOLORMAPSIZE], int *gray){ int i; unsigned char rgb[3]; int flag; flag = TRUE; for (i = 0; i < number; ++i) { if (!ReadOK(src, rgb, sizeof(rgb))) return 1; buffer[CM_RED][i] = rgb[0]; buffer[CM_GREEN][i] = rgb[1]; buffer[CM_BLUE][i] = rgb[2]; flag &= (rgb[0] == rgb[1] && rgb[1] == rgb[2]); }#if 0 if (flag) *gray = (number == 2) ? PBM_TYPE : PGM_TYPE; else *gray = PPM_TYPE;#else *gray = 0;#endif return FALSE;}static intDoExtension(FILE *src, int label){ static unsigned char buf[256]; switch (label) { case 0x01: /* Plain Text Extension */ break; case 0xff: /* Application Extension */ break; case 0xfe: /* Comment Extension */ while (GetDataBlock(src, (unsigned char *) buf) != 0); return FALSE; case 0xf9: /* Graphic Control Extension */ GetDataBlock(src, (unsigned char *) buf); Gif89.disposal = (buf[0] >> 2) & 0x7;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -