You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
335 lines
8.0 KiB
C++
335 lines
8.0 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// ## ######
|
|
// ###### ### Based on:
|
|
// ## ###############
|
|
// ########## # # # Shark 3D Engine (www.shark3d.com)
|
|
// ########
|
|
// ######### # # # Copyright (c) 1996-2001 Spinor GmbH.
|
|
// ## ##########
|
|
// ##
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// G A M E.O.N.E - LOW LEVEL LIB V1.0
|
|
// Copyright (C) 2001 LEVEL ONE ENTERTAINMENT,
|
|
// Licensed under the terms of LGPL.
|
|
//:---------------------------------------------------------------------------
|
|
//:Description
|
|
//
|
|
// LOW LEVEL QUEUE_LIST TEMPLATE HEADERFILE
|
|
//
|
|
|
|
#ifndef _LLQUEUE_LIST_TMPL_H
|
|
#define _LLQUEUE_LIST_TMPL_H
|
|
|
|
#include "llqueue_list.h"
|
|
|
|
namespace llqueue {
|
|
|
|
|
|
template<typename T> class CListNode;
|
|
template<typename T> class CList;
|
|
|
|
/*
|
|
* Double Linked List Node Template
|
|
* Template Argument ist der Node Inhalt (m_Data)
|
|
*/
|
|
|
|
template<typename T>
|
|
class CListNode: private queue_listnode
|
|
{
|
|
public:
|
|
T m_Data; // The content carried by this node:
|
|
|
|
CListNode(); // Default constructor.
|
|
CListNode(const T &Data); // Constructor initializing the content.
|
|
CListNode(const CListNode<T> &Node);// Copy constructor, which copies the content.
|
|
~CListNode(); // Destructor. The node is extracted from the list automatically if necessary.
|
|
|
|
CListNode<T> &operator=(const CListNode<T> &Node);// Assignment operator, which assigns the content.
|
|
CListNode<T> *GetPrev() const;// Get the previous node.
|
|
CListNode<T> *GetNext() const;// Get the next node.
|
|
CList<T> *GetBase() const;// Get the node owning this list.
|
|
|
|
bool IsInserted() const;// Is this node part of a list?
|
|
|
|
// Extract this node from the list.
|
|
// Then the node is not part of the list any more.
|
|
// Note that this method does not destroy the node.
|
|
// If you both want to destroy the node and extract it from the list,
|
|
// use the delete.
|
|
void Extract();
|
|
|
|
private:
|
|
friend class CList<T>;
|
|
};
|
|
|
|
|
|
/*
|
|
* An intrusive list.
|
|
* The template argument is the content carried by the node
|
|
* in CListNode::m_Data.
|
|
*
|
|
* Important!
|
|
* A list CList does not own its nodes.
|
|
* Thus, if you insert a node into a list, the list does not copy
|
|
* or take over ownership of the node.
|
|
*
|
|
* If the node is destroyed, it is automatically also extracted
|
|
* from the list.
|
|
* If the list is destroyed, all its nodes are extracted out of the list,
|
|
* but not destroyed.
|
|
* If you want to free all nodes, use CList::DeleteAll.
|
|
*
|
|
* A list has similar properties like a tree,
|
|
*
|
|
* CList is primary designed for performance critical
|
|
* low-level code. The user has to allocate, manage and deallocate
|
|
* the list nodes explicitely.
|
|
* For a more high-level container template class, see CArray.
|
|
*
|
|
* The entries of a list can be enumerated by a loop like the following:
|
|
* {
|
|
* void HandleEachValue(CList<T> *List)
|
|
* {
|
|
* CList<T>::CNode *Node;
|
|
* for(Node = List->GetFirst(); Node; Node = Node->GetNext())
|
|
* HandleValue(Node->m_Data);
|
|
* }
|
|
* }
|
|
* Warning! As long as such a loop is active, you should ensure
|
|
* that the list is not modified to avoid illegal node pointers.
|
|
*
|
|
*
|
|
* Since the user manages the list nodes, they can be managed
|
|
* efficiently by putting them in structures or classes
|
|
* which anyway are existing.
|
|
* For example, it all objects of an type should be registered
|
|
* somewhere, these objects may carry by themself the list node:
|
|
*
|
|
* {
|
|
* class CMyObj
|
|
* {
|
|
* public:
|
|
* CMyObj() { m_Node.m_Data = this; }
|
|
* friend class CMyObjManager;
|
|
* private:
|
|
* int m_MyData;
|
|
* CList<CMyObj*>::CNode m_Node;
|
|
* }
|
|
*
|
|
* class CMyObjManager
|
|
* {
|
|
* public:
|
|
* CMyObj *CreateMyObj();
|
|
* private:
|
|
* CList<CMyObj*> m_List;
|
|
* }
|
|
*
|
|
* CMyObj *CMyObjManager::CreateMyObj()
|
|
* {
|
|
* CMyObj *MyObj = SYS_NEW CMyObj;
|
|
* m_List.InsertBack(&MyObj->m_Node);
|
|
* return MyObj;
|
|
* }
|
|
* }
|
|
* By this, the manager always has an up-to-date list
|
|
* of all object it has created.
|
|
* But note that the manager is not the owner of these objects.
|
|
* If an object created by @code{CMyObjManager::CreateMyObj} is destroyed
|
|
* by the user, it also automatically is extracted from the list.
|
|
*/
|
|
|
|
template<typename T>
|
|
class CList: private queue_listhead
|
|
{
|
|
public:
|
|
|
|
typedef CListNode<T> CNode;
|
|
|
|
CList();
|
|
~CList();
|
|
|
|
// retrieve elements and information
|
|
CListNode<T> *GetFirst() const;
|
|
CListNode<T> *GetLast() const;
|
|
bool IsEmpty() const;
|
|
bool Contains(CListNode<T> *Node);
|
|
int GetCount() const;
|
|
|
|
// inserting elements
|
|
void InsertFront(CListNode<T> *Node);
|
|
void InsertBack(CListNode<T> *Node);
|
|
void InsertAfter(CListNode<T> *Node, CListNode<T> *Pos);
|
|
void InsertBefore(CListNode<T> *Node, CListNode<T> *Pos);
|
|
void CopyFrom(const CList<T> &Base);
|
|
|
|
// delete ALL
|
|
void ExtractAll();
|
|
void DeleteAll();
|
|
|
|
private:
|
|
CList(const CList &List);
|
|
CList &operator=(const CList &List);
|
|
};
|
|
|
|
/*
|
|
* Implementation
|
|
*/
|
|
template<typename T>
|
|
inline CListNode<T>::CListNode()
|
|
{
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T>::CListNode(const T &Con)
|
|
: m_Data(Con)
|
|
{
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T>::CListNode(const CListNode<T> &Node)
|
|
: m_Data(Node.m_Data)
|
|
{
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T>::~CListNode()
|
|
{
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T> &CListNode<T>::operator=(
|
|
const CListNode &Node)
|
|
{
|
|
m_Data = Node.m_Data;
|
|
return *this;
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T> *CListNode<T>::GetPrev() const
|
|
{
|
|
return reinterpret_cast<CListNode<T>*>(queue_listnode::GetPrev());
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T> *CListNode<T>::GetNext() const
|
|
{
|
|
return reinterpret_cast<CListNode<T>*>(queue_listnode::GetNext());
|
|
}
|
|
|
|
template<typename T>
|
|
inline CList<T> *CListNode<T>::GetBase() const
|
|
{
|
|
return reinterpret_cast<CList<T>*>(queue_listnode::GetBase());
|
|
}
|
|
|
|
template<typename T>
|
|
inline bool CListNode<T>::IsInserted() const
|
|
{
|
|
return queue_listnode::IsInserted();
|
|
}
|
|
|
|
template<typename T>
|
|
inline void CListNode<T>::Extract()
|
|
{
|
|
queue_listnode::Extract();
|
|
}
|
|
|
|
|
|
template<typename T>
|
|
inline CList<T>::CList()
|
|
{
|
|
}
|
|
|
|
template<typename T>
|
|
inline CList<T>::~CList()
|
|
{
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T> *CList<T>::GetFirst() const
|
|
{
|
|
return reinterpret_cast<CListNode<T>*>(queue_listhead::GetFirst());
|
|
}
|
|
|
|
template<typename T>
|
|
inline CListNode<T> *CList<T>::GetLast() const
|
|
{
|
|
return reinterpret_cast<CListNode<T>*>(queue_listhead::GetLast());
|
|
}
|
|
|
|
template<typename T>
|
|
inline bool CList<T>::IsEmpty() const
|
|
{
|
|
return queue_listhead::IsEmpty();
|
|
}
|
|
|
|
template<typename T>
|
|
inline bool CList<T>::Contains(CListNode<T> *Node)
|
|
{
|
|
return queue_listhead::Contains(Node);
|
|
}
|
|
|
|
template<typename T>
|
|
inline int CList<T>::GetCount() const
|
|
{
|
|
return queue_listhead::GetCount();
|
|
}
|
|
|
|
template<typename T>
|
|
inline void CList<T>::InsertFront(CListNode<T> *Node)
|
|
{
|
|
queue_listhead::InsertFront(Node);
|
|
}
|
|
|
|
template<typename T>
|
|
inline void CList<T>::InsertBack(CListNode<T> *Node)
|
|
{
|
|
queue_listhead::InsertBack(Node);
|
|
}
|
|
|
|
template<typename T>
|
|
inline void CList<T>::InsertAfter(CListNode<T> *Node,
|
|
CListNode<T> *Pos)
|
|
{
|
|
queue_listhead::InsertAfter(Node, Pos);
|
|
}
|
|
|
|
template<typename T>
|
|
inline void CList<T>::InsertBefore(CListNode<T> *Node,
|
|
CListNode<T> *Pos)
|
|
{
|
|
queue_listhead::InsertBefore(Node, Pos);
|
|
}
|
|
|
|
template<typename T>
|
|
inline void CList<T>::ExtractAll()
|
|
{
|
|
queue_listhead::ExtractAll();
|
|
}
|
|
|
|
template<typename T>
|
|
void CList<T>::DeleteAll()
|
|
{
|
|
while(m_First)
|
|
delete m_First;
|
|
}
|
|
|
|
template<typename T>
|
|
void CList<T>::CopyFrom(const CList<T> &Base)
|
|
{
|
|
DeleteAll();
|
|
CListNode<T> *Node;
|
|
for(Node = Base.m_First; Node; Node = Node->m_Next)
|
|
{
|
|
CListNode<T> *Copy = new CListNode<T>(*Node);
|
|
InsertBack(Copy);
|
|
}
|
|
}
|
|
|
|
}// namespace ll
|
|
|
|
#endif // _LLQUEUE_LIST_TMPL_H
|