2019-09-04 15:49:53 -04:00
|
|
|
/* Loads a 80x48 (or 40x48) PNG image into a 40x48 Apple II layout */
|
2018-09-27 20:43:47 -04:00
|
|
|
/* It's not interleaved like an actual Apple II */
|
|
|
|
/* But the top/bottom are pre-packed into a naive 40x24 array */
|
|
|
|
|
|
|
|
|
2017-05-03 22:21:19 -04:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
#include <png.h>
|
2019-09-04 15:49:53 -04:00
|
|
|
#include "loadpng.h"
|
2017-07-10 14:52:24 -04:00
|
|
|
|
2019-11-12 10:01:38 -05:00
|
|
|
static int convert_color(int color, char *filename) {
|
2017-07-10 14:52:24 -04:00
|
|
|
|
|
|
|
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 */
|
|
|
|
default:
|
2019-11-12 10:01:38 -05:00
|
|
|
fprintf(stderr,"Unknown color %x, file %s\n",
|
|
|
|
color,filename);
|
|
|
|
exit(-1);
|
2017-07-10 14:52:24 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2019-09-04 15:49:53 -04:00
|
|
|
/* expects a PNG where the xsize is either 40 or 80 */
|
|
|
|
/* if it is 80, it skips every other */
|
|
|
|
|
|
|
|
/* why do that? when editing an image the aspect ratio looks better if */
|
|
|
|
/* it is an 80 wide picture */
|
|
|
|
|
|
|
|
int loadpng(char *filename, unsigned char **image_ptr, int *xsize, int *ysize,
|
|
|
|
int png_type) {
|
2017-05-03 22:21:19 -04:00
|
|
|
|
2019-09-04 15:49:53 -04:00
|
|
|
int x,y,ystart,yadd,xadd;
|
2017-05-03 22:21:19 -04:00
|
|
|
int color;
|
|
|
|
FILE *infile;
|
|
|
|
int debug=0;
|
2017-07-10 14:52:24 -04:00
|
|
|
unsigned char *image,*out_ptr;
|
2017-05-03 22:21:19 -04:00
|
|
|
int width, height;
|
2017-07-10 14:52:24 -04:00
|
|
|
int a2_color;
|
2017-05-03 22:21:19 -04:00
|
|
|
|
|
|
|
png_byte bit_depth;
|
|
|
|
png_structp png_ptr;
|
|
|
|
png_infop info_ptr;
|
|
|
|
png_bytep *row_pointers;
|
2017-07-26 23:55:00 -04:00
|
|
|
png_byte color_type;
|
2017-05-03 22:21:19 -04:00
|
|
|
|
|
|
|
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);
|
|
|
|
|
2019-09-04 15:49:53 -04:00
|
|
|
if (width==40) {
|
|
|
|
*xsize=40;
|
|
|
|
xadd=1;
|
|
|
|
}
|
|
|
|
else if (width==80) {
|
|
|
|
*xsize=40;
|
|
|
|
xadd=2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
fprintf(stderr,"Unsupported width %d\n",width);
|
|
|
|
return -1;
|
|
|
|
}
|
2017-05-03 22:21:19 -04:00
|
|
|
|
2019-09-04 15:49:53 -04:00
|
|
|
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;
|
|
|
|
}
|
2017-07-26 23:55:00 -04:00
|
|
|
|
2019-09-04 15:49:53 -04:00
|
|
|
color_type = png_get_color_type(png_ptr, info_ptr);
|
|
|
|
bit_depth = png_get_bit_depth(png_ptr, info_ptr);
|
2017-07-26 23:55:00 -04:00
|
|
|
|
2017-05-03 22:21:19 -04:00
|
|
|
if (debug) {
|
|
|
|
printf("PNG: width=%d height=%d depth=%d\n",width,height,bit_depth);
|
2017-07-26 23:55:00 -04:00
|
|
|
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");
|
2019-09-04 15:49:53 -04:00
|
|
|
printf("Generating output size %d x %d\n",*xsize,*ysize);
|
2017-05-03 22:21:19 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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<height; y++) {
|
2019-09-04 15:49:53 -04:00
|
|
|
/* FIXME: do we ever free these? */
|
2017-05-03 22:21:19 -04:00
|
|
|
row_pointers[y] = (png_byte*) malloc(png_get_rowbytes(png_ptr,info_ptr));
|
|
|
|
}
|
|
|
|
|
|
|
|
png_read_image(png_ptr, row_pointers);
|
|
|
|
|
|
|
|
fclose(infile);
|
|
|
|
|
2019-09-04 15:49:53 -04:00
|
|
|
/* FIXME: this should be 40x24 max??? */
|
|
|
|
image=calloc(width*height,sizeof(unsigned char));
|
2017-05-03 22:21:19 -04:00
|
|
|
if (image==NULL) {
|
|
|
|
fprintf(stderr,"Memory error!\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2017-07-10 14:52:24 -04:00
|
|
|
out_ptr=image;
|
2017-05-03 22:21:19 -04:00
|
|
|
|
2017-07-26 23:55:00 -04:00
|
|
|
if (color_type==PNG_COLOR_TYPE_RGB_ALPHA) {
|
2019-09-04 15:49:53 -04:00
|
|
|
for(y=ystart;y<height;y+=yadd) {
|
|
|
|
for(x=0;x<width;x+=xadd) {
|
2017-07-10 14:52:24 -04:00
|
|
|
|
2017-07-26 23:55:00 -04:00
|
|
|
/* top color */
|
2019-09-04 15:49:53 -04:00
|
|
|
color= (row_pointers[y][x*xadd*4]<<16)+
|
|
|
|
(row_pointers[y][x*xadd*4+1]<<8)+
|
|
|
|
(row_pointers[y][x*xadd*4+2]);
|
2017-07-26 23:55:00 -04:00
|
|
|
if (debug) {
|
|
|
|
printf("%x ",color);
|
|
|
|
}
|
2017-07-10 14:52:24 -04:00
|
|
|
|
2019-11-12 10:01:38 -05:00
|
|
|
a2_color=convert_color(color,filename);
|
2017-07-10 14:52:24 -04:00
|
|
|
|
2017-07-26 23:55:00 -04:00
|
|
|
/* bottom color */
|
2019-09-04 15:49:53 -04:00
|
|
|
color= (row_pointers[y+1][x*xadd*4]<<16)+
|
|
|
|
(row_pointers[y+1][x*xadd*4+1]<<8)+
|
|
|
|
(row_pointers[y+1][x*xadd*4+2]);
|
2017-07-26 23:55:00 -04:00
|
|
|
if (debug) {
|
|
|
|
printf("%x ",color);
|
|
|
|
}
|
|
|
|
|
2019-11-12 10:01:38 -05:00
|
|
|
a2_color|=(convert_color(color,filename)<<4);
|
2017-07-26 23:55:00 -04:00
|
|
|
|
|
|
|
*out_ptr=a2_color;
|
|
|
|
out_ptr++;
|
2017-05-03 22:21:19 -04:00
|
|
|
}
|
2017-07-26 23:55:00 -04:00
|
|
|
if (debug) printf("\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (color_type==PNG_COLOR_TYPE_PALETTE) {
|
2019-09-04 15:49:53 -04:00
|
|
|
for(y=ystart;y<height;y+=yadd) {
|
|
|
|
for(x=0;x<width;x+=xadd) {
|
2017-07-26 23:55:00 -04:00
|
|
|
|
2019-07-11 13:55:15 -04:00
|
|
|
if (bit_depth==8) {
|
|
|
|
/* top color */
|
2019-09-04 15:49:53 -04:00
|
|
|
a2_color=row_pointers[y][x];
|
2017-07-26 23:55:00 -04:00
|
|
|
|
2019-07-11 13:55:15 -04:00
|
|
|
/* bottom color */
|
2019-09-04 15:49:53 -04:00
|
|
|
color=row_pointers[y+(yadd/2)][x];
|
2017-07-10 14:52:24 -04:00
|
|
|
|
2019-07-11 13:55:15 -04:00
|
|
|
a2_color|=(color<<4);
|
2017-07-10 14:52:24 -04:00
|
|
|
|
2019-07-11 13:55:15 -04:00
|
|
|
if (debug) {
|
|
|
|
printf("%x ",a2_color);
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_ptr=a2_color;
|
|
|
|
out_ptr++;
|
2017-07-26 23:55:00 -04:00
|
|
|
}
|
2019-07-11 13:55:15 -04:00
|
|
|
else if (bit_depth==4) {
|
|
|
|
/* top color */
|
2019-09-04 15:49:53 -04:00
|
|
|
a2_color=row_pointers[y][x/2];
|
2019-09-04 16:04:49 -04:00
|
|
|
if (x%2==0) {
|
2019-09-04 15:49:53 -04:00
|
|
|
a2_color=(a2_color>>4);
|
|
|
|
}
|
|
|
|
a2_color&=0xf;
|
2017-07-26 23:55:00 -04:00
|
|
|
|
2019-07-11 13:55:15 -04:00
|
|
|
/* bottom color */
|
2019-09-04 15:49:53 -04:00
|
|
|
color=row_pointers[y+(yadd/2)][x/2];
|
2019-09-04 16:04:49 -04:00
|
|
|
if (x%2==0) {
|
2019-09-04 15:49:53 -04:00
|
|
|
color=(color>>4);
|
|
|
|
}
|
|
|
|
color&=0xf;
|
2019-07-11 13:55:15 -04:00
|
|
|
|
|
|
|
a2_color|=(color<<4);
|
|
|
|
|
|
|
|
if (debug) {
|
|
|
|
printf("%x ",a2_color);
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_ptr=a2_color;
|
|
|
|
out_ptr++;
|
|
|
|
|
|
|
|
}
|
2017-07-26 23:55:00 -04:00
|
|
|
}
|
|
|
|
if (debug) printf("\n");
|
2017-05-03 22:21:19 -04:00
|
|
|
}
|
2017-07-26 23:55:00 -04:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Unknown color type\n");
|
2017-05-03 22:21:19 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|