Logo Search packages:      
Sourcecode: plib version File versions

ssg.h

/*
     PLIB - A Suite of Portable Game Libraries
     Copyright (C) 1998,2002  Steve Baker
 
     This library is free software; you can redistribute it and/or
     modify it under the terms of the GNU Library General Public
     License as published by the Free Software Foundation; either
     version 2 of the License, or (at your option) any later version.
 
     This library is distributed in the hope that it will be useful,
     but WITHOUT ANY WARRANTY; without even the implied warranty of
     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
     Library General Public License for more details.
 
     You should have received a copy of the GNU Library General Public
     License along with this library; if not, write to the Free Software
     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
 
     For further information visit http://plib.sourceforge.net

     $Id: ssg.h,v 1.176 2004/12/29 07:19:40 sjbaker Exp $
*/


#ifndef _INCLUDED_SSG_H_
#define _INCLUDED_SSG_H_

#include <stdio.h>
#include "sg.h"

#ifdef UL_MAC_OSX
#  include <OpenGL/gl.h>
#else
#  include <GL/gl.h>
#endif

#include "ssgconf.h"

#ifndef _SSG_PUBLIC
#define _SSG_PUBLIC  protected
#endif

extern int sgebug ;

enum ssgCullResult
{
  SSG_OUTSIDE  = SG_OUTSIDE,
  SSG_INSIDE   = SG_INSIDE,
  SSG_STRADDLE = SG_STRADDLE
} ;

#define SSG_MAXPATH   50

#define SSGTRAV_CULL   1
#define SSGTRAV_ISECT  2
#define SSGTRAV_HOT    4
#define SSGTRAV_LOS    8

class ssgList            ;
class ssgKidList         ;
class ssgBase            ;
class ssgEntity          ;
class ssgLeaf            ;
class ssgVTable          ;
class ssgVtxTable        ;
class ssgVtxArray        ;
class ssgTween           ;
class ssgBranch          ;
class ssgInvisible       ;
class ssgBaseTransform   ;
class ssgTransform       ;
class ssgTexTrans        ;
class ssgCutout          ;
class ssgSelector        ;
class ssgRangeSelector   ;
class ssgTimedSelector   ;
class ssgTweenController ;
class ssgRoot            ;

void  ssgDeRefDelete ( ssgBase *br ) ;

/*
  Think twice before assigning new type bits, it is rather complicated.
  Perhaps some kind of automatic assignment would be good?
 */

#define _SSG_BACKWARDS_REFERENCE 0x0000000  /* For SSG format files */

#define _SSG_TYPE_BASE             0x00000001

/* ssgEntities */
#define _SSG_TYPE_ENTITY           0x00000002
#define _SSG_TYPE_LEAF             0x00000020
#define _SSG_TYPE_VTABLE           0x00000080
#define _SSG_TYPE_VTXTABLE         0x00000100
#define _SSG_TYPE_VTXARRAY         0x00000200
#define _SSG_TYPE_BRANCH           0x00000040
#define _SSG_TYPE_BASETRANSFORM    0x00000080
#define _SSG_TYPE_TRANSFORM        0x00001000
#define _SSG_TYPE_TEXTRANS         0x00002000
#define _SSG_TYPE_AXISTRANSFORM    0x00004000
#define _SSG_TYPE_ANIMTRANSFORM    0x00008000
#define _SSG_TYPE_SELECTOR         0x00000100
#define _SSG_TYPE_RANGESELECTOR    0x00001000
#define _SSG_TYPE_TIMEDSELECTOR    0x00002000
#define _SSG_TYPE_TWEEN            0x00008000
#define _SSG_TYPE_TWEENCONTROLLER  0x00010000
#define _SSG_TYPE_ROOT             0x00000200
#define _SSG_TYPE_CUTOUT           0x00000400
#define _SSG_TYPE_INVISIBLE        0x00000800

/* ssgStates */
#define _SSG_TYPE_STATE            0x00000004
#define _SSG_TYPE_SIMPLESTATE      0x00000020
#define _SSG_TYPE_STATESELECTOR    0x00000040

/* ssgSimpleLists */
#define _SSG_TYPE_SIMPLELIST       0x00000008
#define _SSG_TYPE_VERTEXARRAY      0x00000020
#define _SSG_TYPE_NORMALARRAY      0x00000040
#define _SSG_TYPE_TEXCOORDARRAY    0x00000080
#define _SSG_TYPE_COLOURARRAY      0x00000100
#define _SSG_TYPE_INDEXARRAY       0x00000200
#define _SSG_TYPE_TRANSFORMARRAY   0x00000400
#define _SSG_TYPE_INTERLEAVEDARRAY 0x00000800

/* ssgTextures */
#define _SSG_TYPE_TEXTURE          0x00000010

#define SSG_FILE_VERSION       0x01
#define SSG_FILE_MAGIC_NUMBER  (('S'<<24)+('S'<<16)+('G'<<8)+SSG_FILE_VERSION)


inline int ssgTypeBase         () { return _SSG_TYPE_BASE ; }

inline int ssgTypeEntity       () { return _SSG_TYPE_ENTITY    | ssgTypeBase    () ; }
inline int ssgTypeLeaf         () { return _SSG_TYPE_LEAF      | ssgTypeEntity  () ; }
inline int ssgTypeVTable       () { return _SSG_TYPE_VTABLE    | ssgTypeLeaf    () ; }
inline int ssgTypeVtxTable     () { return _SSG_TYPE_VTXTABLE  | ssgTypeLeaf    () ; }
inline int ssgTypeVtxArray     () { return _SSG_TYPE_VTXARRAY  | ssgTypeVtxTable() ; }
inline int ssgTypeTween        () { return _SSG_TYPE_TWEEN     | ssgTypeVtxTable() ; }
inline int ssgTypeBranch       () { return _SSG_TYPE_BRANCH    | ssgTypeEntity  () ; }
inline int ssgTypeTweenController(){ return _SSG_TYPE_TWEENCONTROLLER | ssgTypeBranch() ; }
inline int ssgTypeBaseTransform() { return _SSG_TYPE_BASETRANSFORM | ssgTypeBranch () ; }
inline int ssgTypeTransform    () { return _SSG_TYPE_TRANSFORM | ssgTypeBaseTransform () ; }
inline int ssgTypeAnimTransform() { return _SSG_TYPE_ANIMTRANSFORM  | ssgTypeTransform () ; }
inline int ssgTypeTexTrans     () { return _SSG_TYPE_TEXTRANS  | ssgTypeBaseTransform () ; }
inline int ssgTypeAxisTransform() { return _SSG_TYPE_AXISTRANSFORM  | ssgTypeTransform () ; }
inline int ssgTypeSelector     () { return _SSG_TYPE_SELECTOR  | ssgTypeBranch  () ; }
inline int ssgTypeRangeSelector() { return _SSG_TYPE_RANGESELECTOR | ssgTypeSelector () ; }
inline int ssgTypeTimedSelector() { return _SSG_TYPE_TIMEDSELECTOR | ssgTypeSelector () ; }
inline int ssgTypeRoot         () { return _SSG_TYPE_ROOT      | ssgTypeBranch  () ; }
inline int ssgTypeCutout       () { return _SSG_TYPE_CUTOUT    | ssgTypeBranch  () ; }
inline int ssgTypeInvisible    () { return _SSG_TYPE_INVISIBLE | ssgTypeBranch  () ; }

inline int ssgTypeState        () { return _SSG_TYPE_STATE     | ssgTypeBase    () ; }
inline int ssgTypeSimpleState  () { return _SSG_TYPE_SIMPLESTATE | ssgTypeState () ; }
inline int ssgTypeStateSelector() { return _SSG_TYPE_STATESELECTOR | ssgTypeSimpleState () ; }

inline int ssgTypeSimpleList   () { return _SSG_TYPE_SIMPLELIST | ssgTypeBase   () ; }
inline int ssgTypeVertexArray  () { return _SSG_TYPE_VERTEXARRAY | ssgTypeSimpleList () ; }
inline int ssgTypeNormalArray  () { return _SSG_TYPE_NORMALARRAY | ssgTypeSimpleList () ; }
inline int ssgTypeTexCoordArray() { return _SSG_TYPE_TEXCOORDARRAY | ssgTypeSimpleList () ; }
inline int ssgTypeColourArray  () { return _SSG_TYPE_COLOURARRAY | ssgTypeSimpleList () ; }
inline int ssgTypeIndexArray   () { return _SSG_TYPE_INDEXARRAY | ssgTypeSimpleList () ; }
inline int ssgTypeTransformArray() { return _SSG_TYPE_TRANSFORMARRAY | ssgTypeSimpleList () ; }
inline int ssgTypeInterleavedArray() { return _SSG_TYPE_INTERLEAVEDARRAY | ssgTypeSimpleList () ; }

inline int ssgTypeTexture      () { return _SSG_TYPE_TEXTURE   | ssgTypeBase    () ; }

/*
  It's critical that these numbers don't change without
  some pretty pressing need because significant change to
  ssgSimpleState.cxx and ssgStateTables.cxx would be needed.
*/

#define SSG_GL_TEXTURE_EN        0
#define SSG_GL_CULL_FACE_EN      1
#define SSG_GL_COLOR_MATERIAL_EN 2
#define SSG_GL_BLEND_EN          3
#define SSG_GL_ALPHA_TEST_EN     4
#define SSG_GL_LIGHTING_EN       5
 
#define SSG_GL_TEXTURE           6
#define SSG_GL_COLOR_MATERIAL    7
#define SSG_GL_DIFFUSE           8
#define SSG_GL_AMBIENT           9
#define SSG_GL_SPECULAR         10
#define SSG_GL_EMISSION         11
#define SSG_GL_SHININESS        12
#define SSG_GL_ALPHA_TEST       13
#define SSG_GL_SHADE_MODEL      14

#define SSG_CLONE_RECURSIVE        1
#define SSG_CLONE_GEOMETRY         2
#define SSG_CLONE_USERDATA         4
#define SSG_CLONE_STATE            8
#define SSG_CLONE_STATE_RECURSIVE 16
#define SSG_CLONE_TEXTURE         32

int ssgGetFrameCounter () ;
void ssgSetFrameCounter ( int fc ) ;

class ssgList
{
protected:
  unsigned int total ;  /* The total number of entities in the list */
  unsigned int limit ;  /* The current limit on number of entities  */
  unsigned int next  ;  /* The next entity when we are doing getNext ops */

  ssgEntity **entity_list ;  /* The list. */

  void sizeChk (void) ;

public:

  ssgList ( int init_max = 1 ) ;
  virtual ~ssgList (void) ;

  ssgEntity *getEntity ( unsigned int n )
  {
    next = n + 1 ;
    return ( n >= total ) ? (ssgEntity *) NULL : entity_list [ n ] ;
  }

  virtual void addEntity ( ssgEntity *entity ) ;
  virtual void removeEntity ( unsigned int n ) ;

  void removeAllEntities () ;

  void removeEntity ( ssgEntity *entity )
  {
    removeEntity ( searchForEntity ( entity ) ) ;
  }

  virtual void replaceEntity ( unsigned int n, ssgEntity *new_entity ) ;

  void replaceEntity ( ssgEntity *old_entity, ssgEntity *new_entity )
  {
    replaceEntity ( searchForEntity ( old_entity ), new_entity ) ;
  }

  int        getNumEntities    (void) { return total ; }
  ssgEntity *getNextEntity     (void) { return getEntity ( next ) ; }
  int        searchForEntity   ( ssgEntity *entity ) ;
} ;


