/* zic2xpm.c Program to convert pieces from ZIICS format to XPM & XIM format. (C version) By Frank McIngvale . Copyright (C) 1996 Free Software Foundation, Inc. NOTICE: The piece images distributed with ZIICS are copyrighted works of their original creators. Images converted with zic2xpm may not be redistributed without the permission of the copyright holders. Do not contact the authors of zic2xpm or of ZIICS itself to request permission. NOTICE: The format of the ZIICS piece file was gleaned from SHOWSETS.PAS, a part of ZIICS. Thanks to Andy McFarland (Zek on ICC) for making this source available! ZIICS is a completely separate and copyrighted work of Andy McFarland. Use and distribution of ZIICS falls under the ZIICS license, NOT the GNU General Public License. NOTICE: The format of the VGA imageblocks was determined by experimentation, and without access to any of Borland Inc.'s BGI library source code. 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. However, the above notices MUST BE RETAINED in any copy that you redistribute or modify. 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 USA. ** If you find a bug in zic2xpm.c, please report it to me, Frank McIngvale (frankm@hiwaay.net) so that I may fix it. ** */ /* Usage: zic2xpm file1 [file2 ...] We split the ZIICS file(s) into 24 XPM & 24 XIM files with names: .(xpm|xim). Where: piece = p, n, b, r, q, k type = ll, ld, dl, dd size = Piece size. Plus 4 files for the light & dark squares. This means that you can extract multiple SIZES in one directory without name clashes. Extracting two sets of the SAME size in a directory will cause the second to overwrite the first. */ /* Technical note: Yes, this file is huge. I made it by cramming `zic2xpm' and `zic2xim' together. This should be less confusing to use, though. */ #include "config.h" #include #if STDC_HEADERS #include #endif #ifndef SEEK_SET #define SEEK_SET 0 #endif /* XIM file format: width byte height byte Data (1 byte per pixel, row major) */ /* Map colors from ZIICS -> XIM : 0 0 Dark piece 2 1 Dark square 15 2 Light piece 14 3 Light square */ typedef struct { int zval; /* ZIICS value */ int xval; /* XIM value */ } z2xim; /* Associate VGA color with XPM color/sym */ typedef struct { int cval; /* VGA pixel value */ char xchar; /* XPM character for this color */ char *csym; /* Symbolic name */ char *cdefault; /* Default color */ } z2xpm; #define NR_ZIICS_COLORS 4 /* SHOWSETS.PAS (from ZIICS) states that images may only use color numbers 0, 2, 14, and 15 */ z2xim z2xim_tab[NR_ZIICS_COLORS] = { { 0, 0 }, { 2, 1 }, { 15, 2 }, { 14, 3 } }; z2xpm z2xpm_tab[NR_ZIICS_COLORS] = { { 15, 'X', "light_piece", "white" }, { 0, ' ', "dark_piece", "black" }, { 14, '*', "light_square", "gray" }, { 2, '.', "dark_square", "green" } }; void fatal( str ) char *str; { printf("Fatal error: %s\n", str ); exit(1); } z2xim *lookup_xim_color( color ) int color; { int i; for( i=0; i 0; --i, --w ) { byte = 0; /* 1 bit from each plane */ for( j=0; j<4; ++j ) { bit = b[j]; bit &= (1 << i); bit >>= i; bit <<= 3-j; byte |= bit; } *(dest++) = byte; } return dest; } /* W is width of image in PIXELS. SRC is in packed pixel format. DEST is filled with 1 BYTE per PIXEL. */ unsigned char *decode_line( dest, src, w ) unsigned char *dest, *src; int w; { unsigned int w8; unsigned int bpp; unsigned char b[4]; int i; unsigned char *p; p = src; w8 = up8( w ); /* 4 planes, bpp BYTES per plane */ /* Planes are MSB -> LSB */ bpp = w8 >> 3; while( w > 0 ) { for( i=0; i<4; ++i ) b[i] = p[i*bpp]; if ( w > 8 ) dest = decode_byte( dest, b, 8 ); else dest = decode_byte( dest, b, w ); w -= 8; ++p; } return (src + bpp * 4); } int write_xim_header( fp, w, h ) FILE *fp; int w, h; { fputc( w, fp ); fputc( h, fp ); return 0; } int write_xpm_header( fp, w, h ) FILE *fp; int w, h; { int i; z2xpm *cv; fprintf(fp, "/* XPM */\n"); fprintf(fp, "/* This file was automatically generated from the file %s\n", src_name ); fprintf(fp, "using the program ``zic2xpm''.\n"); fprintf(fp, "\n %s\n %s\n %s\n %s\n %s\n %s */\n", "NOTICE: The piece images distributed with ZIICS are", " copyrighted works of their original creators. Images", " converted with zic2xpm may not be redistributed without", " the permission of the copyright holders. Do not contact", " the authors of zic2xpm or of ZIICS itself to request", " permission."); fprintf( fp, "static char * image_name[] = {\n" ); fprintf( fp, "\"%d %d %d 1\",\n", h, w, NR_ZIICS_COLORS ); cv = z2xpm_tab; for( i=0; ixchar, cv->cdefault, cv->csym ); } return 0; } void create_piece_xim( outname, fpin, W, H ) char *outname; FILE *fpin; int W, H; { FILE *fpout; int w, h, i, j, c; unsigned char *lump, *p, *line; long size; z2xim *ent; fpout = fopen( outname, "wb" ); if ( !fpout ) fatal( "Can't create output file."); /* Header is two ints -- Width then Height, x86 format */ c = fgetc( fpin ); w = (fgetc(fpin) << 8) | c; c = fgetc( fpin ); h = (fgetc(fpin) << 8) | c; ++w; ++h; if ( w != W || h != H ) fatal( "Bad header." ); size = vga_imagesize( w, h ) - 4; lump = (unsigned char*)malloc( size ); line = (unsigned char*)malloc( w ); if ( !lump || !line ) fatal( "Out of memory." ); fread( lump, 1, size, fpin ); /* Write XIM header */ write_xim_header( fpout, w, h ); p = lump; /* Write XIM data */ for( i=0; ixval, fpout ); } } free( lump ); free( line ); fclose( fpout ); } void create_piece_xpm( outname, fpin, W, H ) char *outname; FILE *fpin; int W, H; { FILE *fpout; int w, h, i, j, c; unsigned char *lump, *p, *line; long size; z2xpm *cv; fpout = fopen( outname, "wb" ); if ( !fpout ) fatal( "Can't create output file."); /* Header is two ints -- Width then Height, x86 format */ c = fgetc( fpin ); w = (fgetc(fpin) << 8) | c; c = fgetc( fpin ); h = (fgetc(fpin) << 8) | c; ++w; ++h; if ( w != W || h != H ) fatal( "Bad header." ); size = vga_imagesize( w, h ) - 4; lump = (unsigned char*)malloc( size ); line = (unsigned char*)malloc( w ); if ( !lump || !line ) fatal( "Out of memory." ); fread( lump, 1, size, fpin ); /* Write XPM header */ write_xpm_header( fpout, w, h ); p = lump; /* Write XPM data */ for( i=0; ixchar ); } fprintf( fpout, "\",\n" ); } fprintf( fpout, "};\n" ); free( lump ); free( line ); fclose( fpout ); } /* The order of the pieces in the ZIICS piece file (from SHOWSETS.PAS) */ char *pieces = "prkqbn"; char *pname[] = { "Pawn", "Rook", "King", "Queen", "Bishop", "Knight" }; /* The suborder - Light/Light, Light/Dark, etc. */ char *prefixes[] = { "ll", "ld", "dl", "dd" }; int process_file_xim( filename ) char *filename; { int w, h, piece, kind, c; int nr_pieces = 6; int nr_kinds = 4; FILE *fp; char buf[100]; src_name = filename; fp = fopen( filename, "rb" ); if ( !fp ) fatal( "Can't open input file." ); /* Header is two ints -- Width then Height, x86 format */ c = fgetc( fp ); w = (fgetc(fp) << 8) | c; c = fgetc( fp ); h = (fgetc(fp) << 8) | c; ++w; ++h; if ( w != h ) { printf("ERROR: Can only convert square pieces.\n"); printf(" (This set is %dx%d)\n", w, h ); exit(1); } printf("Creating XIM files...\n"); printf("File: %s, W=%d, H=%d\n", filename, w, h ); fseek( fp, 0, SEEK_SET ); /* Write .XIM files */ for( piece = 0; piece < nr_pieces; ++piece ) { printf("%s ", pname[piece] ); for( kind = 0; kind < nr_kinds; ++kind ) { printf( "." ); /* Form output filename -- .xim */ sprintf(buf, "%c%s%d.xim", pieces[piece], prefixes[kind], w); create_piece_xim( buf, fp, w, h ); } printf("\n"); } /* Write the light & dark squares */ sprintf( buf, "lsq%d.xim", w ); printf("Light Square" ); create_piece_xim( buf, fp, w, h ); sprintf( buf, "dsq%d.xim", w ); printf("\nDark Square" ); create_piece_xim( buf, fp, w, h ); printf("\n"); printf("Successfully converted!!\n" ); fclose( fp ); return 0; } int process_file_xpm( filename ) char *filename; { int w, h, piece, kind, c; int nr_pieces = 6; int nr_kinds = 4; FILE *fp; char buf[100]; src_name = filename; fp = fopen( filename, "rb" ); if ( !fp ) fatal( "Can't open input file." ); /* Header is two ints -- Width then Height, x86 format */ c = fgetc( fp ); w = (fgetc(fp) << 8) | c; c = fgetc( fp ); h = (fgetc(fp) << 8) | c; ++w; ++h; if ( w != h ) { printf("ERROR: Can only convert square pieces.\n"); printf(" (This set is %dx%d)\n", w, h ); exit(1); } printf("Creating XPM files...\n"); printf("File: %s, W=%d, H=%d\n", filename, w, h ); fseek( fp, 0, SEEK_SET ); /* Write .XPM files */ for( piece = 0; piece < nr_pieces; ++piece ) { printf("%s ", pname[piece] ); for( kind = 0; kind < nr_kinds; ++kind ) { printf( "." ); /* Form output filename -- .xpm */ sprintf(buf, "%c%s%d.xpm", pieces[piece], prefixes[kind], w); create_piece_xpm( buf, fp, w, h ); } printf("\n"); } /* Write the light & dark squares */ sprintf( buf, "lsq%d.xpm", w ); printf("Light Square" ); create_piece_xpm( buf, fp, w, h ); sprintf( buf, "dsq%d.xpm", w ); printf("\nDark Square" ); create_piece_xpm( buf, fp, w, h ); printf("\n"); printf("Successfully converted!!\n" ); fclose( fp ); return 0; } int main( argc, argv ) int argc; char *argv[]; { int i; if ( argc < 2 ) { printf("ZIC2XPM 2.01 - by Frank McIngvale (frankm@hiwaay.net)\n"); printf("Copyright (C) 1996 Free Software Foundation, Inc.\n\n"); printf("Usage: zic2xpm file1 [file2 ...]\n\n"); printf(" Splits each file (ZIICS piece files) into 26 XPM & XIM files\n"); printf(" suitable for use in XBoard 3.5 or later.\n"); printf("\n* ZIICS is a copyrighted work of Andy McFarland (Zek on ICC) *\n"); return 1; } setbuf( stdout, NULL ); for( i=1; i