567 lines
14 KiB
C
Raw Normal View History

2015-06-30 15:46:07 +06:00
#ifndef UTLVECTOR_H
#define UTLVECTOR_H
#ifdef _WIN32
#pragma once
#endif
#include "utlmemory.h"
2015-12-09 04:39:54 +06:00
//#include "tier0/platform.h"
2015-06-30 15:46:07 +06:00
template< class T >
class CUtlVector
{
public:
typedef T ElemType_t;
// constructor, destructor
2015-09-17 02:19:21 +06:00
CUtlVector(int growSize = 0, int initSize = 0);
CUtlVector(T* pMemory, int numElements);
2015-06-30 15:46:07 +06:00
~CUtlVector();
2015-09-17 02:19:21 +06:00
2015-06-30 15:46:07 +06:00
// Copy the array.
2015-09-17 02:19:21 +06:00
CUtlVector<T>& operator=(const CUtlVector<T> &other);
2015-06-30 15:46:07 +06:00
// element access
2015-09-17 02:19:21 +06:00
T& operator[](int i);
T const& operator[](int i) const;
T& Element(int i);
T const& Element(int i) const;
2015-06-30 15:46:07 +06:00
// Gets the base address (can change when adding elements!)
T* Base();
T const* Base() const;
// Returns the number of elements in the vector
// SIZE IS DEPRECATED!
int Count() const;
int Size() const; // don't use me!
// Is element index valid?
2015-09-17 02:19:21 +06:00
bool IsValidIndex(int i) const;
static int InvalidIndex(void);
2015-06-30 15:46:07 +06:00
// Adds an element, uses default constructor
int AddToHead();
int AddToTail();
2015-09-17 02:19:21 +06:00
int InsertBefore(int elem);
int InsertAfter(int elem);
2015-06-30 15:46:07 +06:00
// Adds an element, uses copy constructor
2015-09-17 02:19:21 +06:00
int AddToHead(T const& src);
int AddToTail(T const& src);
int InsertBefore(int elem, T const& src);
int InsertAfter(int elem, T const& src);
2015-06-30 15:46:07 +06:00
// Adds multiple elements, uses default constructor
2015-09-17 02:19:21 +06:00
int AddMultipleToHead(int num);
int AddMultipleToTail(int num, const T *pToCopy=NULL);
int InsertMultipleBefore(int elem, int num, const T *pToCopy=NULL); // If pToCopy is set, then it's an array of length 'num' and
int InsertMultipleAfter(int elem, int num);
2015-06-30 15:46:07 +06:00
// Calls RemoveAll() then AddMultipleToTail.
2015-09-17 02:19:21 +06:00
void SetSize(int size);
void SetCount(int count);
2015-06-30 15:46:07 +06:00
// Calls SetSize and copies each element.
2015-09-17 02:19:21 +06:00
void CopyArray(T const *pArray, int size);
2015-06-30 15:46:07 +06:00
// Add the specified array to the tail.
2015-09-17 02:19:21 +06:00
int AddVectorToTail(CUtlVector<T> const &src);
2015-06-30 15:46:07 +06:00
// Finds an element (element needs operator== defined)
2015-09-17 02:19:21 +06:00
int Find(T const& src) const;
2015-06-30 15:46:07 +06:00
2015-09-17 02:19:21 +06:00
bool HasElement(T const& src);
2015-06-30 15:46:07 +06:00
// Makes sure we have enough memory allocated to store a requested # of elements
2015-09-17 02:19:21 +06:00
void EnsureCapacity(int num);
2015-06-30 15:46:07 +06:00
// Makes sure we have at least this many elements
2015-09-17 02:19:21 +06:00
void EnsureCount(int num);
2015-06-30 15:46:07 +06:00
// Element removal
2015-09-17 02:19:21 +06:00
void FastRemove(int elem); // doesn't preserve order
void Remove(int elem); // preserves order, shifts elements
void FindAndRemove(T const& src); // removes first occurrence of src, preserves order, shifts elements
void RemoveMultiple(int elem, int num); // preserves order, shifts elements
2015-06-30 15:46:07 +06:00
void RemoveAll(); // doesn't deallocate memory
// Memory deallocation
void Purge();
// Purges the list and calls delete on each element in it.
void PurgeAndDeleteElements();
// Set the size by which it grows when it needs to allocate more memory.
2015-09-17 02:19:21 +06:00
void SetGrowSize(int size);
2015-06-30 15:46:07 +06:00
protected:
// Can't copy this unless we explicitly do it!
2015-09-17 02:19:21 +06:00
CUtlVector(CUtlVector const& vec) { assert(0); }
2015-06-30 15:46:07 +06:00
// Grows the vector
2015-09-17 02:19:21 +06:00
void GrowVector(int num = 1);
2015-06-30 15:46:07 +06:00
// Shifts elements....
2015-09-17 02:19:21 +06:00
void ShiftElementsRight(int elem, int num = 1);
void ShiftElementsLeft(int elem, int num = 1);
2015-06-30 15:46:07 +06:00
// For easier access to the elements through the debugger
void ResetDbgInfo();
CUtlMemory<T> m_Memory;
int m_Size;
// For easier access to the elements through the debugger
// it's in release builds so this can be used in libraries correctly
T *m_pElements;
};
//-----------------------------------------------------------------------------
// For easier access to the elements through the debugger
//-----------------------------------------------------------------------------
template< class T >
inline void CUtlVector<T>::ResetDbgInfo()
{
m_pElements = m_Memory.Base();
}
//-----------------------------------------------------------------------------
// constructor, destructor
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
inline CUtlVector<T>::CUtlVector(int growSize, int initSize) :
2015-06-30 15:46:07 +06:00
m_Memory(growSize, initSize), m_Size(0)
{
ResetDbgInfo();
}
template< class T >
2015-09-17 02:19:21 +06:00
inline CUtlVector<T>::CUtlVector(T* pMemory, int numElements) :
2015-06-30 15:46:07 +06:00
m_Memory(pMemory, numElements), m_Size(0)
{
ResetDbgInfo();
}
template< class T >
inline CUtlVector<T>::~CUtlVector()
{
Purge();
}
template<class T>
2015-09-17 02:19:21 +06:00
inline CUtlVector<T>& CUtlVector<T>::operator=(const CUtlVector<T> &other)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
CopyArray(other.Base(), other.Count());
2015-06-30 15:46:07 +06:00
return *this;
}
//-----------------------------------------------------------------------------
// element access
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
inline T& CUtlVector<T>::operator[](int i)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(i));
2015-06-30 15:46:07 +06:00
return m_Memory[i];
}
template< class T >
2015-09-17 02:19:21 +06:00
inline T const& CUtlVector<T>::operator[](int i) const
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(i));
2015-06-30 15:46:07 +06:00
return m_Memory[i];
}
template< class T >
2015-09-17 02:19:21 +06:00
inline T& CUtlVector<T>::Element(int i)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(i));
2015-06-30 15:46:07 +06:00
return m_Memory[i];
}
template< class T >
2015-09-17 02:19:21 +06:00
inline T const& CUtlVector<T>::Element(int i) const
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(i));
2015-06-30 15:46:07 +06:00
return m_Memory[i];
}
//-----------------------------------------------------------------------------
// Gets the base address (can change when adding elements!)
//-----------------------------------------------------------------------------
template< class T >
inline T* CUtlVector<T>::Base()
{
return m_Memory.Base();
}
template< class T >
inline T const* CUtlVector<T>::Base() const
{
return m_Memory.Base();
}
//-----------------------------------------------------------------------------
// Count
//-----------------------------------------------------------------------------
template< class T >
inline int CUtlVector<T>::Size() const
{
return m_Size;
}
template< class T >
inline int CUtlVector<T>::Count() const
{
return m_Size;
}
//-----------------------------------------------------------------------------
// Is element index valid?
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
inline bool CUtlVector<T>::IsValidIndex(int i) const
2015-06-30 15:46:07 +06:00
{
return (i >= 0) && (i < m_Size);
}
//-----------------------------------------------------------------------------
// Returns in invalid index
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::InvalidIndex(void)
2015-06-30 15:46:07 +06:00
{
return -1;
}
//-----------------------------------------------------------------------------
// Grows the vector
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::GrowVector(int num)
2015-06-30 15:46:07 +06:00
{
if (m_Size + num - 1 >= m_Memory.NumAllocated())
{
2015-09-17 02:19:21 +06:00
m_Memory.Grow(m_Size + num - m_Memory.NumAllocated());
2015-06-30 15:46:07 +06:00
}
m_Size += num;
ResetDbgInfo();
}
//-----------------------------------------------------------------------------
// Makes sure we have enough memory allocated to store a requested # of elements
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::EnsureCapacity(int num)
2015-06-30 15:46:07 +06:00
{
m_Memory.EnsureCapacity(num);
ResetDbgInfo();
}
//-----------------------------------------------------------------------------
// Makes sure we have at least this many elements
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::EnsureCount(int num)
2015-06-30 15:46:07 +06:00
{
if (Count() < num)
2015-09-17 02:19:21 +06:00
AddMultipleToTail(num - Count());
2015-06-30 15:46:07 +06:00
}
//-----------------------------------------------------------------------------
// Shifts elements
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::ShiftElementsRight(int elem, int num)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(elem) || (m_Size == 0) || (num == 0));
2015-06-30 15:46:07 +06:00
int numToMove = m_Size - elem - num;
if ((numToMove > 0) && (num > 0))
2015-09-17 02:19:21 +06:00
memmove(&Element(elem+num), &Element(elem), numToMove * sizeof(T));
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::ShiftElementsLeft(int elem, int num)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(elem) || (m_Size == 0) || (num == 0));
2015-06-30 15:46:07 +06:00
int numToMove = m_Size - elem - num;
if ((numToMove > 0) && (num > 0))
{
2015-09-17 02:19:21 +06:00
memmove(&Element(elem), &Element(elem+num), numToMove * sizeof(T));
2015-06-30 15:46:07 +06:00
#ifdef _DEBUG
2015-09-17 02:19:21 +06:00
memset(&Element(m_Size-num), 0xDD, num * sizeof(T));
2015-06-30 15:46:07 +06:00
#endif
}
}
//-----------------------------------------------------------------------------
// Adds an element, uses default constructor
//-----------------------------------------------------------------------------
template< class T >
inline int CUtlVector<T>::AddToHead()
{
return InsertBefore(0);
}
template< class T >
inline int CUtlVector<T>::AddToTail()
{
2015-09-17 02:19:21 +06:00
return InsertBefore(m_Size);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::InsertAfter(int elem)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return InsertBefore(elem + 1);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
int CUtlVector<T>::InsertBefore(int elem)
2015-06-30 15:46:07 +06:00
{
// Can insert at the end
2015-09-17 02:19:21 +06:00
assert((elem == Count()) || IsValidIndex(elem));
2015-06-30 15:46:07 +06:00
GrowVector();
ShiftElementsRight(elem);
2015-09-17 02:19:21 +06:00
Construct(&Element(elem));
2015-06-30 15:46:07 +06:00
return elem;
}
//-----------------------------------------------------------------------------
// Adds an element, uses copy constructor
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::AddToHead(T const& src)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return InsertBefore(0, src);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::AddToTail(T const& src)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return InsertBefore(m_Size, src);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::InsertAfter(int elem, T const& src)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return InsertBefore(elem + 1, src);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
int CUtlVector<T>::InsertBefore(int elem, T const& src)
2015-06-30 15:46:07 +06:00
{
// Can insert at the end
2015-09-17 02:19:21 +06:00
assert((elem == Count()) || IsValidIndex(elem));
2015-06-30 15:46:07 +06:00
GrowVector();
ShiftElementsRight(elem);
2015-09-17 02:19:21 +06:00
CopyConstruct(&Element(elem), src);
2015-06-30 15:46:07 +06:00
return elem;
}
//-----------------------------------------------------------------------------
// Adds multiple elements, uses default constructor
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::AddMultipleToHead(int num)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return InsertMultipleBefore(0, num);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::AddMultipleToTail(int num, const T *pToCopy)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return InsertMultipleBefore(m_Size, num, pToCopy);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
int CUtlVector<T>::InsertMultipleAfter(int elem, int num)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return InsertMultipleBefore(elem + 1, num);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::SetCount(int count)
2015-06-30 15:46:07 +06:00
{
RemoveAll();
2015-09-17 02:19:21 +06:00
AddMultipleToTail(count);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
inline void CUtlVector<T>::SetSize(int size)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
SetCount(size);
2015-06-30 15:46:07 +06:00
}
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::CopyArray(T const *pArray, int size)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
SetSize(size);
for(int i=0; i < size; i++)
2015-06-30 15:46:07 +06:00
(*this)[i] = pArray[i];
}
template< class T >
2015-09-17 02:19:21 +06:00
int CUtlVector<T>::AddVectorToTail(CUtlVector const &src)
2015-06-30 15:46:07 +06:00
{
int base = Count();
2015-09-17 02:19:21 +06:00
2015-06-30 15:46:07 +06:00
// Make space.
2015-09-17 02:19:21 +06:00
AddMultipleToTail(src.Count());
2015-06-30 15:46:07 +06:00
2015-09-17 02:19:21 +06:00
// Copy the elements.
for (int i=0; i < src.Count(); i++)
2015-06-30 15:46:07 +06:00
(*this)[base + i] = src[i];
return base;
}
template< class T >
2015-09-17 02:19:21 +06:00
inline int CUtlVector<T>::InsertMultipleBefore(int elem, int num, const T *pToInsert)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
if(num == 0)
2015-06-30 15:46:07 +06:00
return elem;
2015-09-17 02:19:21 +06:00
2015-06-30 15:46:07 +06:00
// Can insert at the end
2015-09-17 02:19:21 +06:00
assert((elem == Count()) || IsValidIndex(elem));
2015-06-30 15:46:07 +06:00
GrowVector(num);
ShiftElementsRight(elem, num);
// Invoke default constructors
for (int i = 0; i < num; ++i)
2015-09-17 02:19:21 +06:00
Construct(&Element(elem+i));
2015-06-30 15:46:07 +06:00
// Copy stuff in?
2015-09-17 02:19:21 +06:00
if (pToInsert)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
for (int i=0; i < num; i++)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
Element(elem+i) = pToInsert[i];
2015-06-30 15:46:07 +06:00
}
}
return elem;
}
//-----------------------------------------------------------------------------
// Finds an element (element needs operator== defined)
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
int CUtlVector<T>::Find(T const& src) const
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
for (int i = 0; i < Count(); ++i)
2015-06-30 15:46:07 +06:00
{
if (Element(i) == src)
return i;
}
return -1;
}
template< class T >
2015-09-17 02:19:21 +06:00
bool CUtlVector<T>::HasElement(T const& src)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
return (Find(src) >= 0);
2015-06-30 15:46:07 +06:00
}
//-----------------------------------------------------------------------------
// Element removal
//-----------------------------------------------------------------------------
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::FastRemove(int elem)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(elem));
2015-06-30 15:46:07 +06:00
2015-09-17 02:19:21 +06:00
Destruct(&Element(elem));
2015-06-30 15:46:07 +06:00
if (m_Size > 0)
{
2015-09-17 02:19:21 +06:00
Q_memcpy(&Element(elem), &Element(m_Size-1), sizeof(T));
2015-06-30 15:46:07 +06:00
--m_Size;
}
}
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::Remove(int elem)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
Destruct(&Element(elem));
2015-06-30 15:46:07 +06:00
ShiftElementsLeft(elem);
--m_Size;
}
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::FindAndRemove(T const& src)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
int elem = Find(src);
if (elem != -1)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
Remove(elem);
2015-06-30 15:46:07 +06:00
}
}
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::RemoveMultiple(int elem, int num)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
assert(IsValidIndex(elem));
assert(elem + num <= Count());
2015-06-30 15:46:07 +06:00
2015-09-17 02:19:21 +06:00
for (int i = elem + num; --i >= elem;)
2015-06-30 15:46:07 +06:00
Destruct(&Element(i));
ShiftElementsLeft(elem, num);
m_Size -= num;
}
template< class T >
void CUtlVector<T>::RemoveAll()
{
2015-09-17 02:19:21 +06:00
for (int i = m_Size; --i >= 0;)
2015-06-30 15:46:07 +06:00
Destruct(&Element(i));
m_Size = 0;
}
//-----------------------------------------------------------------------------
// Memory deallocation
//-----------------------------------------------------------------------------
template< class T >
void CUtlVector<T>::Purge()
{
RemoveAll();
2015-09-17 02:19:21 +06:00
m_Memory.Purge();
2015-06-30 15:46:07 +06:00
ResetDbgInfo();
}
template<class T>
inline void CUtlVector<T>::PurgeAndDeleteElements()
{
2015-09-17 02:19:21 +06:00
for(int i=0; i < m_Size; i++)
2015-06-30 15:46:07 +06:00
delete Element(i);
Purge();
}
template< class T >
2015-09-17 02:19:21 +06:00
void CUtlVector<T>::SetGrowSize(int size)
2015-06-30 15:46:07 +06:00
{
2015-09-17 02:19:21 +06:00
m_Memory.SetGrowSize(size);
2015-06-30 15:46:07 +06:00
}
#endif // CCVECTOR_H