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

📄 bniextract.c

📁 打魔兽战网的都知道他是什么
💻 C
字号:
/*    Copyright (C) 2000  Marco Ziech (mmz@gmx.net)    Copyright (C) 2000  Ross Combs (rocombs@cs.nmsu.edu)    This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published by    the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.    This program 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 General Public License for more details.    You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA*/#include "common/setup_before.h"#ifdef HAVE_STDDEF_H# include <stddef.h>#else# ifndef NULL#  define NULL ((void *)0)# endif#endif#include <stdio.h>#ifdef STDC_HEADERS# include <stdlib.h>#else# ifdef HAVE_MALLOC_H#  include <malloc.h># endif#endif#include "compat/exitstatus.h"#ifdef HAVE_STRING_H# include <string.h>#else# ifdef HAVE_STRINGS_H#  include <strings.h># endif#endif#include <errno.h>#include "compat/strerror.h"#ifdef HAVE_SYS_TYPES_H# include <sys/types.h>#endif#ifdef HAVE_SYS_STAT_H# include <sys/stat.h>#endif#ifdef HAVE_FCNTL_H# include <fcntl.h>#else# ifdef HAVE_SYS_FILE_H#  include <sys/file.h># endif#endif#ifdef HAVE_UNISTD_H# include <unistd.h>#endif#include "compat/statmacros.h"#include "compat/mkdir.h"#include "common/version.h"#include "fileio.h"#include "tga.h"#include "bni.h"#include "common/setup_after.h"/* extract a portion of an image creating a new image */static t_tgaimg * area2img(t_tgaimg *src, int x, int y, int width, int height, t_tgaimgtype type) {	t_tgaimg *dst;	int pixelsize;	unsigned char *datap;	unsigned char *destp;	int i;	if (src == NULL) return NULL;	if ((x+width)>src->width) return NULL;	if ((y+height)>src->height) return NULL;	pixelsize = getpixelsize(src);	if (pixelsize == 0) return NULL;		dst = new_tgaimg(width,height,src->bpp,type);	dst->data = malloc(width*height*pixelsize);		datap = src->data;	datap += y*src->width*pixelsize;	destp = dst->data;	for (i = 0; i < height; i++) {		datap += x*pixelsize;		memcpy(destp,datap,width*pixelsize);		destp += width*pixelsize;		datap += (src->width-x)*pixelsize;	}	return dst;}static void usage(char const * progname){    fprintf(stderr,	    "usage: %s [<options>] [--] <BNI file> <output directory>\n"            "    -h, --help, --usage  show this information and exit\n"            "    -v, --version        print version number and exit\n",progname);    exit(STATUS_FAILURE);}extern int main(int argc, char * argv[]){    char const * outdir=NULL;    char const * bnifile=NULL;    FILE *       fbni;    struct stat  s;    int          a;    int          forcefile=0;    char         dash[]="-"; /* unique address used as flag */        if (argc<1 || !argv || !argv[0])    {	fprintf(stderr,"bad arguments\n");	return STATUS_FAILURE;    }        for (a=1; a<argc; a++)        if (forcefile && !bnifile)            bnifile = argv[a];        else if (strcmp(argv[a],"-")==0 && !bnifile)            bnifile = dash;        else if (argv[a][0]!='-' && !bnifile)            bnifile = argv[a];        else if (forcefile && !outdir)            outdir = argv[a];        else if (strcmp(argv[a],"-")==0 && !outdir)            outdir = dash;        else if (argv[a][0]!='-' && !outdir)            outdir = argv[a];        else if (forcefile || argv[a][0]!='-' || strcmp(argv[a],"-")==0)        {            fprintf(stderr,"%s: extra file argument \"%s\"\n",argv[0],argv[a]);            usage(argv[0]);        }        else if (strcmp(argv[a],"--")==0)            forcefile = 1;        else if (strcmp(argv[a],"-v")==0 || strcmp(argv[a],"--version")==0)        {            printf("version "PVPGN_VERSION"\n");            return STATUS_SUCCESS;        }        else if (strcmp(argv[a],"-h")==0 || strcmp(argv[a],"--help")==0 || strcmp(argv[a],"--usage")==0)            usage(argv[0]);        else        {            fprintf(stderr,"%s: unknown option \"%s\"\n",argv[0],argv[a]);            usage(argv[0]);        }        if (!bnifile)    {	fprintf(stderr,"%s: BNI file not specified\n",argv[0]);	usage(argv[0]);    }    if (!outdir)    {	fprintf(stderr,"%s: output directory not specified\n",argv[0]);	usage(argv[0]);    }        if (bnifile==dash)        fbni = stdin;    else	if (!(fbni = fopen(bnifile,"r")))	{	    fprintf(stderr,"%s: could not open BNI file \"%s\" for reading (fopen: %s)\n",argv[0],bnifile,pstrerror(errno));	    return STATUS_FAILURE;	}        if (outdir==dash)    {	fprintf(stderr,"%s: can not write directory to <stdout>\n",argv[0]);	if (bnifile!=dash && fclose(fbni)<0)	    fprintf(stderr,"%s: could not close BNI file \"%s\" after reading (fclose: %s)\n",argv[0],bnifile,pstrerror(errno));	return STATUS_FAILURE;    }    if (stat(outdir,&s)<0) {	if (errno == ENOENT) {	    fprintf(stderr,"Info: Creating directory \"%s\" ...\n",outdir);	    if (p_mkdir(outdir,S_IRWXU+S_IRWXG+S_IRWXO)<0) {		fprintf(stderr,"%s: could not create output directory \"%s\" (mkdir: %s)",argv[0],outdir,pstrerror(errno));		if (bnifile!=dash && fclose(fbni)<0)		    fprintf(stderr,"%s: could not close BNI file \"%s\" after reading (fclose: %s)\n",argv[0],bnifile,pstrerror(errno));		return STATUS_FAILURE;	    }	} else {	    fprintf(stderr,"%s: could not stat output directory \"%s\" (stat: %s)\n",argv[0],outdir,pstrerror(errno));	    if (bnifile!=dash && fclose(fbni)<0)		fprintf(stderr,"%s: could not close BNI file \"%s\" after reading (fclose: %s)\n",argv[0],bnifile,pstrerror(errno));	    return STATUS_FAILURE;	}    }    else	if (S_ISDIR(s.st_mode) == 0) {	    fprintf(stderr,"%s: \"%s\" is not a directory\n",argv[0],outdir);	    if (bnifile!=dash && fclose(fbni)<0)		fprintf(stderr,"%s: could not close BNI file \"%s\" after reading (fclose: %s)\n",argv[0],bnifile,pstrerror(errno));	    return STATUS_FAILURE;	}        {	unsigned int i;	int          curry;	t_tgaimg *   iconimg;	t_bnifile *  bni;	FILE *       indexfile;	char *       indexfilename;		fprintf(stderr,"Info: Loading \"%s\" ...\n",bnifile);	bni = load_bni(fbni);	if (bni == NULL) return STATUS_FAILURE;	if (fseek(fbni,bni->dataoffset,SEEK_SET)<0) {		fprintf(stderr,"%s: could not seek to TGA data offset %lu (fseek: %s)\n",argv[0],(unsigned long int)bni->dataoffset,pstrerror(errno));		if (bnifile!=dash && fclose(fbni)<0)			fprintf(stderr,"%s: could not close BNI file \"%s\" after reading (fclose: %s)\n",argv[0],bnifile,pstrerror(errno));		return STATUS_FAILURE;	}	fprintf(stderr,"Info: Loading image ...\n");	iconimg = load_tga(fbni);	if (iconimg == NULL) return STATUS_FAILURE;		fprintf(stderr,"Info: Extracting icons ...\n");	indexfilename = malloc(strlen(outdir)+14);	sprintf(indexfilename,"%s/bniindex.lst",outdir);	fprintf(stderr,"Info: Writing Index to \"%s\" ... \n",indexfilename);	indexfile = fopen(indexfilename , "w");	if (indexfile == NULL) {		fprintf(stderr,"%s: could not open index file \"%s\" for writing (fopen: %s)\n",argv[0],indexfilename,pstrerror(errno));		if (bnifile!=dash && fclose(fbni)<0)			fprintf(stderr,"%s: could not close BNI file \"%s\" after reading (fclose: %s)\n",argv[0],bnifile,pstrerror(errno));		return STATUS_FAILURE;	}	fprintf(indexfile,"unknown1 %08x\n",bni->unknown1);	fprintf(indexfile,"unknown2 %08x\n",bni->unknown2);	curry = 0;	for (i=0; i < bni->numicons; i++) {		FILE *dsttga;		char *name;		t_tgaimg *icn;		icn = area2img(iconimg,0,curry,bni->icons->icon[i].x,bni->icons->icon[i].y,tgaimgtype_uncompressed_truecolor);		if (icn == NULL) {			fprintf(stderr,"Error: area2img failed!\n");			return STATUS_FAILURE;		}		if (bni->icons->icon[i].id == 0) {			int tag = bni->icons->icon[i].tag;			name = malloc(strlen(outdir)+10);			sprintf(name,"%s/%c%c%c%c.tga",outdir,((tag >> 24) & 0xff),((tag >> 16) & 0xff),((tag >> 8) & 0xff),((tag >> 0) & 0xff));		} else {			name = malloc(strlen(outdir)+16);			sprintf(name,"%s/%08x.tga",outdir,bni->icons->icon[i].id);		}		fprintf(stderr,"Info: Writing icon %u(%ux%u) to file \"%s\" ... \n",i+1,icn->width,icn->height,name);		curry += icn->height;		dsttga = fopen(name,"w");		if (dsttga == NULL) {			fprintf(stderr,"%s: could not open ouptut TGA file \"%s\" for writing (fopen: %s)\n",argv[0],name,pstrerror(errno));		} else {			if (write_tga(dsttga,icn) < 0) {				fprintf(stderr,"Error: Writing to TGA failed.\n");			} else {				int tag = bni->icons->icon[i].tag;				if (bni->icons->icon[i].id == 0) {					fprintf(indexfile,"icon !%c%c%c%c %d %d %08x\n",((tag >> 24) & 0xff),((tag >> 16) & 0xff),((tag >> 8) & 0xff),((tag >> 0) & 0xff),bni->icons->icon[i].x,bni->icons->icon[i].y,bni->icons->icon[i].unknown);				} else {					fprintf(indexfile,"icon #%08x %d %d %08x\n",bni->icons->icon[i].id,bni->icons->icon[i].x,bni->icons->icon[i].y,bni->icons->icon[i].unknown);				}			}			if (fclose(dsttga)<0)				fprintf(stderr,"%s: could not close TGA file \"%s\" after writing (fclose: %s)\n",argv[0],name,pstrerror(errno));		}		free(name);		destroy_img(icn);	}	destroy_img(iconimg);	if (fclose(indexfile)<0) {		fprintf(stderr,"%s: could not close index file \"%s\" after writing (fclose: %s)\n",argv[0],indexfilename,pstrerror(errno));		return STATUS_FAILURE;	}	free(indexfilename);    }    if (bnifile!=dash && fclose(fbni)<0)	fprintf(stderr,"%s: could not close BNI file \"%s\" after reading (fclose: %s)\n",argv[0],bnifile,pstrerror(errno));        return STATUS_SUCCESS;}

⌨️ 快捷键说明

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