class ssgKidList : public ssgList
{
public:

  ssgKidList ( int init_max = 1 ) ;
  virtual ~ssgKidList (void) ;

  void addEntity ( ssgEntity *entity ) ;
  void removeEntity ( unsigned int n ) ;

  void removeEntity ( ssgEntity *entity )
  {
    removeEntity ( searchForEntity ( entity ) ) ;
  }

  void replaceEntity ( unsigned int n, ssgEntity *new_entity ) ;

  void replaceEntity ( ssgEntity *old_entity, ssgEntity *new_entity )
  {
    replaceEntity ( searchForEntity ( old_entity ), new_entity ) ;
  }
} ;


class ssgBase 
{
  int   refc   ;  /* The number of references to this node */
  int   unique ;  /* A unique number for this node */

protected :

  int   type  ;
  int   spare ;  /* This spare field is used in a bunch of short-term hacks */

  char    *name ;
  ssgBase *user_data ;
  virtual void copy_from ( ssgBase *src, int clone_flags ) ;

public:
  void *operator new  ( size_t size ) ;
  void  operator delete ( void *ptr ) ;

  void deadBeefCheck () { assert ( type != (int) 0xDeadBeef ) ; }

  virtual void zeroSpareRecursive ();
  virtual void zeroSpare ()         ;
  virtual void incSpare  ()         ;
  virtual void setSpare  ( int ss ) ;
  virtual int  getSpare  ()         ;

  int  getUnique () { return unique ; }

  ssgBase (void) ;
  virtual ~ssgBase (void) ;

  void ref   () { refc++ ; }
  void deRef () { assert ( refc > 0 ) ; refc-- ; }
  int  getRef() { return refc ; }

  virtual ssgBase *clone ( int clone_flags = 0 ) ;

  /* Type checking mechanism */

  virtual const char *getTypeName(void) ;

  int   getType    (void)     { return type ; }
  int   isA        ( int ty ) { return getType() == ty ; }
  int   isAKindOf  ( int ty ) { return ( getType() & ty ) == ty ; }

  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;

  ssgBase *getUserData () { return user_data ; }

  void setUserData ( ssgBase *s )
  {
    ssgDeRefDelete ( user_data ) ;

    user_data = s ;
    if ( s != NULL )
      s -> ref () ;
  }

  void  setName ( const char *nm ) ;
  char *getName () { return name ; }
  const char *getPrintableName () { return (name == NULL) ? "NoName" : name ; }
} ;



class ssgSimpleList : public ssgBase
{
protected:

  unsigned int total   ;  /* The total number of things in the list */
  unsigned int limit   ;  /* The current limit on number of things  */
  unsigned int size_of ;  /* The size of each thing */
  char         *list   ;  /* The list. */
  bool         own_mem ;  /* Do we own the list memory ? */

  void sizeChk ( unsigned int n )
  {
    if ( (total+n) > limit )
    {
      if ( ! own_mem )
        ulSetError( UL_FATAL, "ssgSimpleList: Cannot resize array." );

      /* how big should we make the list? */
      limit += limit ;  /* double it */
                  if ( limit <= 0 )
                        limit = 3;
      if ( (total+n) > limit )
        limit = (total+n) ;

      /* re-allocate the list */
      char *nlist = new char [ limit * size_of ] ;
      memmove ( nlist, list, total * size_of ) ;
      delete [] list ;
      list = nlist ;
    }
  }

  virtual void copy_from ( ssgSimpleList *src, int clone_flags ) ;

_SSG_PUBLIC:

  ssgSimpleList ()
  {
    type = ssgTypeSimpleList () ;
    limit = 0 ;
    size_of = 0 ;
    total = 0 ;
    list = NULL ;
    own_mem = true ;
  }

public:

  ssgSimpleList ( int sz, int init = 3, char* things = 0 )
  {
    type = ssgTypeSimpleList () ;
    limit = init ;
    size_of = sz ;
    if ( things )
    {
      total = init ;
      list = things ;
      own_mem = false ;
    }
    else
    {
      total = 0 ;
      list = new char [ limit * size_of ] ;
      own_mem = true ;
    }
  }

  virtual ssgBase *clone ( int clone_flags = 0 ) ;

  virtual ~ssgSimpleList (void)
  {
    if ( own_mem )
      delete [] list ;
    list = NULL ;
  } ;

  char *raw_get ( unsigned int n )
  {
    return ( n >= total ) ? ((char *) 0) : & list [ n * size_of ] ;
  }

  void raw_add ( char *thing )
  {
    sizeChk ( 1 ) ;
    memcpy ( & list [ size_of * total++ ], thing, size_of ) ;
  } ;

  void raw_add ( char *things, unsigned int n )
  {
    sizeChk ( n ) ;
    memcpy ( & list [ size_of * total ], things, size_of * n ) ;
    total += n ;
  }

  void raw_set ( char *thing, unsigned int n  )
  {
    memcpy ( & list [ size_of * n ], thing, size_of ) ;
  } ;

  void removeLast ()
  {
    if ( total > 0 )
      total-- ;
  }

  void removeAll ()
  {
    delete [] list ;
    list = NULL ;
    limit = total = 0 ;
  }

      int compare(ssgSimpleList *other, int print_result=TRUE);
  int getSizeOf (void) { return size_of ; }
  int getNum (void) { return total ; }

  void rawSetNum ( unsigned int n ) /* Better know what you're doing!! */
  {
    total = n ;
  }

  void setNum ( unsigned int n )
  {
    if ( total < n )
    {
      sizeChk ( n ) ;
      memset ( & list [ size_of * total ], 0, size_of * (n-total) ) ;
      total = n ;
    }
  }

  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual const char *getTypeName(void) ;
} ;


class ssgVertexArray : public ssgSimpleList
{
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgVertexArray ( int init = 3, sgVec3* things = 0 )
    : ssgSimpleList ( sizeof(sgVec3), init, (char*)things )
  {
    type = ssgTypeVertexArray () ;
  } 
  float *get ( unsigned int n ) { return (float *) raw_get ( n ) ; }
  void   add ( sgVec3   thing ) { raw_add ( (char *) thing ) ; } ;
  void   set ( sgVec3   thing, unsigned int n ) { raw_set ( (char *) thing, n ) ; } ;
  void   set ( float x, float y, float z, unsigned int n )
  { sgVec3 tmp = { x, y, z } ; raw_set ( (char *) tmp, n ) ; } ;
  void   add ( float x, float y, float z )
  { sgVec3 tmp = { x, y, z } ; raw_add ( (char *) tmp ) ; } ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual const char *getTypeName(void) ;
} ;


class ssgNormalArray : public ssgSimpleList
{
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgNormalArray ( int init = 3, sgVec3* things = 0 )
    : ssgSimpleList ( sizeof(sgVec3), init, (char*)things )
  {
    type = ssgTypeNormalArray () ;
  }
  float *get ( unsigned int n ) { return (float *) raw_get ( n ) ; }
  void   add ( sgVec3   thing ) { raw_add ( (char *) thing ) ; } ;
  void   set ( sgVec3   thing, unsigned int n ) { raw_set ( (char *) thing, n ) ; } ;
  void   set ( float x, float y, float z, unsigned int n )
  { sgVec3 tmp = { x, y, z } ; raw_set ( (char *) tmp, n ) ; } ;
  void   add ( float x, float y, float z )
  { sgVec3 tmp = { x, y, z } ; raw_add ( (char *) tmp ) ; } ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual const char *getTypeName(void) ;
} ;


class ssgTexCoordArray : public ssgSimpleList
{
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgTexCoordArray ( int init = 3, sgVec2* things = 0 )
    : ssgSimpleList ( sizeof(sgVec2), init, (char*)things )
  {
    type = ssgTypeTexCoordArray () ;
  }
  float *get ( unsigned int n ) { return (float *) raw_get ( n ) ; }
  void   add ( sgVec2   thing ) { raw_add ( (char *) thing ) ; } ;
  void   set ( sgVec2   thing, unsigned int n ) { raw_set ( (char *) thing, n ) ; } ;
  void   set ( float u, float v, unsigned int n )
  { sgVec2 tmp = { u, v } ; raw_set ( (char *) tmp, n ) ; } ;
  void   add ( float u, float v )
  { sgVec2 tmp = { u, v } ; raw_add ( (char *) tmp ) ; } ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual const char *getTypeName(void) ;
} ;


class ssgColourArray : public ssgSimpleList
{
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgColourArray ( int init = 3, sgVec4* things = 0 )
    : ssgSimpleList ( sizeof(sgVec4), init, (char*)things )
  {
    type = ssgTypeColourArray () ;
  }
  float *get ( unsigned int n ) { return (float *) raw_get ( n ) ; }
  void   add ( sgVec4   thing ) { raw_add ( (char *) thing ) ; } ;
  void   set ( sgVec4   thing, unsigned int n ) { raw_set ( (char *) thing, n ) ; } ;
  void   set ( float r, float g, float b, float a, unsigned int n )
  { sgVec4 tmp = { r,g,b,a } ; raw_set ( (char *) tmp, n ) ; } ;
  void   add ( float r, float g, float b, float a )
  { sgVec4 tmp = { r,g,b,a } ; raw_add ( (char *) tmp ) ; } ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual const char *getTypeName(void) ;
} ;


class ssgIndexArray : public ssgSimpleList
{
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgIndexArray ( int init = 3, short* things = 0 )
    : ssgSimpleList ( sizeof(short), init, (char*)things )
  {
    type = ssgTypeIndexArray () ;
  }
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  short *get ( unsigned int n ) { return (short *) raw_get ( n ) ; }
  void   add ( short    thing ) { raw_add ( (char *) &thing ) ; } ;
  void   set ( short    thing, unsigned int n ) { raw_set ( (char *) &thing, n ) ; } ;
  virtual const char *getTypeName(void) ;
} ;


class ssgTransformArray : public ssgSimpleList
{
public:
  int selection ;  /* used to remember last transform selected */

  ssgTransformArray ( int init = 3, sgMat4* things = 0 )
    : ssgSimpleList ( sizeof(sgMat4), init, (char*)things )
  {
    type = ssgTypeTransformArray () ;
    selection = 0 ;
  } 
  sgMat4 *get ( unsigned int n ) { return (sgMat4 *) raw_get ( n ) ; }
  void   add ( sgMat4  thing ) { raw_add ( (char *) thing ) ; } ;
  virtual const char *getTypeName(void) ;
} ;


class ssgTexture ;
class ssgTextureArray : private ssgSimpleList
{
public:

  ssgTextureArray ( int init = 3 )
    : ssgSimpleList ( sizeof(ssgTexture*), init )
  {
    //type = ssgTypeTextureArray () ;
  }

  int getNum (void) { return total ; }
  ssgTexture* get ( unsigned int n )
    { return *( (ssgTexture**) raw_get ( n ) ) ; }
  void add ( ssgTexture* thing ) ;
  void removeAll () ;
  ssgTexture* findByFilename ( const char* fname ) ;
} ;


class ssgSimpleState ;
class ssgSimpleStateArray : private ssgSimpleList
{
  void collect_recursive ( ssgEntity *e );

public:

  ssgSimpleStateArray ( int init = 3 )
    : ssgSimpleList ( sizeof(ssgSimpleState*), init )
  {
    //type = ssgTypeSimpleStateArray () ;
  }

