#ifndef __STATICLIST_H__ #define __STATICLIST_H__ /* =============================================================================== Static list template A non-growing, memset-able list using no memory allocation. =============================================================================== */ template class idStaticList { public: idStaticList(); idStaticList( const idStaticList &other ); ~idStaticList( void ); void Clear( void ); // marks the list as empty. does not deallocate or intialize data. int Num( void ) const; // returns number of elements in list int Max( void ) const; // returns the maximum number of elements in the list void SetNum( int newnum ); // set number of elements in list size_t Allocated( void ) const; // returns total size of allocated memory size_t Size( void ) const; // returns total size of allocated memory including size of list type size_t MemoryUsed( void ) const; // returns size of the used elements in the list const type & operator[]( int index ) const; type & operator[]( int index ); type * Ptr( void ); // returns a pointer to the list const type * Ptr( void ) const; // returns a pointer to the list type * Alloc( void ); // returns reference to a new data element at the end of the list. returns NULL when full. int Append( const type & obj ); // append element int Append( const idStaticList &other ); // append list int AddUnique( const type & obj ); // add unique element int Insert( const type & obj, int index ); // insert the element at the given index int FindIndex( const type & obj ) const; // find the index for the given element type * Find( type const & obj ) const; // find pointer to the given element int FindNull( void ) const; // find the index for the first NULL pointer in the list int IndexOf( const type *obj ) const; // returns the index for the pointer to an element in the list bool RemoveIndex( int index ); // remove the element at the given index bool Remove( const type & obj ); // remove the element void Swap( idStaticList &other ); // swap the contents of the lists void DeleteContents( bool clear ); // delete the contents of the list private: int num; type list[ size ]; }; /* ================ idStaticList::idStaticList() ================ */ template ID_INLINE idStaticList::idStaticList() { num = 0; } /* ================ idStaticList::idStaticList( const idStaticList &other ) ================ */ template ID_INLINE idStaticList::idStaticList( const idStaticList &other ) { *this = other; } /* ================ idStaticList::~idStaticList ================ */ template ID_INLINE idStaticList::~idStaticList( void ) { } /* ================ idStaticList::Clear Sets the number of elements in the list to 0. Assumes that type automatically handles freeing up memory. ================ */ template ID_INLINE void idStaticList::Clear( void ) { num = 0; } /* ================ idStaticList::DeleteContents Calls the destructor of all elements in the list. Conditionally frees up memory used by the list. Note that this only works on lists containing pointers to objects and will cause a compiler error if called with non-pointers. Since the list was not responsible for allocating the object, it has no information on whether the object still exists or not, so care must be taken to ensure that the pointers are still valid when this function is called. Function will set all pointers in the list to NULL. ================ */ template ID_INLINE void idStaticList::DeleteContents( bool clear ) { int i; for( i = 0; i < size; i++ ) { delete list[ i ]; list[ i ] = NULL; } if ( clear ) { Clear(); } else { memset( list, 0, sizeof( list ) ); } } /* ================ idStaticList::Num Returns the number of elements currently contained in the list. ================ */ template ID_INLINE int idStaticList::Num( void ) const { return num; } /* ================ idStaticList::Num Returns the maximum number of elements in the list. ================ */ template ID_INLINE int idStaticList::Max( void ) const { return size; } /* ================ idStaticList::Allocated ================ */ template ID_INLINE size_t idStaticList::Allocated( void ) const { return size * sizeof( type ); } /* ================ idStaticList::Size ================ */ template ID_INLINE size_t idStaticList::Size( void ) const { return sizeof( idStaticList ) + Allocated(); } /* ================ idStaticList::Num ================ */ template ID_INLINE size_t idStaticList::MemoryUsed( void ) const { return num * sizeof( list[ 0 ] ); } /* ================ idStaticList::SetNum Set number of elements in list. ================ */ template ID_INLINE void idStaticList::SetNum( int newnum ) { assert( newnum >= 0 ); assert( newnum <= size ); num = newnum; } /* ================ idStaticList::operator[] const Access operator. Index must be within range or an assert will be issued in debug builds. Release builds do no range checking. ================ */ template ID_INLINE const type &idStaticList::operator[]( int index ) const { assert( index >= 0 ); assert( index < num ); return list[ index ]; } /* ================ idStaticList::operator[] Access operator. Index must be within range or an assert will be issued in debug builds. Release builds do no range checking. ================ */ template ID_INLINE type &idStaticList::operator[]( int index ) { assert( index >= 0 ); assert( index < num ); return list[ index ]; } /* ================ idStaticList::Ptr Returns a pointer to the begining of the array. Useful for iterating through the list in loops. Note: may return NULL if the list is empty. FIXME: Create an iterator template for this kind of thing. ================ */ template ID_INLINE type *idStaticList::Ptr( void ) { return &list[ 0 ]; } /* ================ idStaticList::Ptr Returns a pointer to the begining of the array. Useful for iterating through the list in loops. Note: may return NULL if the list is empty. FIXME: Create an iterator template for this kind of thing. ================ */ template ID_INLINE const type *idStaticList::Ptr( void ) const { return &list[ 0 ]; } /* ================ idStaticList::Alloc Returns a pointer to a new data element at the end of the list. ================ */ template ID_INLINE type *idStaticList::Alloc( void ) { if ( num >= size ) { return NULL; } return &list[ num++ ]; } /* ================ idStaticList::Append Increases the size of the list by one element and copies the supplied data into it. Returns the index of the new element, or -1 when list is full. ================ */ template ID_INLINE int idStaticList::Append( type const & obj ) { assert( num < size ); if ( num < size ) { list[ num ] = obj; num++; return num - 1; } return -1; } /* ================ idStaticList::Insert Increases the size of the list by at leat one element if necessary and inserts the supplied data into it. Returns the index of the new element, or -1 when list is full. ================ */ template ID_INLINE int idStaticList::Insert( type const & obj, int index ) { int i; assert( num < size ); if ( num >= size ) { return -1; } assert( index >= 0 ); if ( index < 0 ) { index = 0; } else if ( index > num ) { index = num; } for( i = num; i > index; --i ) { list[i] = list[i-1]; } num++; list[index] = obj; return index; } /* ================ idStaticList::Append adds the other list to this one Returns the size of the new combined list ================ */ template ID_INLINE int idStaticList::Append( const idStaticList &other ) { int i; int n = other.Num(); if ( num + n > size ) { n = size - num; } for( i = 0; i < n; i++ ) { list[i + num] = other.list[i]; } num += n; return Num(); } /* ================ idStaticList::AddUnique Adds the data to the list if it doesn't already exist. Returns the index of the data in the list. ================ */ template ID_INLINE int idStaticList::AddUnique( type const & obj ) { int index; index = FindIndex( obj ); if ( index < 0 ) { index = Append( obj ); } return index; } /* ================ idStaticList::FindIndex Searches for the specified data in the list and returns it's index. Returns -1 if the data is not found. ================ */ template ID_INLINE int idStaticList::FindIndex( type const & obj ) const { int i; for( i = 0; i < num; i++ ) { if ( list[ i ] == obj ) { return i; } } // Not found return -1; } /* ================ idStaticList::Find Searches for the specified data in the list and returns it's address. Returns NULL if the data is not found. ================ */ template ID_INLINE type *idStaticList::Find( type const & obj ) const { int i; i = FindIndex( obj ); if ( i >= 0 ) { return &list[ i ]; } return NULL; } /* ================ idStaticList::FindNull Searches for a NULL pointer in the list. Returns -1 if NULL is not found. NOTE: This function can only be called on lists containing pointers. Calling it on non-pointer lists will cause a compiler error. ================ */ template ID_INLINE int idStaticList::FindNull( void ) const { int i; for( i = 0; i < num; i++ ) { if ( list[ i ] == NULL ) { return i; } } // Not found return -1; } /* ================ idStaticList::IndexOf Takes a pointer to an element in the list and returns the index of the element. This is NOT a guarantee that the object is really in the list. Function will assert in debug builds if pointer is outside the bounds of the list, but remains silent in release builds. ================ */ template ID_INLINE int idStaticList::IndexOf( type const *objptr ) const { int index; index = objptr - list; assert( index >= 0 ); assert( index < num ); return index; } /* ================ idStaticList::RemoveIndex Removes the element at the specified index and moves all data following the element down to fill in the gap. The number of elements in the list is reduced by one. Returns false if the index is outside the bounds of the list. Note that the element is not destroyed, so any memory used by it may not be freed until the destruction of the list. ================ */ template ID_INLINE bool idStaticList::RemoveIndex( int index ) { int i; assert( index >= 0 ); assert( index < num ); if ( ( index < 0 ) || ( index >= num ) ) { return false; } num--; for( i = index; i < num; i++ ) { list[ i ] = list[ i + 1 ]; } return true; } /* ================ idStaticList::Remove Removes the element if it is found within the list and moves all data following the element down to fill in the gap. The number of elements in the list is reduced by one. Returns false if the data is not found in the list. Note that the element is not destroyed, so any memory used by it may not be freed until the destruction of the list. ================ */ template ID_INLINE bool idStaticList::Remove( type const & obj ) { int index; index = FindIndex( obj ); if ( index >= 0 ) { return RemoveIndex( index ); } return false; } /* ================ idStaticList::Swap Swaps the contents of two lists ================ */ template ID_INLINE void idStaticList::Swap( idStaticList &other ) { idStaticList temp = *this; *this = other; other = temp; } #endif /* !__STATICLIST_H__ */