mirror of
https://github.com/TTimo/GtkRadiant.git
synced 2025-01-10 20:10:56 +00:00
350 lines
7 KiB
C
350 lines
7 KiB
C
|
/*
|
||
|
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 <stdlib.h>
|
||
|
#include <assert.h>
|
||
|
|
||
|
template< class type >
|
||
|
class idList {
|
||
|
private:
|
||
|
int m_num;
|
||
|
int m_size;
|
||
|
int m_granularity;
|
||
|
type *m_list;
|
||
|
|
||
|
public:
|
||
|
idList( int granularity = 16 );
|
||
|
~idList<type>();
|
||
|
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<type>::idList( int )
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline idList<type>::idList( int granularity ) {
|
||
|
assert( granularity > 0 );
|
||
|
|
||
|
m_list = NULL;
|
||
|
m_granularity = granularity;
|
||
|
Clear();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::~idList<type>
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline idList<type>::~idList() {
|
||
|
Clear();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::Clear
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline void idList<type>::Clear( void ) {
|
||
|
if ( m_list ) {
|
||
|
delete[] m_list;
|
||
|
}
|
||
|
|
||
|
m_list = NULL;
|
||
|
m_num = 0;
|
||
|
m_size = 0;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::Num
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline int idList<type>::Num( void ) {
|
||
|
return m_num;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::SetNum
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline void idList<type>::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<type>::SetGranularity
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline void idList<type>::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<type>::Condense
|
||
|
|
||
|
Resizes the array to exactly the number of elements it contains
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline void idList<type>::Condense( void ) {
|
||
|
if ( m_list ) {
|
||
|
if ( m_num ) {
|
||
|
Resize( m_num );
|
||
|
} else {
|
||
|
Clear();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::Size
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline int idList<type>::Size( void ) {
|
||
|
return m_size;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::Resize
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline void idList<type>::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<type>::operator[] const
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline type idList<type>::operator[]( int index ) const {
|
||
|
assert( index >= 0 );
|
||
|
assert( index < m_num );
|
||
|
|
||
|
return m_list[ index ];
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::operator[]
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline type &idList<type>::operator[]( int index ) {
|
||
|
assert( index >= 0 );
|
||
|
assert( index < m_num );
|
||
|
|
||
|
return m_list[ index ];
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::Append
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline int idList<type>::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<type>::AddUnique
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline int idList<type>::AddUnique( type const & obj ) {
|
||
|
int index;
|
||
|
|
||
|
if ( !Find( obj, &index ) ) {
|
||
|
index = Append( obj );
|
||
|
}
|
||
|
|
||
|
return index;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::Find
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline type *idList<type>::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<type>::RemoveIndex
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline bool idList<type>::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<type>::Remove
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline bool idList<type>::Remove( type const & obj ) {
|
||
|
int index;
|
||
|
|
||
|
if ( Find( obj, &index ) ) {
|
||
|
return RemoveIndex( index );
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
================
|
||
|
idList<type>::Sort
|
||
|
================
|
||
|
*/
|
||
|
template< class type >
|
||
|
inline void idList<type>::Sort( cmp_t *compare ) {
|
||
|
if ( !m_list ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
qsort( ( void * )m_list, ( size_t )m_num, sizeof( type ), compare );
|
||
|
}
|
||
|
|
||
|
#endif /* !__UTIL_LIST_H__ */
|