  int getNum (void) { return total ; }
  ssgSimpleState* get ( unsigned int n )
  {
    assert(n<total);
    return *( (ssgSimpleState**) raw_get ( n ) ) ;
  }
  void add ( ssgSimpleState* ss ) ;
  void removeAll () ;
  ssgSimpleState* findMatch ( ssgSimpleState* ss ) ;

  void collect ( ssgEntity *e );
      int findIndex ( ssgSimpleState* st ) ;  //returns -1 if not found
} ;


struct ssgInterleavedArrayElement
{
  sgVec2 texCoord ;
  sgVec4 colour ;
  sgVec3 normal ;
  sgVec3 vertex ;
} ;


class ssgInterleavedArray : public ssgSimpleList
{
public:

  ssgInterleavedArray ( int init = 3, ssgInterleavedArrayElement* things = 0 )
    : ssgSimpleList ( sizeof(ssgInterleavedArrayElement), init, (char*)things )
  {
    type = ssgTypeInterleavedArray () ;
  }
  ssgInterleavedArrayElement *get ( unsigned int n ) { return (ssgInterleavedArrayElement *) raw_get ( n ) ; }
  void add ( ssgInterleavedArrayElement  thing ) { raw_add ( (char *) &thing ) ; } ;
  void add ( ssgInterleavedArrayElement *thing ) { raw_add ( (char *)  thing ) ; } ;
  void set ( ssgInterleavedArrayElement  thing, unsigned int n ) { raw_set ( (char *) &thing, n ) ; } ;
  void set ( ssgInterleavedArrayElement *thing, unsigned int n ) { raw_set ( (char *)  thing, n ) ; } ;
  virtual const char *getTypeName(void) ;
} ;


struct ssgTextureInfo
{
      unsigned int width ;
      unsigned int height ;
      unsigned int depth ;
      unsigned int alpha ;
} ;


bool ssgLoadTexture ( const char *fname, ssgTextureInfo* info=0 ) ;
  bool ssgLoadPNG ( const char *fname, ssgTextureInfo* info ) ;
  bool ssgLoadSGI ( const char *fname, ssgTextureInfo* info ) ;
  bool ssgLoadBMP ( const char *fname, ssgTextureInfo* info ) ;
  bool ssgLoadTGA ( const char *fname, ssgTextureInfo* info ) ;
  bool ssgLoadMDLTexture ( const char *fname, ssgTextureInfo* info ) ;
      bool ssgLoadPCX ( const char *fname, ssgTextureInfo* info ) ;

void ssgAddTextureFormat ( const char* extension,
                          bool (*) (const char*, ssgTextureInfo* info) ) ;

int ssgGetNumTexelsLoaded () ;
bool ssgMakeMipMaps ( GLubyte *image, int xsize, int ysize, int zsize ) ;


class ssgTexture : public ssgBase
{
protected:
  char *filename ; 
  int own_handle ;
  GLuint handle ;
      int wrapu, wrapv, mipmap ;
  bool has_alpha ;

  void alloc_handle () ;
  void free_handle () ;

protected:

  virtual void copy_from ( ssgTexture *src, int clone_flags ) ;
      void setDefaultGlParams(int wrapu, int wrapv, int mipmap);

public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;

  ssgTexture () ;

  // This constructor loads the texture from file for you:
  ssgTexture ( const char *fname, int wrapu = TRUE, int wrapv = TRUE,
             int mipmap = TRUE );

  // use this constructor if you already have the texture in memory
  ssgTexture ( const char *fname, GLubyte *image, int xsize, int ysize, int zsize, 
                 int wrapu = TRUE, int wrapv = TRUE);

  virtual ~ssgTexture (void) ;

  bool hasAlpha () const { return has_alpha ; }

  GLuint getHandle () { return handle ; }

  /*
    WARNING - setHandle() IS DEPRECATED
    BECAUSE IT LEAKS texture handles!
  */

  void setHandle ( GLuint handle ) ;

  char *getFilename(void) { return filename ; }

  void  setFilename(const char *fname)
  {
    delete [] filename ;

    if ( fname == NULL )
      filename = NULL ;
    else
      filename = ulStrDup ( fname ) ;
  }

  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual const char *getTypeName(void) ;
} ;

typedef int (*ssgCallback)( ssgEntity * ) ;
#define SSG_CALLBACK_PREDRAW   1
#define SSG_CALLBACK_POSTDRAW  2

typedef int (*ssgTravCallback)( ssgEntity *entity, int traversal_mask ) ;
#define SSG_CALLBACK_PRETRAV   1
#define SSG_CALLBACK_POSTTRAV  2

class ssgState ;

typedef int (*ssgStateCallback)( ssgState *entity ) ;

#define SSG_CALLBACK_PREAPPLY   3

class ssgState : public ssgBase
{
  int translucent ;

  int external_property_index ;

  ssgStateCallback preApplyCB ;
  ssgStateCallback preDrawCB  ;
  ssgStateCallback postDrawCB ;


protected:

  void preApply () ;
  void preDraw  () ;

public:

  virtual void copy_from ( ssgState *src, int clone_flags ) ;
  ssgState (void) ;
  virtual ~ssgState (void) ;

  virtual const char *getTypeName(void) ;

  ssgStateCallback getStateCallback ( int cb_type )
  {
    return ( cb_type == SSG_CALLBACK_PREAPPLY ) ? preApplyCB :
           ( cb_type == SSG_CALLBACK_PREDRAW  ) ? preDrawCB :
                                                  postDrawCB ;
  }
 
  void setStateCallback ( int cb_type, ssgStateCallback cb )
  {
    if ( cb_type == SSG_CALLBACK_PREAPPLY )
      preApplyCB = cb ;
    else
    if ( cb_type == SSG_CALLBACK_PREDRAW )
      preDrawCB = cb ;
    else
      postDrawCB = cb ;
  }

  int  getExternalPropertyIndex (void) { return external_property_index ; }
  void setExternalPropertyIndex ( int i ) { external_property_index = i ; }

  virtual int  isTranslucent (void)  { return translucent ;  }
  virtual void setTranslucent (void) { translucent = TRUE  ; }
  virtual void setOpaque      (void) { translucent = FALSE ; }
  virtual void force (void) = 0 ;
  virtual void apply (void) = 0 ;

  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;


class ssgSimpleState : public ssgState
{
_SSG_PUBLIC:

  int    dont_care ; 
  int    enables   ; 
  ssgTexture* texture ;
  int    mipmap;

  int colour_material_mode ;
  sgVec4 specular_colour ;
  sgVec4 emission_colour ;
  sgVec4  ambient_colour ;
  sgVec4  diffuse_colour ;

  GLenum shade_model ;

  float  shininess ;
  float  alpha_clamp ;

  ssgSimpleState ( int I_am_current_state ) ;

protected:

public:

  virtual void copy_from ( ssgSimpleState *src, int clone_flags ) ;
  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgSimpleState (void) ;
  virtual ~ssgSimpleState (void) ;
  virtual const char *getTypeName(void) ;

  virtual void force (void) ;
  virtual void apply (void) ;

  virtual void      care_about ( int mode ) { dont_care &= ~(1<<mode) ; }
  virtual void dont_care_about ( int mode ) { dont_care |=  (1<<mode) ; }
  virtual int     getCareAbout ( int mode ) { return dont_care & (1<<mode) ; }

  virtual int  isEnabled ( GLenum mode ) ;
  virtual void disable   ( GLenum mode ) ;
  virtual void enable    ( GLenum mode ) ;
  virtual void set       ( GLenum mode, int val )
                                 { val ? enable(mode) : disable(mode) ; }

  virtual GLuint getTextureHandle (void)
  {
    if ( texture != NULL )
      return texture -> getHandle () ;
    return 0 ;
  }

  virtual char *getTextureFilename(void)
  {
    if ( texture != NULL )
      return texture -> getFilename () ;
    return NULL ;
  }

  virtual ssgTexture* getTexture (void)
  {
    return texture ;
  }

  virtual void setTexture ( ssgTexture *tex )
  {
    ssgDeRefDelete ( texture ) ;
    texture = tex ;
    if ( texture != NULL )
    {
      care_about ( SSG_GL_TEXTURE ) ;
      texture -> ref () ;
    }
    else
    {
      dont_care_about ( SSG_GL_TEXTURE ) ;
    }
  }

  virtual void  setTexture ( const char *fname,
    int _wrapu = TRUE, int _wrapv = TRUE, int _mipmap = TRUE )
  {
    mipmap = _mipmap ;
    setTexture ( new ssgTexture ( fname, _wrapu, _wrapv, mipmap ) ) ;
  }
 
  /*
    WARNING - THIS FORM OF setTexture IS DEPRECATED
    BECAUSE IT PREVENTS ssgSave FROM SAVING THE
    TEXTURE FILENAME and it LEAKS texture handles!
  */

  virtual void setTexture ( GLuint tex )
  {
    if ( !texture )
      setTexture ( new ssgTexture ) ;

    texture -> setHandle ( tex ) ;
    texture -> setFilename ( NULL ) ;
  }

  virtual void setTextureFilename ( const char *fname )
  {
    if ( !texture )
      setTexture ( new ssgTexture ) ;

    texture -> setFilename ( fname ) ;
  }

  virtual GLenum getColourMaterial ()
  {
    return (GLenum) colour_material_mode ;
  }

  virtual void setColourMaterial ( GLenum which )
  {
    colour_material_mode = (int) which ;
    care_about ( SSG_GL_COLOR_MATERIAL ) ;
  }

  virtual void setMaterial ( GLenum which, float r, float g,
                                           float b, float a = 1.0f )
  {
    sgVec4 rgba ;
    sgSetVec4 ( rgba, r, g, b, a ) ;
    setMaterial ( which, rgba ) ;
  }

  virtual void setMaterial   ( GLenum which, sgVec4 rgba )
  {
    switch ( which )
    {
      case GL_EMISSION : sgCopyVec4 ( emission_colour, rgba ) ;
                         care_about ( SSG_GL_EMISSION ) ;
                         break ;
      case GL_SPECULAR : sgCopyVec4 ( specular_colour, rgba ) ;
                         care_about ( SSG_GL_SPECULAR ) ;
                         break ;
      case GL_AMBIENT  : sgCopyVec4 ( ambient_colour , rgba ) ;
                         care_about ( SSG_GL_AMBIENT  ) ;
                         break ;
      case GL_DIFFUSE  : sgCopyVec4 ( diffuse_colour , rgba ) ;
                         care_about ( SSG_GL_DIFFUSE  ) ;
                         break ;
      case GL_AMBIENT_AND_DIFFUSE :  // in compliance with glMaterial
                         sgCopyVec4 ( ambient_colour , rgba ) ;
                         sgCopyVec4 ( diffuse_colour , rgba ) ;
                         care_about ( SSG_GL_AMBIENT | SSG_GL_DIFFUSE ) ;
                   break ;
      default :          break ;
    }
  }

  virtual float *getMaterial ( GLenum which )
  {
    switch ( which )
    {
      case GL_EMISSION : return emission_colour ;
      case GL_SPECULAR : return specular_colour ;
      case GL_AMBIENT  : return ambient_colour  ;
      case GL_DIFFUSE  : return diffuse_colour  ;
      default: break ;
    }

    return NULL ;
  }

  virtual float getShininess (void)
  {
    return shininess ;
  }
 
  virtual void setShininess ( float sh )
  {
    care_about ( SSG_GL_SHININESS ) ;
    shininess = sh ;
  }
 
