/*==========================================================
  File:  texture.c
  Author:  _pragma

  Description:  Loads and maintains texture objects.
  ==========================================================*/


#include <SDL/SDL.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <GL/gl.h>
#include <GL/glu.h>

#include "tga.h"
#include "util.h"
#include "texture.h"
#include "console.h"

texture_t gt_textures[T_MAX_TEXTURES];  // fixme: perhaps a linked list instead
int gt_textureIndex = 0;

void T_initTextures(void)
{
  CON_printf("---------- Init Textures -----------");

  if(!T_loadTextureTGA("data/textures/system/font.tga"true))
    CON_printf("texture:  failed to load font.tga");

  // load console texture
  CON_loadTexture();

  CON_printf("------------------------------------");
}

void T_destroyTextures(void)
{
  int i;
  int a[T_MAX_TEXTURES] = {0};

  CON_printf("------- Destroying Textures --------");

  for(i = 0i < gt_textureIndexi++)
  {
    a[i] = gt_textures[i].id;

    if(gt_textures[i].filename)
    {
      CON_printf("Destroying %s"gt_textures[i].filename);
      free(gt_textures[i].filename);
    }
  }

#if 0
  for(;i < T_MAX_TEXTURESi++)
    a[i] = 0;
#endif

  glDeleteTextures(gt_textureIndexa);

  CON_printf("------------------------------------");
}

#if 0
GLuint T_loadTexturePCX(char *filenamechar wrap)
{
  // fixme: duh.
  return 0;
}
#endif

/* from craterz@hotmail.com, 
   http://www.geocities.com/SiliconValley/Code/1219/
 */

GLuint T_loadTextureRAW(char *filenamechar wrap)
{
    GLuint texture;
    int widthheight;
    unsigned char * data;
    FILE * file;

    // open texture data
    file = fopenfilename"rb" );
    if ( file == NULL )
    {
      perror(filename);
      return 0;
    }

    // allocate buffer
    width = 256;
    height = 256;
    data = mallocwidth * height * 3 );

    // read texture data
    freaddatawidth * height * 31file );
    fclosefile );

    // allocate a texture name
    glGenTextures1, &texture );

    // select our current texture
    glBindTextureGL_TEXTURE_2Dtexture );

    // select modulate to mix texture with color for shading
    glTexEnvfGL_TEXTURE_ENVGL_TEXTURE_ENV_MODEGL_MODULATE );
//    glTexEnvf( GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_DECAL );
//    glTexEnvf( GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE );

    // when texture area is small, bilinear filter the closest mipmap
    glTexParameterfGL_TEXTURE_2DGL_TEXTURE_MIN_FILTER,
                     GL_LINEAR_MIPMAP_NEAREST );
    // when texture area is large, bilinear filter the first mipmap
    glTexParameterfGL_TEXTURE_2DGL_TEXTURE_MAG_FILTERGL_LINEAR);

    // if wrap is true, the texture wraps over at the edges (repeat)
    //       ... false, the texture ends at the edges (clamp)
    glTexParameterfGL_TEXTURE_2DGL_TEXTURE_WRAP_S,
                     wrap ? GL_REPEAT : GL_CLAMP );
    glTexParameterfGL_TEXTURE_2DGL_TEXTURE_WRAP_T,
                     wrap ? GL_REPEAT : GL_CLAMP );

    // build our texture mipmaps
    gluBuild2DMipmapsGL_TEXTURE_2D3widthheight,
                       GL_RGBGL_UNSIGNED_BYTEdata );

    // free buffer
    freedata );
    return texture;
}

