/*
 *		Lecture des images PNG
 *		----------------------
 */

#include <stdio.h>
#include <stdlib.h>
#include   <stdint.h>
#include <string.h>

#include <pnglite.h>        

#include "../floatimg.h"

extern int		verbosity;

/* --------------------------------------------------------------------- */
/*
 *	warning : this func has been tested only with
 * 	RGB (3x8bits) PNG files
 *
 * 	Attention : certains fichiers, tels que ceux
 * 	produits par ImageMagick ne sont pas lisibles
 * 	par cette fonction :(
 */
int fimg_create_from_png(char *filename, FloatImg *fimg)
{
png_t		png;
int		foo, idx;
unsigned char	*datas, *ptr;
int		datasize;

#if DEBUG_LEVEL
fprintf(stderr, ">>> %-25s ( '%s' %p )\n", __func__, filename, fimg);
#endif

/*	We MUST clear the fimg destination header first */
memset(fimg, 0, sizeof(FloatImg));

memset(&png, 0, sizeof(png_t));
png_init(NULL, NULL);			/* this is VITAL ! */

foo = png_open_file_read(&png, filename);
if (PNG_NO_ERROR != foo) {
	fprintf(stderr, "%s :\n\topen_file '%s' = %d %s\n", __func__,
			filename, foo, png_error_string(foo));
	png_close_file(&png);
	return foo;
	}
#if DEBUG_LEVEL > 1
fprintf(stderr, "%s opened\n", filename);
#endif

if (verbosity) {
	printf("----------- %s ---------\n", filename);
	png_print_info(&png);		puts("");
	fflush(stdout);
	}

if ( 3 != png.bpp ) {
	fprintf(stderr, "bpp format %d of '%s' not supported\n",
			png.color_type, filename);
	return -21;
	}

datasize = png.width * png.height * png.bpp;
datas = malloc(datasize);
if (NULL==datas) {
	fprintf(stderr, "%s : fatal memory failure\n", __func__);
	exit(1);
	}

memset(fimg, 0, sizeof(FloatImg));
foo = fimg_create(fimg, png.width, png.height, 3);
if (foo) {
	fprintf(stderr, "can't create fimg\n");
	exit(1);
	}

/* I GET AN 'Unknown file error' HERE, WHY ???*/
foo = png_get_data(&png, datas);
if (PNG_NO_ERROR != foo) {
	fprintf(stderr, "err in '%s:%s' :\n\tpng_get_data -> %d = %s\n\n",
			__FILE__, __func__, foo, png_error_string(foo));
	png_close_file(&png);
	return foo;
	}

ptr = datas;
for (idx=0; idx<png.width * png.height; idx++) {
	fimg->R[idx] = (float)*ptr++;
	fimg->G[idx] = (float)*ptr++;
	fimg->B[idx] = (float)*ptr++;
	}
free(datas);

png_close_file(&png);
// fprintf(stderr, "png closed...\n");
return foo;
}
/* --------------------------------------------------------------------- */
/*
 *	warning : this func has been tested only with
 * 	RGB (3x8bits) PNG files
 */
int fimg_load_from_png(char *filename, FloatImg *fimg)
{
png_t		png;
int		foo, idx, datasize;
unsigned char	*datas, *ptr;

#if DEBUG_LEVEL
fprintf(stderr, ">>> %-25s ( '%s' %p )\n", __func__, filename, fimg);
#endif

memset(&png, 0, sizeof(png_t));
png_init(NULL, NULL);			/* this is VITAL ! */

foo = png_open_file_read(&png, filename);
if (PNG_NO_ERROR != foo) {
#if DEBUG_LEVEL
	fprintf(stderr, "open png -> %d\n", foo);
#endif
	return foo;
	}

// puts(""); png_print_info(&png); puts("");

if ( 3 != png.bpp ) {			/* TO BE PATCHED */
	fprintf(stderr, "format of '%s' not supported\n", filename);
	return -21;
	}

/* check if floatimg and PNG have the same size */
if ((fimg->width != png.width) || (fimg->height != png.height)) {
	fprintf(stderr, "%s : fatal error on images sizes\n", __func__);
	exit(1);
	}

datasize = png.width * png.height * png.bpp;
datas = malloc(datasize);
if (NULL==datas) {
	fprintf(stderr, "%s : fatal memory failure\n", __func__);
	exit(1);
	}
foo = png_get_data(&png, datas);
if (PNG_NO_ERROR != foo) {
	fprintf(stderr, "error in '%s' : read png -> %d\n",
						__func__, foo);
	return foo;
	}

ptr = datas;
for (idx=0; idx<png.width * png.height; idx++) {
	fimg->R[idx] = (float)*ptr++;
	fimg->G[idx] = (float)*ptr++;
	fimg->B[idx] = (float)*ptr++;
	}

free(datas);
png_close_file(&png);
return 0;
}
/* --------------------------------------------------------------------- */
/* nouveau 13 decembre 2019 */
int fimg_save_as_png(FloatImg *src, char *outname, int flags)
{
png_t			png;
int			foo, sz, idx;
unsigned char		*bytes, *bptr;
double			maximum, fk;

#if DEBUG_LEVEL
fprintf(stderr, ">>> %-25s ( %p '%s' 0x%x )\n", __func__, src, outname, flags);
#endif

/* convert ou floating datas to a byte/rgb array */
   /* first, alloc a buffer */
sz = src->width * src->height;
bytes = calloc(sz, 3);
if (NULL==bytes) {
	fprintf(stderr, "%s : no mem ?\n", __func__);
	exit(3);
	}
   /* compute max value */
maximum = (double)fimg_get_maxvalue(src);
fk = maximum / 255.0;
if (verbosity > 1)
	fprintf(stderr, "%s: max val %g fk %g\n", __func__, maximum, fk);
   /* massage des pixels */
bptr = bytes;
for (idx=0; idx<sz; idx++) {
	*bptr++ = (unsigned char) (src->R[idx] / fk);
	*bptr++ = (unsigned char) (src->G[idx] / fk);
	*bptr++ = (unsigned char) (src->B[idx] / fk);
	}

memset(&png, 0, sizeof(png_t));
png_init(NULL, NULL);			/* this is VITAL ! */

foo = png_open_file_write(&png, outname);
if (PNG_NO_ERROR != foo) {
	fprintf(stderr, "error in '%s' : open_file_write -> %d\n",
						__func__, foo);
	return foo;
	}

foo = png_set_data(&png, src->width, src->height, 8, PNG_TRUECOLOR, bytes);
if (PNG_NO_ERROR != foo) {
	fprintf(stderr, "error in '%s' : set_data -> %d\n",
						__func__, foo);
	return foo;
	}

png_close_file(&png);

free(bytes);		/* yolo ? */

return 0;
}
/* --------------------------------------------------------------------- */