  virtual void setShadeModel ( GLenum model )
  {
    care_about ( SSG_GL_SHADE_MODEL ) ;
    shade_model = model ;
  }
 
  virtual void setAlphaClamp ( float clamp )
  {
    care_about ( SSG_GL_ALPHA_TEST ) ;
    alpha_clamp = clamp ;
  }
 
      /*int getWrapU();
  int getWrapV();*/
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;


class ssgStateSelector : public ssgSimpleState
{
  int              nstates   ;
  int              selection ;
_SSG_PUBLIC:
  ssgSimpleState **statelist ;

protected:
  virtual void copy_from ( ssgStateSelector *src, int clone_flags ) ;

public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgStateSelector () ;
  ssgStateSelector ( int ns ) ;

  virtual ~ssgStateSelector (void) ;
  virtual const char *getTypeName(void) ;

  int             getNumSteps ( void ) { return nstates ; }
  void            selectStep ( unsigned int s ) ;
  unsigned int    getSelectStep   (void) ;
  ssgSimpleState *getCurrentStep  (void) ;
  void            setStep ( int i, ssgSimpleState *step ) ;
  ssgSimpleState *getStep ( int i ) ;

  virtual int  isTranslucent (void)  ;
  virtual void setTranslucent (void) ;
  virtual void setOpaque      (void) ;
  void force (void) ;
  void apply (void) ;

  void      care_about ( int mode ) ;
  void dont_care_about ( int mode ) ;

  int    isEnabled ( GLenum mode ) ;
  void   disable   ( GLenum mode ) ;
  void   enable    ( GLenum mode ) ;

  char *getTextureFilename(void) ;
  void  setTextureFilename(char *fname) ;

  void   setTexture ( char *fname,
                  int _wrapu = TRUE,
                  int _wrapv = TRUE,
                  int _mipmap = TRUE ) ;
  GLuint getTextureHandle (void)   ; 
  void   setTexture ( ssgTexture *tex ) ; 

  /*
    WARNING - THIS FORM OF setTexture IS DEPRECATED
    BECAUSE IT PREVENTS ssgSave FROM SAVING THE
    TEXTURE FILENAME and LEAKS texture handles!
  */

  void   setTexture ( GLuint      tex ) ; 

  void   setColourMaterial(GLenum which); 
  void   setMaterial ( GLenum which, float r, float g,
                                   float b, float a = 1.0f ) ;

  void   setMaterial   ( GLenum which, sgVec4 rgba ) ; 
  float *getMaterial ( GLenum which ) ; 
  float  getShininess (void) ; 
  void   setShininess ( float sh ) ; 
  void   setShadeModel ( GLenum model ) ; 
  void   setAlphaClamp ( float clamp ) ; 
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;

struct ssgEntityBinding
{
  ssgEntity **entity     ;
  char       *nameOrPath ;
} ;


class ssgEntity : public ssgBase
{
  ssgList parents ;

  int traversal_mask ;
  ssgTravCallback  preTravCB ;
  ssgTravCallback postTravCB ;

protected:
  sgSphere bsphere ;
  int bsphere_is_invalid ;

  void emptyBSphere  ()              { bsphere.empty  ()    ; }
  void visualiseBSphere () ;
  void extendBSphere ( sgSphere *s ) { bsphere.extend ( s ) ; }
  void extendBSphere ( sgBox    *b ) { bsphere.extend ( b ) ; }
  void extendBSphere ( sgVec3    v ) { bsphere.extend ( v ) ; }

  virtual ssgCullResult cull_test  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual ssgCullResult isect_test ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual ssgCullResult hot_test   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual ssgCullResult los_test   ( sgVec3     s, sgMat4 m, int test_needed ) ;

  virtual void copy_from ( ssgEntity *src, int clone_flags ) ;
public:
 
  ssgEntity (void) ;
  virtual ~ssgEntity (void) ;
  
  int  getTraversalMask     ()        { return traversal_mask ;}
  void setTraversalMask     ( int t ) { traversal_mask  =  t ; }
  void setTraversalMaskBits ( int t ) { traversal_mask |=  t ; }
  void clrTraversalMaskBits ( int t ) { traversal_mask &= ~t ; }

  ssgTravCallback getTravCallback ( int cb_type )
  {
    return ( cb_type == SSG_CALLBACK_PRETRAV ) ? preTravCB : postTravCB ;
  }

  void setTravCallback ( int cb_type, ssgTravCallback cb )
  {
    if ( cb_type == SSG_CALLBACK_PRETRAV )
      preTravCB = cb ;
    else
      postTravCB = cb ;
  }

  int preTravTests ( int *test_needed, int which ) ;
  void postTravTests ( int which ) ;

  virtual void getNetTransform     ( sgMat4 xform ) ;
  virtual void getLastNetTransform ( sgMat4 xform ) ;

  /* for backward compatibility */
  virtual ssgCallback getCallback ( int cb_type ) ;
  virtual void setCallback ( int cb_type, ssgCallback cb ) ;

  virtual ssgEntity* getByName  ( char *nm ) ;
  virtual ssgEntity* getByPath  ( char *path ) ;
  int  bindEntities ( ssgEntityBinding *bind ) ;

  virtual void recalcBSphere (void) = 0 ;
  int  isDirtyBSphere (void) { return bsphere_is_invalid ; }
  void dirtyBSphere  () ;

  sgSphere *getBSphere ()
  {
    if ( isDirtyBSphere () )
      recalcBSphere () ;

    return & bsphere ;
  }

  virtual int getNumKids (void) { return 0 ; }
  int getNumParents () { return parents.getNumEntities () ; }
  ssgBranch *getParent ( int p ) { return (ssgBranch *) parents.getEntity ( p ) ; }
  ssgBranch *getNextParent () { return (ssgBranch *) parents.getNextEntity () ; }
  void addParent    ( ssgEntity *entity ) { parents.addEntity    ( entity ) ; }
  void removeParent ( ssgEntity *entity ) { parents.removeEntity ( entity ) ; }

  virtual const char *getTypeName(void) ;

  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) = 0 ;
  virtual void isect ( sgSphere  *s, sgMat4 m, int test_needed ) = 0 ;
  virtual void hot   ( sgVec3     s, sgMat4 m, int test_needed ) = 0 ;
  virtual void los   ( sgVec3     s, sgMat4 m, int test_needed ) = 0 ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual void getStats ( int *num_branches, int *num_leaves, int *num_tris, int *num_vertices ) = 0 ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;



class ssgLeaf : public ssgEntity
{
  int cull_face ;
  ssgState *state ;

protected:
  GLenum gltype ;

  ssgCallback  preDrawCB ;
  ssgCallback postDrawCB ;

#ifdef _SSG_USE_DLIST
  GLuint dlist ;
#endif

  virtual void draw_geometry () = 0 ;

  int preDraw () ;

  virtual void copy_from ( ssgLeaf *src, int clone_flags ) ;
public:
  ssgLeaf (void) ;
  virtual ~ssgLeaf (void) ;

  virtual void drawHighlight ( sgVec4 colour ) = 0 ;
  virtual void drawHighlight ( sgVec4 colour, int i ) = 0 ;
  virtual void pick ( int baseName ) = 0 ;

#ifdef _SSG_USE_DLIST
  void makeDList () ;
  void deleteDList () ;
  GLuint getDListIndex () { return dlist ; }
#endif

  int  getExternalPropertyIndex ()
                 { return state ? state->getExternalPropertyIndex() : 0 ; }

  int  isTranslucent () { return state ? state->isTranslucent() : FALSE ; }
  int       hasState () { return state != NULL ; }

  ssgState *getState () { return state ; }
  void      setState ( ssgState *st ); //~~ T.G. Body extended & moved into CXX file

  virtual ssgCallback getCallback ( int cb_type )
  {
    return ( cb_type == SSG_CALLBACK_PREDRAW ) ? preDrawCB : postDrawCB ;
  }

  virtual void setCallback ( int cb_type, ssgCallback cb )
  {
    if ( cb_type == SSG_CALLBACK_PREDRAW )
      preDrawCB = cb ;
    else
      postDrawCB = cb ;
  }

  virtual void setPrimitiveType ( GLenum ty ) { gltype = ty ; }
  virtual GLenum getPrimitiveType () { return gltype ; }

  /* For backwards compatibility only -- DEPRECATED */
  GLenum getGLtype () { return getPrimitiveType() ; }

  virtual int getNumVertices  () { return 0 ; }
  virtual int getNumNormals   () { return 0 ; }
  virtual int getNumColours   () { return 0 ; }
  virtual int getNumTexCoords () { return 0 ; }

  virtual float *getVertex   ( int i ) = 0 ;
  virtual float *getNormal   ( int i ) = 0 ;
  virtual float *getColour   ( int i ) = 0 ;
  virtual float *getTexCoord ( int i ) = 0 ;
  virtual int  getNumTriangles () = 0 ;
  virtual void getTriangle ( int n, short *v1, short *v2, short *v3 ) = 0 ;
  virtual int  getNumLines () = 0 ;
  virtual void getLine ( int n, short *v1, short *v2 ) = 0 ;

  virtual void transform ( const sgMat4 m ) = 0 ;

  void setCullFace ( int cf ) { cull_face = cf ; }
  int  getCullFace () { return cull_face ; }

  virtual void recalcBSphere () = 0 ;
  virtual const char *getTypeName(void) ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual void getStats ( int *num_branches, int *num_leaves, int *num_tris, int *num_vertices ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;

  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual void isect ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void isect_triangles ( sgSphere *s, sgMat4 m, int test_needed ) = 0 ;
  virtual void hot_triangles   ( sgVec3    s, sgMat4 m, int test_needed ) = 0 ;
  virtual void los_triangles   ( sgVec3    s, sgMat4 m, int test_needed ) = 0 ;
  virtual void draw  () = 0 ;
} ;

extern sgVec3 _ssgVertex000   ;
extern sgVec4 _ssgColourWhite ;
extern sgVec3 _ssgNormalUp    ;
extern sgVec2 _ssgTexCoord00  ;
extern short  _ssgIndex0      ;
extern float _ssgGlobTime     ; // used by ssgAnimTransform. Has to be set by the appliation!


class ssgVTable : public ssgLeaf
{
protected:
  sgBox bbox ;
  int indexed ;

  sgVec3 *vertices  ; int num_vertices  ; unsigned short *v_index ;
  sgVec3 *normals   ; int num_normals   ; unsigned short *n_index ;
  sgVec2 *texcoords ; int num_texcoords ; unsigned short *t_index ;
  sgVec4 *colours   ; int num_colours   ; unsigned short *c_index ;

  virtual void draw_geometry () ;
  virtual void copy_from ( ssgVTable *src, int clone_flags ) ;

_SSG_PUBLIC:
  int isIndexed () { return indexed ; }

public:
  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgVTable () ;
  ssgVTable ( GLenum ty,
              int nv, unsigned short *vi, sgVec3 *vl,
              int nn, unsigned short *ni, sgVec3 *nl,
              int nt, unsigned short *ti, sgVec2 *tl,
              int nc, unsigned short *ci, sgVec4 *cl ) ;

  ssgVTable ( GLenum ty,
              int nv, sgVec3 *vl,
              int nn, sgVec3 *nl,
              int nt, sgVec2 *tl,
              int nc, sgVec4 *cl ) ;

