/* loadpng, Allegro wrapper routines for libpng
 * by Peter Wang (tjaden@users.sf.net).
 */


#include <png.h>
#include <allegro.h>
#include "loadpng.h"



static void user_error_fn(png_structp png_ptr, png_const_charp message)
{
   jmp_buf *jmpbuf = (jmp_buf *)png_get_error_ptr(png_ptr);
   (void)message;
   longjmp(*jmpbuf, 1);
}



/* write_data:
 *  Custom write function to use Allegro packfile routines,
 *  rather than C streams.
 */
static void write_data(png_structp png_ptr, png_bytep data, png_uint_32 length)
{
    PACKFILE *f = (PACKFILE *)png_get_io_ptr(png_ptr);
    if ((png_uint_32)pack_fwrite(data, length, f) != length)
        png_error(png_ptr, "write error (loadpng calling pack_fwrite)");
}

/* Don't think Allegro has any problem with buffering
 * (rather, Allegro provides no way to flush packfiles).
 */
static void flush_data(png_structp png_ptr) { (void)png_ptr; }



/* save_indexed:
 *  Core save routine for 8 bpp images.
 * */
static int save_indexed(png_structp png_ptr, BITMAP *bmp)
{
    ASSERT(bitmap_color_depth(bmp) == 8);

    if (is_memory_bitmap(bmp)) { /* fast path */
        int y;

        for (y=0; y<bmp->h; y++) {
            png_write_row(png_ptr, bmp->line[y]);
        }

        return 1;
    }
    else {                      /* generic case */
        unsigned char *rowdata;
        int x, y;

        rowdata = (unsigned char *)malloc(bmp->w * 3);
        if (!rowdata)
            return 0;

        for (y=0; y<bmp->h; y++) {
            unsigned char *p = rowdata;

            for (x=0; x<bmp->w; x++) {
                *p++ = getpixel(bmp, x, y);
            }

            png_write_row(png_ptr, rowdata);
        }

        free(rowdata);

        return 1;
    }
}



/* save_rgb:
 *  Core save routine for 15/16/24 bpp images (original by Martijn Versteegh).
 */
static int save_rgb(png_structp png_ptr, BITMAP *bmp)
{
    AL_CONST int depth = bitmap_color_depth(bmp);
    unsigned char *rowdata;
    int y, x;

    ASSERT(depth == 15 || depth == 16 || depth == 24);

    rowdata = (unsigned char *)malloc(bmp->w * 3);
    if (!rowdata)
        return 0;

    for (y=0; y<bmp->h; y++) {
        unsigned char *p = rowdata;

        if (depth == 15) {
            for (x = 0; x < bmp->w; x++) {
                int c = getpixel(bmp, x, y);
                *p++ = getr15(c);
                *p++ = getg15(c);
                *p++ = getb15(c);
            }
        }
        else if (depth == 16) {
            for (x = 0; x < bmp->w; x++) {
                int c = getpixel(bmp, x, y);
                *p++ = getr16(c);
                *p++ = getg16(c);
                *p++ = getb16(c);
            }
        }
        else { /* depth == 24 */
            for (x = 0; x < bmp->w; x++) {
                int c = getpixel(bmp, x, y);
                *p++ = getr24(c);
                *p++ = getg24(c);
                *p++ = getb24(c);
            }
        }

        png_write_row(png_ptr, rowdata);
    }

    free(rowdata);

    return 1;
}



/* save_rgba:
 *  Core save routine for 32 bpp images.
 */
static int save_rgba(png_structp png_ptr, BITMAP *bmp)
{
    unsigned char *rowdata;
    int x, y;

    ASSERT(bitmap_color_depth(bmp) == 32);

    rowdata = (unsigned char *)malloc(bmp->w * 4);
    if (!rowdata)
        return 0;

    for (y=0; y<bmp->h; y++) {
        unsigned char *p = rowdata;

        for (x=0; x<bmp->w; x++) {
            int c = getpixel(bmp, x, y);
            *p++ = getr32(c);
            *p++ = getg32(c);
            *p++ = getb32(c);
            *p++ = geta32(c);
        }

        png_write_row(png_ptr, rowdata);
    }

    free(rowdata);

    return 1;
}



/* save_png:
 *  Writes a non-interlaced, no-frills PNG, taking the usual save_xyz
 *  parameters.  Returns non-zero on error.
 */