GLuint T_loadTextureTGA(char *filenamechar wrap)
{
  GLuint texture;
  unsigned char *data;
  int wh;
  int i;
  char *p = filename;

  while(*p++)
    if(*p == '\\')
      *p = '/';

  for(i = 0i < gt_textureIndexi++)
    if(strcmp(gt_textures[i].filenamefilename) == 0)
    {
      CON_printf("%s already loaded as %d"filenamegt_textures[i].id);
      return gt_textures[i].id;
    }

  if(!(data = TGA_getData(filename, &w, &h)))
  {
    CON_printf("warning:  could not get TGA data for \'%s\'"filename);
    return 0;
  }

  CON_printf("loading texture:  %s (%dx%d)"filenamewh);

  glGenTextures1, &texture );
  //printf("gettex: %s\n", gluErrorString(glGetError()));

  gt_textures[gt_textureIndex].filename = strdup(filename);
  gt_textures[gt_textureIndex].width = w;
  gt_textures[gt_textureIndex].height = h;
  gt_textures[gt_textureIndex++].id = texture;
  

  glBindTextureGL_TEXTURE_2Dtexture );

#ifdef DEBUG_TEXTURES
  printf("bindtext: %s\n"gluErrorString(glGetError()));
#endif

  glTexEnvfGL_TEXTURE_ENVGL_TEXTURE_ENV_MODEGL_MODULATE );

#ifdef DEBUG_TEXTURES
  printf("texenv: %s\n"gluErrorString(glGetError()));
#endif

  glTexParameterfGL_TEXTURE_2DGL_TEXTURE_MIN_FILTERGL_LINEAR );

#ifdef DEBUG_TEXTURES
  printf("texparam: %s\n"gluErrorString(glGetError()));
#endif

  glTexParameterfGL_TEXTURE_2DGL_TEXTURE_MAG_FILTERGL_LINEAR);

#ifdef DEBUG_TEXTURES
  printf("texparam: %s\n"gluErrorString(glGetError()));
#endif

  glTexParameterfGL_TEXTURE_2DGL_TEXTURE_WRAP_S,
                   wrap ? GL_REPEAT : GL_CLAMP );
#ifdef DEBUG_TEXTURES
  printf("texparam: %s\n"gluErrorString(glGetError()));
#endif

  glTexParameterfGL_TEXTURE_2DGL_TEXTURE_WRAP_T,
                   wrap ? GL_REPEAT : GL_CLAMP );

#ifdef DEBUG_TEXTURES
  printf("texparam: %s\n"gluErrorString(glGetError()));
#endif

  glTexImage2D(GL_TEXTURE_2D03wh0GL_RGBAGL_UNSIGNED_BYTEdata);

#ifdef DEBUG_TEXTURES
  printf("textimage: %s\n"gluErrorString(glGetError()));
#endif

  free(data);
  return texture;
}

void T_changeGamma(char *imageint sizefloat factor)
{
  int iafter;
  float scale = 1.0temp = 0.0r = 0.0g = 0.0b = 0.0;

  return;

/* whoever wrote this was on crack */

    // Go through every pixel in the lightmap
    for(i = 0i < size / 3i++, image += 3)
    {
        // extract the current RGB values
        r = (float)image[0];
        g = (float)image[1];
        b = (float)image[2];

        if(image[0] != 0 && image[1] != 0 && image[2] != 0)
        {
//          CON_printf("before: %d, %d, %d", image[0], image[1], image[2]);
          after = true;
        }
        else
          after = false;

       // Multiply the factor by the RGB values, while keeping it to a 255 ratio
       r = r * factor / 255.0f;
       g = g * factor / 255.0f;
       b = b * factor / 255.0f;

       // Check if the the values went past the highest value
       if(r > 1.0f && (temp = (1.0f/r)) < scale)
           scale=temp;
       if(g > 1.0f && (temp = (1.0f/g)) < scale)
           scale=temp;
       if(b > 1.0f && (temp = (1.0f/b)) < scale)
           scale=temp;

       // Get the scale for this pixel and multiply it by our pixel values
       scale*=255.0f;

       r*=scale;
       g*=scale;
       b*=scale;

        // Assign the new gamma'nized RGB values to our image
        image[0] = (char)r;
        image[1] = (char)g;
        image[2] = (char)b;
/*
        if(after)
          CON_printf(" after: %d, %d, %d", image[0], image[1], image[2]);
*/

    }
}