/**********************************************************************
This file is part of Crack dot Com's free source code release of Golgotha.
for information about compiling & licensing issues visit this URL
 If that doesn't help, contact Jonathan Clark at 
  golgotha_source@usa.net (Subject should have "GOLG" in it) 
***********************************************************************/

#include "objs/model_id.hh"
#include "string/string.hh"
#include "error/alert.hh"
#include "error/error.hh"
#include "load3d.hh"
#include "obj3d.hh"
#include "saver.hh"
#include "r1_api.hh"
#include "status/status.hh"
#include "time/profile.hh"

i4_profile_class pf_load_models("load_models");
g1_model_list_class g1_model_list_man;

g1_model_ref *model_references=0;


g1_model_ref::g1_model_ref(char *name)
{
  next = model_references;
  model_references = this;
  set_name(name);
}

void g1_model_ref::set_name(char *_name)
{
  name=_name;
  value=g1_model_list_man.find_handle(name);
}

g1_model_ref::~g1_model_ref()
{
//   i4_debug->printf("cleaning up for %s\n", name);

//   for (g1_model_ref *m=model_references; m; m=m->next)
//     i4_debug->printf("%s ", m->name);
//   i4_debug->printf("\n");


  if (model_references==this)
    model_references = next;
  else
  {
    g1_model_ref *p;

    for (p = model_references; p->next && p->next!=this; p=p->next) ;

    if (!p->next)
      i4_error("model reference not in list");
    else
      p->next = p->next->next;
  }
}


i4_grow_heap_class *g1_object_heap=0;


int g1_model_info_compare(const void *a, const void *b)
{
  return strcmp(((g1_model_list_class::model_info *)a)->name_start,
                ((g1_model_list_class::model_info *)b)->name_start);
}

void g1_model_list_class::cleanup()
{
  if (g1_object_heap)  
    delete g1_object_heap;

  if (name_buffer)
    delete name_buffer;
  
  if (array)
  {
    i4_free(array);
    array=0;
  }
}

static i4_profile_class pf_model_load_open("models:open");


void g1_model_list_class::reset(i4_array &model_names, r1_texture_manager_class *tmap)
{
  if (g1_object_heap)  
    g1_object_heap->clear();

  if (name_buffer)
    name_buffer->clear();
  
  if (array)
  {
    i4_free(array);
    array=0;
  }

  pf_load_models.start();

  i4_status_class *stat=i4_create_status(i4gets("loading_models"));

  total_models=model_names.size();
  array=(model_info *)i4_malloc(total_models * sizeof(model_info), "model list");

  int actual_total=0;
  g1_quad_object_loader_class loader(g1_object_heap);

  for (int i=0; iupdate(i/(float)model_names.size());

    pf_model_load_open.start();
    i4_file_class *in_file=i4_open(*model_names[i]);
    if (in_file)
    {
      g1_loader_class *fp=g1_open_save_file(in_file);        
      pf_model_load_open.stop();
      if (fp)
      {
        array[actual_total].model=loader.load(fp, *model_names[i], tmap);

        if (array[actual_total].model)
        {
          i4_filename_struct fn;
          i4_split_path(*model_names[i], fn);

          array[actual_total].model->scale(0.1);

          // copy the name into the name buffer
          int len=strlen(fn.filename)+1;
          char *c=(char *)name_buffer->malloc(len, "name");
          strcpy(c, fn.filename);

          array[actual_total].name_start=c;
          actual_total++;        
        }
        delete fp;

      }
      else
        i4_alert(i4gets("old_model_file"),200, model_names[i]);

    }
    else
    {
      pf_model_load_open.stop();
      i4_alert(i4gets("file_missing"), 200, model_names[i]);
    }

  }

  delete stat;

  total_models=actual_total;
  qsort(array, total_models, sizeof(model_info), g1_model_info_compare);


  // reset the model_reference values
  for (g1_model_ref *mi=model_references; mi; mi=mi->next)
    mi->value=find_handle(mi->name);

  pf_load_models.stop();

}

w16 g1_model_list_class::find_handle(const char *name) const
{
  if (!name || !total_models) return 0;

  sw32 lo=0,hi=total_models-1,mid;

  mid=(lo+hi+1)/2;
  while (1)
  {
    int comp=strcmp(array[mid].name_start,name);
    if (comp==0)
      return mid;
    else if (comp<0)
      lo=mid+1;
    else hi=mid-1;
    
    w32 last_mid=mid;
    mid=(hi+lo)/2;

    if (last_mid==mid)
    {
      i4_warning("Unable to find model %s, using default", name);
      return 0;
    }
  }
  return 0;
}

void g1_model_list_class::init()
{
  name_buffer=new i4_grow_heap_class(2048,1024);
  g1_object_heap=new i4_grow_heap_class(1000*1024,0);
  array=0;
}

void g1_model_list_class::uninit()
{
  cleanup();
}