/* This code is based on source provided under the terms of the Id Software LIMITED USE SOFTWARE LICENSE AGREEMENT, a copy of which is included with the GtkRadiant sources (see LICENSE_ID). If you did not receive a copy of LICENSE_ID, please contact Id Software immediately at info@idsoftware.com. All changes and additions to the original source which have been developed by other contributors (see CONTRIBUTORS) are provided under the terms of the license the contributors choose (see LICENSE), to the extent permitted by the LICENSE_ID. If you did not receive a copy of the contributor license, please contact the GtkRadiant maintainers at info@gtkradiant.com immediately. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef __UTIL_LIST_H__ #define __UTIL_LIST_H__ #include #include template< class type > class idList { private: int m_num; int m_size; int m_granularity; type *m_list; public: idList( int granularity = 16 ); ~idList(); void Clear( void ); int Num( void ); void SetNum( int num ); void SetGranularity( int granularity ); void Condense( void ); int Size( void ); void Resize( int size ); type operator[]( int index ) const; type &operator[]( int index ); int Append( type const & obj ); int AddUnique( type const & obj ); type *Find( type const & obj, int *index = NULL ); bool RemoveIndex( int index ); bool Remove( type const & obj ); typedef int cmp_t ( const void *, const void * ); void Sort( cmp_t *compare ); }; /* ================ idList::idList( int ) ================ */ template< class type > inline idList::idList( int granularity ) { assert( granularity > 0 ); m_list = NULL; m_granularity = granularity; Clear(); } /* ================ idList::~idList ================ */ template< class type > inline idList::~idList() { Clear(); } /* ================ idList::Clear ================ */ template< class type > inline void idList::Clear( void ) { if ( m_list ) { delete[] m_list; } m_list = NULL; m_num = 0; m_size = 0; } /* ================ idList::Num ================ */ template< class type > inline int idList::Num( void ) { return m_num; } /* ================ idList::SetNum ================ */ template< class type > inline void idList::SetNum( int num ) { assert( num >= 0 ); if ( num > m_size ) { // resize it up to the closest level of granularity Resize( ( ( num + m_granularity - 1 ) / m_granularity ) * m_granularity ); } m_num = num; } /* ================ idList::SetGranularity ================ */ template< class type > inline void idList::SetGranularity( int granularity ) { int newsize; assert( granularity > 0 ); m_granularity = granularity; if ( m_list ) { // resize it to the closest level of granularity newsize = ( ( m_num + m_granularity - 1 ) / m_granularity ) * m_granularity; if ( newsize != m_size ) { Resize( newsize ); } } } /* ================ idList::Condense Resizes the array to exactly the number of elements it contains ================ */ template< class type > inline void idList::Condense( void ) { if ( m_list ) { if ( m_num ) { Resize( m_num ); } else { Clear(); } } } /* ================ idList::Size ================ */ template< class type > inline int idList::Size( void ) { return m_size; } /* ================ idList::Resize ================ */ template< class type > inline void idList::Resize( int size ) { type *temp; int i; assert( size > 0 ); if ( size <= 0 ) { Clear(); return; } temp = m_list; m_size = size; if ( m_size < m_num ) { m_num = m_size; } m_list = new type[ m_size ]; for ( i = 0; i < m_num; i++ ) { m_list[ i ] = temp[ i ]; } if ( temp ) { delete[] temp; } } /* ================ idList::operator[] const ================ */ template< class type > inline type idList::operator[]( int index ) const { assert( index >= 0 ); assert( index < m_num ); return m_list[ index ]; } /* ================ idList::operator[] ================ */ template< class type > inline type &idList::operator[]( int index ) { assert( index >= 0 ); assert( index < m_num ); return m_list[ index ]; } /* ================ idList::Append ================ */ template< class type > inline int idList::Append( type const & obj ) { if ( !m_list ) { Resize( m_granularity ); } if ( m_num == m_size ) { Resize( m_size + m_granularity ); } m_list[ m_num ] = obj; m_num++; return m_num - 1; } /* ================ idList::AddUnique ================ */ template< class type > inline int idList::AddUnique( type const & obj ) { int index; if ( !Find( obj, &index ) ) { index = Append( obj ); } return index; } /* ================ idList::Find ================ */ template< class type > inline type *idList::Find( type const & obj, int *index ) { int i; for ( i = 0; i < m_num; i++ ) { if ( m_list[ i ] == obj ) { if ( index ) { *index = i; } return &m_list[ i ]; } } return NULL; } /* ================ idList::RemoveIndex ================ */ template< class type > inline bool idList::RemoveIndex( int index ) { int i; if ( !m_list || !m_num ) { return false; } assert( index >= 0 ); assert( index < m_num ); if ( ( index < 0 ) || ( index >= m_num ) ) { return false; } m_num--; for ( i = index; i < m_num; i++ ) { m_list[ i ] = m_list[ i + 1 ]; } return true; } /* ================ idList::Remove ================ */ template< class type > inline bool idList::Remove( type const & obj ) { int index; if ( Find( obj, &index ) ) { return RemoveIndex( index ); } return false; } /* ================ idList::Sort ================ */ template< class type > inline void idList::Sort( cmp_t *compare ) { if ( !m_list ) { return; } qsort( ( void * )m_list, ( size_t )m_num, sizeof( type ), compare ); } #endif /* !__UTIL_LIST_H__ */