  virtual void drawHighlight ( sgVec4 colour ) ;
  virtual void drawHighlight ( sgVec4 colour, int i ) ;
  virtual void pick ( int baseName ) ;
  virtual void transform ( const sgMat4 m ) ;

  int getNumVertices  () { return num_vertices  ; }
  int getNumNormals   () { return num_normals   ; }
  int getNumColours   () { return num_colours   ; }
  int getNumTexCoords () { return num_texcoords ; }
  int getNumTriangles () ;
  void getTriangle ( int n, short *v1, short *v2, short *v3 ) ;
  int  getNumLines () ;
  void getLine ( int n, short *v1, short *v2 ) ;

  void getColourList ( void **list, unsigned short **idx )
  {
    *list = colours ;
    *idx  = c_index  ; 
  }

  void getTexCoordList ( void **list, unsigned short **idx )
  {
    *list = texcoords ;
    *idx  = t_index  ; 
  }

  void getNormalList ( void **list, unsigned short **idx )
  {
    *list = normals ;
    *idx  = n_index  ; 
  }

  void getVertexList ( void **list, unsigned short **idx )
  {
    *list = vertices ;
    *idx  = v_index  ; 
  }

  float *getVertex  (int i){ if(i>=num_vertices)i=num_vertices-1;
                             return (num_vertices<=0) ? _ssgVertex000 :
                                    ((indexed)?vertices [v_index[i]]:vertices [i]);}
  float *getColour  (int i){ if(i>=num_colours)i=num_colours-1;
                             return (num_colours<=0) ? _ssgColourWhite :
                                    ((indexed)?colours  [c_index[i]]:colours  [i]);}
  float *getNormal  (int i){ if(i>=num_normals)i=num_normals-1;
                             return (num_normals<=0) ? _ssgNormalUp :
                                    ((indexed)?normals  [n_index[i]]:normals  [i]);}
  float *getTexCoord(int i){ if(i>=num_texcoords)i=num_texcoords-1;
                             return (num_texcoords<=0) ? _ssgTexCoord00 :
                                    ((indexed)?texcoords[t_index[i]]:texcoords[i]);}


  virtual ~ssgVTable (void) ;

  virtual const char *getTypeName(void) ;
  virtual void recalcBSphere () ;
  virtual void draw () ;

  virtual void isect_triangles ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot_triangles   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los_triangles   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;


class ssgVtxTable : public ssgLeaf
{
protected:
  sgBox bbox ;

  virtual void draw_geometry () ;
  virtual void copy_from ( ssgVtxTable *src, int clone_flags ) ;

_SSG_PUBLIC:

  ssgVertexArray   *vertices  ;
  ssgNormalArray   *normals   ;
  ssgTexCoordArray *texcoords ;
  ssgColourArray   *colours   ;

public:
      int compare(ssgVtxTable *other, int print_result=TRUE);
  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgVtxTable () ;

  ssgVtxTable ( GLenum ty, ssgVertexArray   *vl,
                           ssgNormalArray   *nl,
                           ssgTexCoordArray *tl,
                           ssgColourArray   *cl ) ;

  void textureMe(float uP, float vP) 
      { texcoords = new ssgTexCoordArray;
        for(int i=vertices->getNum()-1; i>=0; i--)
            { sgVec2 v;
                  v[0] = uP; v[1] = vP; 
                  texcoords->add(v);
            }
      }
      virtual void drawHighlight ( sgVec4 colour ) ;
  virtual void drawHighlight ( sgVec4 colour, int i ) ;
  virtual void pick ( int baseName ) ;
  virtual void transform ( const sgMat4 m ) ;

  ssgVertexArray   *getVertices  () { return vertices  ; }
  ssgNormalArray   *getNormals   () { return normals   ; }
  ssgTexCoordArray *getTexCoords () { return texcoords ; }
  ssgColourArray   *getColours   () { return colours   ; }

  virtual void setVertices  ( ssgVertexArray   *vl ) ;
  virtual void setNormals   ( ssgNormalArray   *nl ) ;
  virtual void setTexCoords ( ssgTexCoordArray *tl ) ;
  virtual void setColours   ( ssgColourArray   *cl ) ;

  int getNumVertices  () { return vertices  -> getNum () ; }
  int getNumNormals   () { return normals   -> getNum () ; }
  int getNumColours   () { return colours   -> getNum () ; }
  int getNumTexCoords () { return texcoords -> getNum () ; }

  /* Don't call this unless you know what you are doing!! */
  void rawSetNumVertices ( unsigned int n ) { vertices -> rawSetNum ( n ) ; }

  int  getNumTriangles () ;
  void getTriangle ( int n, short *v1, short *v2, short *v3 ) ;
  int  getNumLines () ;
  void getLine ( int n, short *v1, short *v2 ) ;

  void getVertexList   ( void **list ) { *list = vertices  -> get ( 0 ) ; }
  void getNormalList   ( void **list ) { *list = normals   -> get ( 0 ) ; } 
  void getTexCoordList ( void **list ) { *list = texcoords -> get ( 0 ) ; } 
  void getColourList   ( void **list ) { *list = colours   -> get ( 0 ) ; } 

  float *getVertex  (int i){ int nv=getNumVertices(); if(i>=nv)i=nv-1;
                             return (nv<=0) ? _ssgVertex000:vertices->get(i);}
  float *getNormal  (int i){ int nn=getNumNormals(); if(i>=nn)i=nn-1;
                       return (nn<=0) ? _ssgNormalUp:normals->get(i);}
  float *getTexCoord(int i){ int nc=getNumTexCoords(); if(i>=nc)i=nc-1;
                             return (nc<=0) ? _ssgTexCoord00:texcoords->get(i);}
  float *getColour  (int i){ int nc=getNumColours(); if(i>=nc)i=nc-1;
                       return (nc<=0) ? _ssgColourWhite:colours->get(i);}

      ssgVtxArray *getAs_ssgVtxArray ();

  virtual ~ssgVtxTable (void) ;

  virtual const char *getTypeName(void) ;
  virtual void recalcBSphere () ;
  virtual void draw () ;

  virtual void isect_triangles ( sgSphere *s, sgMat4 m, int test_needed ) ;
  virtual void hot_triangles   ( sgVec3    s, sgMat4 m, int test_needed ) ;
  virtual void los_triangles   ( sgVec3    s, sgMat4 m, int test_needed ) ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2);
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;

} ;


class ssgTween : public ssgVtxTable
{
  virtual void copy_from ( ssgTween *src, int clone_flags ) ;
  void init () ;
protected:
_SSG_PUBLIC:

  int curr_bank ;

  ssgVertexArray   *render_vertices  ;
  ssgNormalArray   *render_normals   ;
  ssgTexCoordArray *render_texcoords ;
  ssgColourArray   *render_colours   ;

  ulList *banked_vertices  ;
  ulList *banked_normals   ;
  ulList *banked_texcoords ;
  ulList *banked_colours   ;

public:
  virtual ssgBase *clone ( int clone_flags = 0 ) ;

  ssgTween () ;
  ssgTween ( GLenum ty ) ;

  virtual void setVertices  ( ssgVertexArray   *vl ) ;
  virtual void setNormals   ( ssgNormalArray   *nl ) ;
  virtual void setTexCoords ( ssgTexCoordArray *tl ) ;
  virtual void setColours   ( ssgColourArray   *cl ) ;

  virtual void recalcBSphere () ;
  virtual void draw () ;
  virtual void transform ( const sgMat4 m ) ;

  virtual const char *getTypeName(void) ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2);
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;

  int getNumBanks () { return banked_vertices -> getNumEntities () ; }

  int  newBank ( ssgVertexArray   *vl, ssgNormalArray   *nl,
                 ssgTexCoordArray *tl, ssgColourArray   *cl ) ;
  int  newBank ( int newVertices , int newNormals,
                 int newTexCoords, int newColours ) ;
  void setBank ( int bank ) ;

  virtual ~ssgTween (void) ;
} ;


class ssgVtxArray : public ssgVtxTable
{
protected:
  ssgIndexArray      *indices;

  virtual void draw_geometry () ;
  virtual void copy_from ( ssgVtxArray *src, int clone_flags ) ;

public:
  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgVtxArray () ;

  ssgVtxArray ( GLenum ty, ssgVertexArray   *vl,
                           ssgNormalArray   *nl,
                           ssgTexCoordArray *tl,
                           ssgColourArray   *cl,
                     ssgIndexArray    *il ) ;

  virtual void drawHighlight ( sgVec4 colour ) ;
  virtual void drawHighlight ( sgVec4 colour, int i ) ;
  virtual void pick ( int baseName ) ;

  void setIndices ( ssgIndexArray *il ) ;
      void addIndex ( short i) { indices->add(i); }

  int getNumIndices () { return indices -> getNum () ; }

  int getNumTriangles () ;
  void getTriangle ( int n, short *v1, short *v2, short *v3 ) ;

  int  getNumLines () ;
  void getLine ( int n, short *v1, short *v2 ) ;

  void getIndexList ( void **list ) { *list = indices  -> get ( 0 ) ; }

  short *getIndex  (int i){ int ni=getNumIndices();if(i>=ni)i=ni-1;
                             return (ni<=0) ? &_ssgIndex0 : indices->get(i);}

      void removeUnusedVertices();
  virtual ~ssgVtxArray (void) ;

  virtual const char *getTypeName(void) ;

  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;


// class ssgVtxInterleavedArray


class ssgBranch : public ssgEntity
{
  ssgKidList kids ;

protected:
  virtual void copy_from ( ssgBranch *src, int clone_flags ) ;

public:
  virtual void zeroSpareRecursive ();

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgBranch (void) ;
  virtual ~ssgBranch (void) ;

  virtual int getNumKids (void)   { return kids.getNumEntities() ; }
  ssgEntity *getKid     ( int n ) { return kids.getEntity  ( n ) ; }
  ssgEntity *getNextKid (void)    { return kids.getNextEntity () ; }
  int        searchForKid ( ssgEntity *entity )
                                  { return kids.searchForEntity(entity); }

  void addKid        ( ssgEntity *entity ) ;
  void removeKid     ( int n ) ;
  void removeKid     ( ssgEntity *entity ) ;
  void removeAllKids (void) ;
  void replaceKid    ( int n, ssgEntity *new_entity ) ;
  void replaceKid    ( ssgEntity *old_entity, ssgEntity *new_entity ) ;

      void mergeHNodes();

  virtual ssgEntity *getByName ( char *match ) ;
  virtual ssgEntity *getByPath ( char *path  ) ;
 
  virtual const char *getTypeName(void) ;
  virtual void cull          ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual void isect         ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot           ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los           ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void print         ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual void getStats ( int *num_branches, int *num_leaves, int *num_tris, int *num_vertices ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual void recalcBSphere () ;
} ;


#define SSGTWEEN_STOP_AT_END    0
#define SSGTWEEN_REPEAT         1

class ssgTweenController : public ssgBranch
{
  float curr_bank ;
  int   mode ;  /* STOP_AT_END or REPEAT */

protected:

  virtual void copy_from ( ssgTweenController *src, int clone_flags ) ;

public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgTweenController (void) ;
  virtual ~ssgTweenController (void) ;

  void  setMode ( int _mode ) { mode = _mode ; }
  int   getMode ()            { return mode ; }

  void  selectBank ( float f ) { curr_bank = f ; }
  float getCurrBank () { return curr_bank ; }

  virtual void cull ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual const char *getTypeName(void) ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;



class ssgInvisible : public ssgBranch
{
protected:

