/* Loads a 80x48 (or 40x48) PNG image into a 40x48 Apple II layout */ /* Also supports 280x192 */ /* It's not interleaved like an actual Apple II */ /* But the top/bottom are pre-packed into a naive 40x24 array */ /* Note there's a hack to treat color 17 (0xf5ff00) as an alias for color 10 */ /* for transparency reasons */ #include #include #include #include #include #include #include #include #include "loadpng.h" static int convert_color(int color, char *filename) { int c=0; switch(color) { case 0x000000: c=0; break; /* black */ case 0xe31e60: c=1; break; /* magenta */ case 0x604ebd: c=2; break; /* dark blue */ case 0xff44fd: c=3; break; /* purple */ case 0x00a360: c=4; break; /* dark green */ case 0x9c9c9c: c=5; break; /* grey 1 */ case 0x14cffd: c=6; break; /* medium blue */ case 0xd0c3ff: c=7; break; /* light blue */ case 0x607203: c=8; break; /* brown */ case 0xff6a3c: c=9; break; /* orange */ case 0x9d9d9d: c=10; break; /* grey 2 */ case 0xffa0d0: c=11; break; /* pink */ case 0x14f53c: c=12; break; /* bright green */ case 0xd0dd8d: c=13; break; /* yellow */ case 0x72ffd0: c=14; break; /* aqua */ case 0xffffff: c=15; break; /* white */ case 0xf5ff00: c=10; break; /* transparent */ default: fprintf(stderr,"Unknown color %x, file %s\n", color,filename); exit(-1); break; } return c; } /* expects a PNG where the xsize is either 40 or 80 or 280 */ /* if it is 80, it skips every other */ /* if 280, it skips every 7 */ /* why do that? when editing an image the aspect ratio looks better if */ /* it is an 80 wide picture */ /* xsize, ysize is the size of the result, not size of */ /* the input image */ int loadpng(char *filename, unsigned char **image_ptr, int *xsize, int *ysize, int png_type) { int x,y,ystart,yadd,xadd; int color; FILE *infile; int debug=0; unsigned char *image,*out_ptr; int width, height; int a2_color; png_byte bit_depth; png_structp png_ptr; png_infop info_ptr; png_bytep *row_pointers; png_byte color_type; unsigned char header[8]; /* open file and test for it being a png */ infile = fopen(filename, "rb"); if (infile==NULL) { fprintf(stderr,"Error! Could not open %s\n",filename); return -1; } /* Check the header */ fread(header, 1, 8, infile); if (png_sig_cmp(header, 0, 8)) { fprintf(stderr,"Error! %s is not a PNG file\n",filename); return -1; } /* initialize stuff */ png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); if (!png_ptr) { fprintf(stderr,"Error create_read_struct\n"); exit(-1); } info_ptr = png_create_info_struct(png_ptr); if (!info_ptr) { fprintf(stderr,"Error png_create_info_struct\n"); exit(-1); } png_init_io(png_ptr, infile); png_set_sig_bytes(png_ptr, 8); png_read_info(png_ptr, info_ptr); width = png_get_image_width(png_ptr, info_ptr); height = png_get_image_height(png_ptr, info_ptr); /* get the xadd */ if (width==40) { *xsize=40; xadd=1; yadd=1; } else if (width==80) { if (png_type==PNG_NO_ADJUST) { *xsize=80; xadd=1; } else { *xsize=40; xadd=2; } yadd=1; } else if (width==280) { *xsize=40; xadd=7; yadd=4; /* FIXME: check we are 192 in ysize */ } else if (width==16) { *xsize=16; xadd=1; yadd=1; } else { fprintf(stderr,"Unsupported width %d\n",width); return -1; } if ((png_type==PNG_WHOLETHING) || (png_type==PNG_NO_ADJUST)) { *ysize=height; ystart=0; yadd*=2; } else if (png_type==PNG_ODDLINES) { *ysize=height/2; ystart=1; yadd*=4; } else if (png_type==PNG_EVENLINES) { *ysize=height/2; ystart=0; yadd*=4; } else if (png_type==PNG_RAW) { /* FIXME, not working */ *ysize=height; ystart=0; yadd=1; } else { fprintf(stderr,"Unknown PNG type\n"); return -1; } color_type = png_get_color_type(png_ptr, info_ptr); bit_depth = png_get_bit_depth(png_ptr, info_ptr); if (debug) { printf("PNG: width=%d height=%d depth=%d\n",width,height,bit_depth); if (color_type==PNG_COLOR_TYPE_RGB) printf("Type RGB\n"); else if (color_type==PNG_COLOR_TYPE_RGB_ALPHA) printf("Type RGBA\n"); else if (color_type==PNG_COLOR_TYPE_PALETTE) printf("Type palette\n"); printf("Generating output size %d x %d\n",*xsize,*ysize); } // number_of_passes = png_set_interlace_handling(png_ptr); png_read_update_info(png_ptr, info_ptr); row_pointers = (png_bytep*) malloc(sizeof(png_bytep) * height); for (y=0; y16) { fprintf(stderr,"Error color %d\n",a2_color); } /* bottom color */ color=row_pointers[y+(yadd/2)][x]; if (color==16) { color=10; } if (color>16) { fprintf(stderr,"Error color %d\n",color); } a2_color|=(color<<4); if (debug) { printf("%x ",a2_color); } *out_ptr=a2_color; out_ptr++; } else if (bit_depth==4) { /* top color */ a2_color=row_pointers[y][x/2]; if (x%2==0) { a2_color=(a2_color>>4); } a2_color&=0xf; /* bottom color */ color=row_pointers[y+(yadd/2)][x/2]; if (x%2==0) { color=(color>>4); } color&=0xf; a2_color|=(color<<4); if (debug) { printf("%x ",a2_color); } *out_ptr=a2_color; out_ptr++; } } if (debug) printf("\n"); } } else { printf("Unknown color type\n"); } /* Stripe test image */ // for(x=0;x<40;x++) for(y=0;y<40;y++) image[(y*width)+x]=y%16; /* Addr Row /80 %40 $400 0 0 0 0 $428 28 16 0 $450 50 32 0 $480 80 2 1 $4A8 a8 18 1 $4D0 d0 34 1 $500 100 3 2 0,0 0,1 0,2....0,39 16,0 16,1 ....16,39 32,0..32,39, X X X X X X X X */ *image_ptr=image; return 0; } /* for 80 column mode or double-lores */ int loadpng80(char *filename, unsigned char **image_ptr, int *xsize, int *ysize, int png_type) { int x,y,ystart,yadd,xadd; int color; FILE *infile; int debug=0; unsigned char *image,*out_ptr; int width, height; int a2_color; png_byte bit_depth; png_structp png_ptr; png_infop info_ptr; png_bytep *row_pointers; png_byte color_type; unsigned char header[8]; /* open file and test for it being a png */ infile = fopen(filename, "rb"); if (infile==NULL) { fprintf(stderr,"Error! Could not open %s\n",filename); return -1; } /* Check the header */ fread(header, 1, 8, infile); if (png_sig_cmp(header, 0, 8)) { fprintf(stderr,"Error! %s is not a PNG file\n",filename); return -1; } /* initialize stuff */ png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); if (!png_ptr) { fprintf(stderr,"Error create_read_struct\n"); exit(-1); } info_ptr = png_create_info_struct(png_ptr); if (!info_ptr) { fprintf(stderr,"Error png_create_info_struct\n"); exit(-1); } png_init_io(png_ptr, infile); png_set_sig_bytes(png_ptr, 8); png_read_info(png_ptr, info_ptr); width = png_get_image_width(png_ptr, info_ptr); height = png_get_image_height(png_ptr, info_ptr); if (width==80) { *xsize=80; xadd=1; } else { fprintf(stderr,"Unsupported width %d\n",width); return -1; } if (png_type==PNG_WHOLETHING) { *ysize=height; ystart=0; yadd=2; } else if (png_type==PNG_ODDLINES) { *ysize=height/2; ystart=1; yadd=4; } else if (png_type==PNG_EVENLINES) { *ysize=height/2; ystart=0; yadd=4; } else { fprintf(stderr,"Unknown PNG type\n"); return -1; } color_type = png_get_color_type(png_ptr, info_ptr); bit_depth = png_get_bit_depth(png_ptr, info_ptr); if (debug) { printf("PNG: width=%d height=%d depth=%d\n",width,height,bit_depth); if (color_type==PNG_COLOR_TYPE_RGB) printf("Type RGB\n"); else if (color_type==PNG_COLOR_TYPE_RGB_ALPHA) printf("Type RGBA\n"); else if (color_type==PNG_COLOR_TYPE_PALETTE) printf("Type palette\n"); printf("Generating output size %d x %d\n",*xsize,*ysize); } // number_of_passes = png_set_interlace_handling(png_ptr); png_read_update_info(png_ptr, info_ptr); row_pointers = (png_bytep*) malloc(sizeof(png_bytep) * height); for (y=0; y>4); } a2_color&=0xf; /* bottom color */ color=row_pointers[y+(yadd/2)][x/2]; if (x%2==0) { color=(color>>4); } color&=0xf; a2_color|=(color<<4); if (debug) { printf("%x ",a2_color); } *out_ptr=a2_color; out_ptr++; } } if (debug) printf("\n"); } } else { printf("Unknown color type\n"); } /* Stripe test image */ // for(x=0;x<40;x++) for(y=0;y<40;y++) image[(y*width)+x]=y%16; /* Addr Row /80 %40 $400 0 0 0 0 $428 28 16 0 $450 50 32 0 $480 80 2 1 $4A8 a8 18 1 $4D0 d0 34 1 $500 100 3 2 0,0 0,1 0,2....0,39 16,0 16,1 ....16,39 32,0..32,39, X X X X X X X X */ *image_ptr=image; return 0; }