static int really_save_png(PACKFILE *fp, BITMAP *bmp, AL_CONST RGB *pal)
{
    jmp_buf jmpbuf;
    png_structp png_ptr = NULL;
    png_infop info_ptr = NULL;
    int depth;
    int colour_type;

    depth = bitmap_color_depth(bmp);
    if (depth == 8 && !pal)
        return -1;

    /* Create and initialize the png_struct with the
     * desired error handler functions.
     */
    png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING,
                                      (void *)NULL, NULL, NULL);
    if (!png_ptr)
        goto Error;

    /* Allocate/initialize the image information data. */
    info_ptr = png_create_info_struct(png_ptr);
    if (!info_ptr)
        goto Error;

    /* Set error handling. */
    if (setjmp(jmpbuf)) {
        /* If we get here, we had a problem reading the file. */
        goto Error;
    }
    png_set_error_fn(png_ptr, jmpbuf, user_error_fn, NULL);

    /* Use packfile routines. */
    png_set_write_fn(png_ptr, fp, (png_rw_ptr)write_data, flush_data);

    /* Set the image information here.  Width and height are up to 2^31,
     * bit_depth is one of 1, 2, 4, 8, or 16, but valid values also depend on
     * the color_type selected. color_type is one of PNG_COLOR_TYPE_GRAY,
     * PNG_COLOR_TYPE_GRAY_ALPHA, PNG_COLOR_TYPE_PALETTE, PNG_COLOR_TYPE_RGB,
     * or PNG_COLOR_TYPE_RGB_ALPHA.  interlace is either PNG_INTERLACE_NONE or
     * PNG_INTERLACE_ADAM7, and the compression_type and filter_type MUST
     * currently be PNG_COMPRESSION_TYPE_BASE and PNG_FILTER_TYPE_BASE.
     */
    if (depth == 8)
        colour_type = PNG_COLOR_TYPE_PALETTE;
    else if (depth == 32)
        colour_type = PNG_COLOR_TYPE_RGB_ALPHA;
    else
        colour_type = PNG_COLOR_TYPE_RGB;

    /* Set compression level. */
    png_set_compression_level(png_ptr, _png_compression_level);

    png_set_IHDR(png_ptr, info_ptr, bmp->w, bmp->h, 8, colour_type,
                 PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_BASE,
                 PNG_FILTER_TYPE_BASE);

    /* Set the palette if there is one.  Required for indexed-color images. */
    if (colour_type == PNG_COLOR_TYPE_PALETTE) {
        png_color palette[256];
        int i;

        for (i = 0; i < 256; i++) {
            palette[i].red   = _rgb_scale_6[pal[i].r];   /* 64 -> 256 */
            palette[i].green = _rgb_scale_6[pal[i].g];
            palette[i].blue  = _rgb_scale_6[pal[i].b];
        }

        /* Set palette colors. */
        png_set_PLTE(png_ptr, info_ptr, palette, 256);
    }

    /* Optionally write comments into the image ... Nah. */

    /* Write the file header information. */
    png_write_info(png_ptr, info_ptr);

    /* Once we write out the header, the compression type on the text
     * chunks gets changed to PNG_TEXT_COMPRESSION_NONE_WR or
     * PNG_TEXT_COMPRESSION_zTXt_WR, so it doesn't get written out again
     * at the end.
     */

    /* Save the data. */
    switch (depth) {
        case 8:
            if (!save_indexed(png_ptr, bmp))
                goto Error;
            break;
        case 15:
        case 16:
        case 24:
            if (!save_rgb(png_ptr, bmp))
                goto Error;
            break;
        case 32:
            if (!save_rgba(png_ptr, bmp))
                goto Error;
            break;
        default:
            ASSERT(FALSE);
            goto Error;
    }

    png_write_end(png_ptr, info_ptr);

    png_destroy_write_struct(&png_ptr, &info_ptr);

    return 0;

  Error:

    if (png_ptr) {
        if (info_ptr)
            png_destroy_write_struct(&png_ptr, &info_ptr);
        else
            png_destroy_write_struct(&png_ptr, NULL);
    }

    return -1;
}


int save_png(AL_CONST char *filename, BITMAP *bmp, AL_CONST RGB *pal)
{
    PACKFILE *fp;
    int result;

    ASSERT(filename);
    ASSERT(bmp);

    fp = pack_fopen(filename, "w");
    if (!fp)
        return -1;

    acquire_bitmap(bmp);
    result = really_save_png(fp, bmp, pal);
    release_bitmap(bmp);

    pack_fclose(fp);

    return result;
}