  virtual void copy_from ( ssgInvisible *src, int clone_flags ) ;

public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgInvisible (void) ;
  virtual ~ssgInvisible (void) ;

  virtual void cull ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;



class ssgSelector : public ssgBranch
{
  unsigned char* selection ;

protected:
  int max_kids ;

  virtual void copy_from ( ssgSelector *src, int clone_flags ) ;

public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgSelector ( int max_kids = 32 ) ;
  virtual ~ssgSelector (void) ;

  void selectStep ( unsigned int s )
  {
    memset ( selection, 0, max_kids ) ;
    selection [s] = 1 ;
  }

  void select ( unsigned int s )
  {
    for ( int i=0; i<max_kids && i<32; i++ ) 
      selection [i] = ( (1<<i) & s ) != 0 ; 
  }

  unsigned int getSelect ()
  {
    unsigned int s = 0 ;
    for ( int i=0; i<max_kids && i<32; i++ ) 
      if ( selection [i] ) s |= (1<<i) ;
    return s ;
  }

  int isSelected ( unsigned int i ) const
  {
    return (i<(unsigned int)max_kids) ? selection [ i ] : FALSE ;
  }
  
  int getFirstSelection ()
  {
    int res ;
    
    for ( res = 0 ; res < max_kids && ! selection[res] ; res++ )
      /* Look for first selection */ ;
      
    return ( res < max_kids ) ? res : -1 ;
  }
  
  ssgEntity *getFirstSelectedKid ()
  {
    int k = getFirstSelection () ;
    return (k >= 0) ? getKid ( k ) : NULL ;
  }
  
  int getMaxKids (void) const { return max_kids ; }

  virtual const char *getTypeName(void) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual void isect ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los   ( sgVec3     s, sgMat4 m, int test_needed ) ;
} ;


class ssgRangeSelector : public ssgSelector
{
  int additive ;
  float rng_list [ 33 ] ;

protected:
  virtual void copy_from ( ssgRangeSelector *src, int clone_flags ) ;
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgRangeSelector (void) ;
  virtual ~ssgRangeSelector (void) ;

  void setRanges ( float *ranges, unsigned int nranges )
  {
    for ( unsigned int i = 0 ; i < 33 ; i++ )
      if ( i < nranges )
        rng_list [ i ] = ranges [ i ] ;
      else
        rng_list [ i ] = SG_MAX ;
  }

  void setRange ( unsigned int which, float range )
  {
    if ( which < 33 )
      rng_list [ which ] = range ;
  }

  float *getRanges ()
  {
    return rng_list ;
  }

  float getRange ( unsigned int which )
  {
    return ( which < 33 ) ? rng_list[which] : SG_MAX ;
  }

  void setAdditive ( int add ) { additive = add ; }
  int  isAdditive  () { return additive ; }

  virtual const char *getTypeName(void) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual void isect ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los   ( sgVec3     s, sgMat4 m, int test_needed ) ;
} ;


enum ssgAnimEnum
{
  SSG_ANIM_START,
  SSG_ANIM_STOP,
  SSG_ANIM_PAUSE,
  SSG_ANIM_RESUME
} ;

enum ssgAnimDirection
{
  SSG_ANIM_SWING,
  SSG_ANIM_ONESHOT,
  SSG_ANIM_SHUTTLE
} ;

enum ssgAnimTimeMode
{
    SSG_ANIM_FRAME,
    SSG_ANIM_CLOCK
};


class ssgTimedSelector : public ssgSelector
{
_SSG_PUBLIC:
  ssgAnimEnum      running ;
  ssgAnimDirection mode    ;

  double  start_time ;
  double  pause_time ;
  double  loop_time  ;
  float  *times      ;
  int   curr  ;
  int   start ;
  int   end   ;

protected:
  ssgAnimTimeMode time_mode ;
  static ulClock ck ;

  void compute_loop_time ()
  {
    loop_time = 0 ;

    for ( int k = start ; k <= end ; k++ )
      loop_time += times [ k ] ;
  }

  virtual void copy_from ( ssgTimedSelector *src, int clone_flags ) ;

  double get_time() const
  {
    if (time_mode == SSG_ANIM_FRAME)
      return (double) ssgGetFrameCounter() ;
    else
      return ck.update(), ck.getAbsTime();
  }
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgTimedSelector ( int max_kids = 32 ) ;
  virtual ~ssgTimedSelector (void) ;

  virtual const char *getTypeName(void) ;

  int getStep () ;      

  ssgAnimTimeMode getTimeMode() const { return time_mode; }

  float getDuration ( int i = 0 ) { return times [ i ] ; }

  void setDuration ( float ti, int i = -1, ssgAnimTimeMode m = SSG_ANIM_FRAME )
  {
    time_mode = m;
    if ( i >= 0 && i < max_kids )
      times [ i ] = ti ;
    else
    for ( int j = 0 ; j < max_kids ; j++ )
      times [ j ] = ti ;

    compute_loop_time () ;
  }

  void control ( ssgAnimEnum m )
  {
    compute_loop_time () ;

    if ( m == SSG_ANIM_PAUSE )
    {
      pause_time = get_time() ;
      curr = getStep () ;
    }
    else
    if ( m == SSG_ANIM_RESUME )
    {
      start_time += get_time () - pause_time ;
      
      if ( running != SSG_ANIM_STOP )
        m = SSG_ANIM_START ;
    }
    else
    if ( m == SSG_ANIM_START )
    {
      start_time = get_time () ;
      curr = getStep () ;
    }

    running = m ;
  }

  ssgAnimEnum getControl () { return running ; }

  void setMode ( ssgAnimDirection m ) { mode = m ; }
  ssgAnimDirection getMode () { return mode ; }

  void setLimits ( int st, int en )
  {
    curr  = st ;
    start = st ;
    end   = en ;
    compute_loop_time () ;
  }

  void getLimits ( int *st, int *en )
  {
    if ( st != NULL ) *st = start ;
    if ( en != NULL ) *en = end ;
  }

  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual void isect ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los   ( sgVec3     s, sgMat4 m, int test_needed ) ;
} ;


class ssgBaseTransform : public ssgBranch
{
protected:

  sgMat4 transform ;
  sgMat4 last_transform ;
  int    last_updated ;
  int    first_time ;
  virtual void copy_from ( ssgBaseTransform *src, int clone_flags ) ;
public:

  ssgBaseTransform (void) ;
  virtual ~ssgBaseTransform (void) ;

  void firsttime ()
  {
    if ( first_time )
    {
      first_time = FALSE ;
      updateTransform () ;
    }
  }

  void updateTransform ()
  {
    sgCopyMat4 ( last_transform, transform ) ;
    last_updated = ssgGetFrameCounter () ;
  }

  void getLastTransform ( sgMat4 xform )
  {
    /*
      If the transform was not updated this - or last frame
      then we need to equate the two transforms.
    */

    if ( last_updated < ssgGetFrameCounter () - 1 )
      updateTransform () ;

    sgCopyMat4 ( xform, last_transform ) ;
  }

  void getTransform ( sgMat4 xform )
  {
    sgCopyMat4 ( xform, transform ) ;
  }

  virtual void setTransform ( sgVec3 xyz ) = 0 ;
  virtual void setTransform ( sgCoord *xform ) = 0 ;
  virtual void setTransform ( sgCoord *xform, float sx, float sy, float sz )=0;
  virtual void setTransform ( sgMat4 xform ) = 0 ;

  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;

  virtual const char *getTypeName(void) ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
} ;


class ssgTransform : public ssgBaseTransform
{
protected:
  virtual void copy_from ( ssgTransform *src, int clone_flags ) ;
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgTransform (void) ;
  ssgTransform ( sgCoord *c ) ;
  virtual ~ssgTransform (void) ;

  virtual void setTransform ( sgVec3 xyz ) ;
  virtual void setTransform ( sgCoord *xform ) ;
  virtual void setTransform ( sgCoord *xform, float sx, float sy, float sz ) ;
  virtual void setTransform ( sgMat4 xform ) ;

  virtual void getNetTransform     ( sgMat4 xform ) ;
  virtual void getLastNetTransform ( sgMat4 xform ) ;

  virtual const char *getTypeName(void) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual void isect ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void recalcBSphere () ;
} ;

class ssgAnimTransform  : public ssgTransform 
{
  float curr_bank ;
  int   mode ;  /* SSGTWEEN_STOP_AT_END or SSGTWEEN_REPEAT */
  class ssgTransformArray transformations;

protected:

  virtual void copy_from ( ssgAnimTransform *src, int clone_flags ) ;

public:

  void setNum ( unsigned int n ) { transformations.setNum( n ); }
  void setATransform ( sgMat4 thing, unsigned int n ) { transformations.raw_set ( (char *) thing, n ) ; } ;
  

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgAnimTransform (void) ;
  virtual ~ssgAnimTransform (void) ;

  void  setMode ( int _mode ) { mode = _mode ; }
  int   getMode ()            { return mode ; }

  void  selectBank ( float f ) { curr_bank = f ; }
  float getCurrBank () { return curr_bank ; }

  virtual void cull ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual const char *getTypeName(void) ;
  virtual void print ( FILE *fd = stderr, char *indent = "", int how_much = 2 ) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;


class ssgAxisTransform : public ssgTransform
{
  sgVec3 rotation_axis   ;
  sgVec3 rotation_center ;
  float  limit_low  ;
  float  limit_high ;
protected:
  virtual void copy_from ( ssgAxisTransform *src, int clone_flags ) ;
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgAxisTransform (void) ;
  ssgAxisTransform ( sgVec3 axis, sgVec3 center ) ;
  virtual ~ssgAxisTransform (void) ;

  virtual void  setAxis     ( sgVec3 axis   ) ;
  virtual float *getAxis    () ;
  virtual void  setCenter   ( sgVec3 center ) ;
  virtual float *getCenter  () ;
  virtual void  setRotation ( float rot ) ;

  virtual void setRotationLimits ( float low, float high ) ;
  virtual void setLinearRotation ( float rot ) ;

  virtual const char *getTypeName(void) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
};


class ssgTexTrans : public ssgBaseTransform
{
protected:
  virtual void copy_from ( ssgTexTrans *src, int clone_flags ) ;
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgTexTrans (void) ;
  ssgTexTrans ( sgCoord *c ) ;
  virtual ~ssgTexTrans (void) ;

  virtual void setTransform ( sgVec3 xyz ) ;
  virtual void setTransform ( sgCoord *xform ) ;
  virtual void setTransform ( sgCoord *xform, float sx, float sy, float sz ) ;
  virtual void setTransform ( sgMat4 xform ) ;

  virtual const char *getTypeName(void) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
} ;


class ssgCutout : public ssgBranch
{
  int point_rotate ;
protected:
  virtual void copy_from ( ssgCutout *src, int clone_flags ) ;
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgCutout (int pntrot=FALSE) ;
  virtual ~ssgCutout (void) ;

  int isPointRotate (void) { return point_rotate ; }

  virtual const char *getTypeName(void) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
  virtual void cull  ( sgFrustum *f, sgMat4 m, int test_needed ) ;
  virtual void isect ( sgSphere  *s, sgMat4 m, int test_needed ) ;
  virtual void hot   ( sgVec3     s, sgMat4 m, int test_needed ) ;
  virtual void los   ( sgVec3     s, sgMat4 m, int test_needed ) ;
} ;


class ssgRoot : public ssgBranch
{
protected:
  virtual void copy_from ( ssgRoot *src, int clone_flags ) ;
public:

  virtual ssgBase *clone ( int clone_flags = 0 ) ;
  ssgRoot (void) ;
  virtual ~ssgRoot (void) ;
  virtual const char *getTypeName(void) ;
  virtual int load ( FILE *fd ) ;
  virtual int save ( FILE *fd ) ;
} ;


class ssgLight
{
  int id ;
  int is_headlight ;
  int is_turned_on ;

  sgVec4 ambient ;
  sgVec4 diffuse ;
  sgVec4 specular ;

  sgVec4 position ;

  sgVec3 direction ;
  float  exponent ;
  float  cutoff ;
  sgVec3 atten ;

public:

  ssgLight ()
  {
    id = 0 ;
    is_turned_on = FALSE ;
    is_headlight = FALSE ;
    sgSetVec4 ( position, 0.0f, 0.0f, 1.0f, 0.0f ) ;
    sgSetVec4 ( ambient , 0.2f, 0.2f, 0.2f, 1.0f ) ;
    sgSetVec4 ( diffuse , 1.0f, 1.0f, 1.0f, 1.0f ) ;
    sgSetVec4 ( specular, 1.0f, 1.0f, 1.0f, 1.0f ) ;
    sgSetVec3 ( direction, 0.0f, 0.0f, -1.0f ) ;
    exponent = 1.0f;
    cutoff   = 90.0f;
    sgSetVec3 ( atten, 1.0f, 0.0f, 0.0f ) ;
  }

  void setID ( int i ) { id = i ; }
  int  isOn () { return is_turned_on  ; }
  void on   () { is_turned_on = TRUE  ; }
  void off  () { is_turned_on = FALSE ; }

  void setPosition ( const sgVec3 pos ) { sgCopyVec3 ( position, pos ) ; }
  void getPosition ( sgVec3 pos )       { sgCopyVec3 ( pos, position ) ; }
  void setPosition ( float x, float y, float z ) { sgSetVec3 ( position, x, y, z ) ; }

  void setColour   ( GLenum which, const sgVec4 col )
  {
    switch ( which )
    {
      case GL_AMBIENT  : sgCopyVec4 ( ambient , col ) ; break ;
      case GL_DIFFUSE  : sgCopyVec4 ( diffuse , col ) ; break ;
      case GL_SPECULAR : sgCopyVec4 ( specular, col ) ; break ;
      default : break ;
    }
  }

  void getColour   ( GLenum which, sgVec4 col )
  {
    switch ( which )
    {
      case GL_AMBIENT  : sgCopyVec4 ( col , ambient ) ; break ;
      case GL_DIFFUSE  : sgCopyVec4 ( col , diffuse ) ; break ;
      case GL_SPECULAR : sgCopyVec4 ( col, specular ) ; break ;
      default : break ;
    }
  }

  void setColour   ( GLenum which, float r, float g, float b )
  {
    sgVec4 colour = { r, g, b, 1.0f };
    setColour(which, colour);
  }

  void setHeadlight ( int head ) { is_headlight = head ; }
  int  isHeadlight () { return is_headlight ; }

  void setSpotlight ( int spot ) { position[3] = ( spot ? 1.0f : 0.0f ) ; }
  int  isSpotlight () { return position[3] != 0.0f; }
  
  void setSpotDirection ( const sgVec3 dir ) { sgCopyVec3 ( direction, dir ) ; }
  void getSpotDirection ( sgVec3 dir )       { sgCopyVec3 ( dir, direction ) ; }
  void setSpotDirection ( float x, float y, float z ) { sgSetVec3 ( direction, x, y, z ) ; }

  void setSpotDiffusion ( float exp, float cut = 90.0f )
  {
    exponent = exp;
    cutoff   = cut;
  }

  void getSpotDiffusion ( float *exp, float *cut = 0 )
  {
    if (exp != 0) *exp = exponent;
    if (cut != 0) *cut = cutoff;
  }

  void setSpotAttenuation ( float constant, float linear, float quadratic )
  {
    sgSetVec3( atten, constant, linear, quadratic );
  }

  void getSpotAttenuation ( float *constant, float *linear, float *quadratic )
  {
    if (constant  != 0) *constant  = atten[0];
    if (linear    != 0) *linear    = atten[1];
    if (quadratic != 0) *quadratic = atten[2];
  }

  void setup ()
  {
    if ( is_turned_on )
    {
      glEnable  ( (GLenum)(GL_LIGHT0+id) ) ;
      glLightfv ( (GLenum)(GL_LIGHT0+id), GL_AMBIENT , ambient  ) ;
      glLightfv ( (GLenum)(GL_LIGHT0+id), GL_DIFFUSE , diffuse  ) ;
      glLightfv ( (GLenum)(GL_LIGHT0+id), GL_SPECULAR, specular ) ;
      glLightfv ( (GLenum)(GL_LIGHT0+id), GL_POSITION, position ) ;
      if ( isSpotlight() ) {
       glLightfv ( (GLenum)(GL_LIGHT0+id), GL_SPOT_DIRECTION, direction ) ;
       glLightf  ( (GLenum)(GL_LIGHT0+id), GL_SPOT_EXPONENT,  exponent  ) ;
       glLightf  ( (GLenum)(GL_LIGHT0+id), GL_SPOT_CUTOFF,    cutoff    ) ;
       glLightf  ( (GLenum)(GL_LIGHT0+id), GL_CONSTANT_ATTENUATION,  atten[0] );
       glLightf  ( (GLenum)(GL_LIGHT0+id), GL_LINEAR_ATTENUATION,    atten[1] );
       glLightf  ( (GLenum)(GL_LIGHT0+id), GL_QUADRATIC_ATTENUATION, atten[2] );
      }
    }
    else
      glDisable ( (GLenum)(GL_LIGHT0+id) ) ;
  }

} ;

class ssgHit
{
_SSG_PUBLIC:

  int num_entries ;
  ssgEntity *path [ SSG_MAXPATH ] ;

public:
  ssgLeaf *leaf ;
  int      triangle ;
  sgVec4   plane ;
  sgMat4   matrix ;

  ssgHit ()
  {
    leaf = NULL ;
    init () ;
  } ;

  void init () { num_entries = 0 ; }

  void addPath ( ssgEntity *e )
  {
    if ( num_entries < SSG_MAXPATH )
      path [ num_entries++ ] = e ;
  }
 
  int getNumPathEntries () { return num_entries ; }

  ssgEntity *getPathEntry ( int i )
  {
    return ( i >= 0 && i < num_entries ) ? path[i] : (ssgEntity *) NULL ;
  }
} ;

class ssgContext
{
 protected:

  int enabledClipPlanes ;
  sgVec4 clipPlane [ 6 ] ;
  ssgSimpleState *currentState ;
  ssgSimpleState *basicState   ;
  sgFrustum      *frustum      ;

  sgMat4    cameraMatrix       ;
  int       cullFace           ;
  int       ovTexture          ;
  int       ovCullface         ;
  ssgState *ovState            ;

public:


   ssgContext () ;
  ~ssgContext () ;

  void forceBasicState () ;

  void applyClipPlanes () ;
  void removeClipPlanes () ;

  void makeCurrent () ;
  int    isCurrent () ;

  void clrClipPlane ( int i )
  {
    if ( i >= 0 && i < 6 )
      enabledClipPlanes &= ~(1<<i) ;
  }

  void setClipPlane ( int i, sgVec4 plane )
  {
    if ( i >= 0 && i < 6 )
    {
      sgCopyVec4 ( clipPlane [ i ], plane ) ;
      enabledClipPlanes |= (1<<i) ;
    }
  }

  int  isSetClipPlane ( int i ) { return enabledClipPlanes & (1<<i) ; }
  float *getClipPlane ( int i )
  {
    return ((i<0)||(i>=6)) ? NULL : clipPlane [i] ;
  }

  void overrideState     ( ssgState *s ) ;
  void overrideTexture   ( int on_off ) ;
  void overrideCullface  ( int on_off ) ;

  void setCullface       ( int on_off )
  {
    if ( cullFace == on_off || ovCullface )
      return ;

    cullFace = on_off ;

    if ( cullFace ) glEnable  ( GL_CULL_FACE ) ;
               else glDisable ( GL_CULL_FACE ) ;
  }

  ssgState *overriddenState () { return ovState ; }
  int  stateOverridden    () { return ovState != NULL ; }
  int  textureOverridden  () { return ovTexture  ; }
  int  cullfaceOverridden () { return ovCullface ; }
  int  cullfaceIsEnabled  () { return cullFace   ; }

  sgFrustum *getFrustum () { return frustum ; }

  // make a perspective projection
  void setFrustum ( float l, float r, float b, float t, float n, float f ) ;

  // make an orthographic projection
  void setOrtho   ( float l, float r, float b, float t, float n, float f ) ;
  
  void getNearFar ( float *n, float *f ) ;
  void getFOV     ( float *w, float *h ) ;
  void getOrtho   ( float *w, float *h ) ;

  void setNearFar ( float  n, float  f ) ;
  void setOrtho   ( float  w, float  h ) ; // make orthographic
  void setFOV     ( float  w, float  h ) ; // make perspective

  int  isOrtho () { return frustum -> isOrtho () ; } // is orthographic

  ssgSimpleState *getState () { return currentState ; }
  void cull ( ssgBranch *r ) ;

  void getCameraPosition ( sgVec3   pos ) ;
  void setCamera ( sgMat4 mat ) ;
  void setCamera ( const sgCoord *coord ) ;
  void setCameraLookAt ( const sgVec3 eye, const sgVec3 center, const sgVec3 up ) ;
  void setCameraLookAt ( const sgVec3 eye, const sgVec3 center ) ;

  void loadProjectionMatrix () ;
  void loadModelviewMatrix  () ;

  void getProjectionMatrix  ( sgMat4 dst ) ;
  void getModelviewMatrix   ( sgMat4 dst ) ;

  void pushProjectionMatrix () ;
  void pushProjectionMatrix ( sgFrustum *f ) ;
  void loadModelviewMatrix  ( sgMat4 mat ) ;

} ;



/* Compatibility inlines for pre-ssgContext applications */

extern ssgContext *_ssgCurrentContext ;

inline ssgContext *ssgGetCurrentContext () { return _ssgCurrentContext ; }

inline void ssgGetCameraPosition ( sgVec3 pos )
{
  _ssgCurrentContext -> getCameraPosition ( pos ) ;
}

inline void ssgOverrideState ( ssgState *s )
{
  _ssgCurrentContext->overrideState ( s ) ;
}

inline void ssgOverrideTexture ( int on_off )
{
  _ssgCurrentContext->overrideTexture ( on_off ) ;
}

inline void ssgOverrideCullface ( int on_off )
{
  _ssgCurrentContext->overrideCullface ( on_off ) ;
}

inline void ssgGetNearFar ( float *n, float *f )
{
  _ssgCurrentContext->getNearFar ( n, f ) ;
}

inline void ssgGetFOV ( float *w, float *h )
{
  _ssgCurrentContext->getFOV ( w, h ) ;
}

inline void ssgGetOrtho ( float *w, float *h )
{
  _ssgCurrentContext->getOrtho ( w, h ) ;
}

inline void ssgSetFrustum ( float l, float r, float b, float t, float n, float f )
{
  _ssgCurrentContext->setFrustum ( l, r, b, t, n, f ) ;
}

inline void ssgSetOrtho ( float l, float r, float b, float t, float n, float f )
{
  _ssgCurrentContext->setOrtho ( l, r, b, t, n, f ) ;
}

inline void ssgSetFOV ( float w, float h )
{
  _ssgCurrentContext->setFOV ( w, h ) ;
}

inline void ssgSetOrtho ( float w, float h )
{
  _ssgCurrentContext->setOrtho ( w, h ) ;
}

inline void ssgSetNearFar ( float n, float f )
{
  _ssgCurrentContext->setNearFar ( n, f ) ;
}

inline void ssgSetCamera ( sgMat4 mat )
{
  _ssgCurrentContext-> setCamera ( mat ) ;
}

inline void ssgSetCamera ( sgCoord *coord )
{
  _ssgCurrentContext-> setCamera ( coord ) ;
}

inline void ssgSetCameraLookAt ( const sgVec3 eye, const sgVec3 center, const sgVec3 up )
{
  _ssgCurrentContext-> setCameraLookAt ( eye, center, up ) ;
}

inline void ssgSetCameraLookAt ( const sgVec3 eye, const sgVec3 center )
{
  _ssgCurrentContext-> setCameraLookAt ( eye, center ) ;
}

inline void ssgLoadProjectionMatrix ()
{
  _ssgCurrentContext->loadProjectionMatrix () ;
}

inline void ssgLoadProjectionMatrix ( sgFrustum *f )
{
  glLoadIdentity () ;
  _ssgCurrentContext->pushProjectionMatrix ( f ) ;
}

inline void ssgGetProjectionMatrix ( sgMat4 dst )
{
  _ssgCurrentContext->getProjectionMatrix ( dst ) ;
}

inline void ssgGetModelviewMatrix ( sgMat4 dst )
{
  _ssgCurrentContext->getModelviewMatrix ( dst ) ;
}

inline void ssgLoadModelviewMatrix ()
{
  _ssgCurrentContext->loadModelviewMatrix () ;
}

inline void ssgLoadModelviewMatrix ( sgMat4 mat )
{
  _ssgCurrentContext->loadModelviewMatrix ( mat ) ;
}

inline void ssgForceBasicState ()
{
  _ssgCurrentContext -> forceBasicState () ;
}

inline  sgFrustum *ssgGetFrustum ()
{
  return _ssgCurrentContext -> getFrustum() ;
}

void ssgInit () ;

void ssgCullAndDraw ( ssgBranch *root ) ;
void ssgCullAndPick ( ssgBranch *root, sgVec2 botleft, sgVec2 topright ) ;
int  ssgIsect       ( ssgBranch *root, sgSphere*s,sgMat4 m, ssgHit **results ) ;
int  ssgHOT         ( ssgBranch *root, sgVec3  s, sgMat4 m, ssgHit **results ) ;
int  ssgLOS         ( ssgBranch *root, sgVec3  s, sgMat4 m, ssgHit **results ) ;

/* Load/Save functions */

enum {
      SSG_MD2_STAND,
      SSG_MD2_RUN,
      SSG_MD2_ATTACK,
      SSG_MD2_PAIN_1,
      SSG_MD2_PAIN_2,
      SSG_MD2_PAIN_3,
      SSG_MD2_JUMP,
      SSG_MD2_FLIPOFF,
      SSG_MD2_SALUTE,
      SSG_MD2_TAUNT,
      SSG_MD2_WAVE,
      SSG_MD2_POINT,
      SSG_MD2_CROUCH_STAND,
      SSG_MD2_CROUCH_WALK,
      SSG_MD2_CROUCH_PAIN,
      SSG_MD2_CROUCH_DEATH,
      SSG_MD2_DEATH_1,
      SSG_MD2_DEATH_2,
      SSG_MD2_DEATH_3,
      SSG_MD2_ALL,
      SSG_MD2_POSE
};

class ssgLoaderOptions
{
  //NOTES: we could add more later
  //...model scale factor...
  //...create normals or read them from the file?...
  //...cutoff angle for smooth shading where there isn`t one in the file format...
  //...which direction is front-facing in formats that don`t get it right...
  //...etc...

protected:


  /* for backward compatibility */
  ssgState *(*create_state_cb)( char *) ;
  ssgBranch *(*create_branch_cb)(char *) ;

  char* model_dir ;
  char* texture_dir ;

  char* make_path ( char* path, const char* dir, const char* fname ) const ;

public:
  ssgSimpleStateArray shared_states ;
  ssgTextureArray shared_textures ;

  ssgLoaderOptions ()
  {
    model_dir = 0 ;
    texture_dir = 0 ;

    /* for backward compatibility */
    create_state_cb = 0 ;
    create_branch_cb = 0 ;
  }

  virtual ~ssgLoaderOptions()
  {
    if (model_dir)
    {
      delete [] model_dir;
      model_dir = 0;
    }
  
    if (texture_dir)
    {
      delete [] texture_dir;
      texture_dir = 0;
    }
  }

  const char* getModelDir ( void ) const { return model_dir ; }
  const char* getTextureDir ( void ) const { return texture_dir ; }
  void setModelDir ( const char *s ) ;
  void setTextureDir ( const char *s ) ;

  //
  // the idea is that you derive a class from ssgLoaderOptions and
  // override these methods to customize how the loaders work
  //
  virtual void makeModelPath ( char* path, const char *fname ) const ;
  virtual void makeTexturePath ( char* path, const char *fname ) const ;

  virtual ssgLeaf* createLeaf ( ssgLeaf* leaf, const char* parent_name ) ;
  virtual ssgTexture* createTexture ( char* tfname, 
                        int wrapu  = TRUE, int wrapv = TRUE, 
                        int mipmap = TRUE ) ;
  virtual ssgTransform* createTransform ( ssgTransform* tr,
      ssgTransformArray* ta ) const ;
  virtual ssgSelector* createSelector ( ssgSelector* s ) const ;
  virtual ssgBranch* createBranch ( char* text ) const
  {
    if ( create_branch_cb != NULL )
      return (*create_branch_cb)(text) ;
    return NULL ;
  }
  virtual ssgState* createState ( char* tfname ) const
  {
    if ( create_state_cb != NULL )
      return (*create_state_cb)(tfname) ;
    return NULL ;
  }
  virtual ssgSimpleState* createSimpleState ( char* tfname ) const
  {
    ssgState *st = createState ( tfname ) ;
    if ( st != NULL )
    {
      if ( st -> isAKindOf ( ssgTypeSimpleState() ) )
      {
        return (ssgSimpleState*) st ;
      }
      else
      {
        ulSetError(UL_WARNING, "createState() did not return simple state");
      }
    }
    return NULL ;
  }

  virtual void endLoad ()
  {
    /* default behavior is to reset sharing after loading a model */
    shared_textures.removeAll () ;
    shared_states.removeAll () ;
  }

  //
  // *** DEPRECATED INTERFACE ***
  // for backward compatibility
  // derive and use virtual methods to override
  //
  void setCreateBranchCallback ( ssgBranch *(*cb)(char *) )
  {
    create_branch_cb = cb ;
  }
  void setCreateStateCallback ( ssgState *(*cb)(char *) )
  {
    create_state_cb = cb ;
  }
} ;


int        ssgSave     ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveAC   ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveASE  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveSSG  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveDXF  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveTRI  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveOBJ  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveX    ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveM    ( const char *fname, ssgEntity *ent ) ;
int        ssgSave3ds  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveFLT  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveOFF  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveQHI  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveATG  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveVRML1( const char *fname, ssgEntity *ent ) ;
int        ssgSaveASC  ( const char *fname, ssgEntity *ent ) ;
int        ssgSaveIV   ( const char *fname, ssgEntity *ent ) ;
int                      ssgSavePOV  ( const char *fname, ssgEntity *ent ) ;


ssgEntity *ssgLoad     ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoad3ds  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadAC3D ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadASC  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadSSG  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadASE  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadDOF  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadDXF  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadTRI  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadOBJ  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadMD2  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadMDL  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadX    ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadFLT  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadM    ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadStrip( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadOFF  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadATG  ( const char *fname, const ssgLoaderOptions *options = NULL ) ;
ssgEntity *ssgLoadVRML1( const char *fname, const ssgLoaderOptions* options = NULL ) ;
ssgEntity *ssgLoadIV   ( const char *fname, const ssgLoaderOptions* options = NULL ) ;
ssgEntity *ssgLoadXPlaneOBJ ( const char *fname, const ssgLoaderOptions *options = NULL ) ;


typedef ssgEntity *ssgLoadFunc ( const char *, const ssgLoaderOptions * ) ;
typedef int        ssgSaveFunc ( const char *, ssgEntity * ) ;

void ssgAddModelFormat ( const char* extension,
                        ssgLoadFunc *loadfunc , ssgSaveFunc  *savefunc ) ;

/* current loader options */

extern ssgLoaderOptions* _ssgCurrentOptions ;

inline ssgLoaderOptions* ssgGetCurrentOptions ()
{
  return _ssgCurrentOptions ;
}

inline void ssgSetCurrentOptions ( ssgLoaderOptions* options )
{
  if ( options != NULL )
    _ssgCurrentOptions = options ;
}

/* For backwards compatibility */

inline ssgEntity *ssgLoad ( char *fname, ssgBranch *(*cb)(char *))
{
  _ssgCurrentOptions -> setCreateBranchCallback ( cb ) ;
  return ssgLoad ( fname ) ;
}

inline void ssgSetAppStateCallback ( ssgState *(*cb)(char *) )
{
  _ssgCurrentOptions -> setCreateStateCallback ( cb ) ;
}

inline void ssgModelPath   ( const char *path )
{
  _ssgCurrentOptions -> setModelDir ( path ) ;
}

inline void ssgTexturePath ( const char *path )
{
  _ssgCurrentOptions -> setTextureDir ( path ) ;
}

ssgEntity *ssgLoadAC   ( const char *fname, const ssgLoaderOptions *options = NULL ) ;


void ssgGetValuesFromLastATGFile(double *x, double *y, double *z, double *r);

 bool ssgConvertTexture( char * fname_output, char * fname_input ) ;


class ssgStatistics
{
  int vertex_count ;
  int leaf_count   ;

public:

  void reset () ;

  void bumpVertexCount ( int i ) { vertex_count += i ; }
  void bumpLeafCount   ( int i ) {   leaf_count += i ; }

  int getVertexCount () { return vertex_count ; }
  int getLeafCount   () { return leaf_count ; }

  ssgStatistics ()
  {
    reset () ;
  }
} ;

ssgStatistics *ssgGetLatestStatistics () ;

/* scene walkers */

void ssgFlatten  ( ssgEntity *ent ) ;
void ssgStripify ( ssgEntity *ent ) ;
void ssgArrayTool ( ssgEntity *ent, float* vtol = 0, bool make_normals = false ) ;
void ssgTransTool ( ssgEntity *ent, const sgMat4 trans ) ;

ssgLight *ssgGetLight ( int i ) ;

char *ssgShowStats () ;
void  ssgDelete ( ssgBranch *br ) ;
const char *ssgGetVersion () ;

void ssgSetLoadOFFTranslucent ( int i );

void ssgRegisterType ( int type, ssgBase * ( *create_func ) () ) ;
ssgBase *ssgCreateOfType ( int type ) ;

#define SSG_BACKFACE_COLLISIONS_SUPPORTED 1
void ssgSetBackFaceCollisions ( bool b ) ;

#endif


Generated by  Doxygen 1.6.0   Back to index