mirror of
				https://github.com/isledecomp/isle.git
				synced 2025-10-26 01:44:19 +00:00 
			
		
		
		
	Merge branch 'master' into mxregion
This commit is contained in:
		
							
								
								
									
										292
									
								
								LEGO1/mxlist.h
									
									
									
									
									
								
							
							
						
						
									
										292
									
								
								LEGO1/mxlist.h
									
									
									
									
									
								
							| @@ -1,268 +1,274 @@ | ||||
| #ifndef MXLIST_H | ||||
| #define MXLIST_H | ||||
| 
 | ||||
| #include "mxtypes.h" | ||||
| #include "mxcore.h" | ||||
| #include "mxtypes.h" | ||||
| 
 | ||||
| template <class T> | ||||
| class MxListEntry | ||||
| { | ||||
| class MxList; | ||||
| template <class T> | ||||
| class MxListCursor; | ||||
| 
 | ||||
| template <class T> | ||||
| class MxListEntry { | ||||
| public: | ||||
|   MxListEntry() {} | ||||
|   MxListEntry(T p_obj, MxListEntry *p_prev) { | ||||
|     m_obj = p_obj; | ||||
|     m_prev = p_prev; | ||||
|     m_next = NULL; | ||||
|   } | ||||
|   MxListEntry(T p_obj, MxListEntry *p_prev, MxListEntry *p_next) { | ||||
|     m_obj = p_obj; | ||||
|     m_prev = p_prev; | ||||
|     m_next = p_next; | ||||
|   } | ||||
| 	MxListEntry() {} | ||||
| 	MxListEntry(T p_obj, MxListEntry* p_prev) | ||||
| 	{ | ||||
| 		m_obj = p_obj; | ||||
| 		m_prev = p_prev; | ||||
| 		m_next = NULL; | ||||
| 	} | ||||
| 	MxListEntry(T p_obj, MxListEntry* p_prev, MxListEntry* p_next) | ||||
| 	{ | ||||
| 		m_obj = p_obj; | ||||
| 		m_prev = p_prev; | ||||
| 		m_next = p_next; | ||||
| 	} | ||||
| 
 | ||||
|   T GetValue() { return this->m_obj; } | ||||
| 	T GetValue() { return this->m_obj; } | ||||
| 
 | ||||
| 	friend class MxList<T>; | ||||
| 	friend class MxListCursor<T>; | ||||
| 
 | ||||
|   friend class MxList<T>; | ||||
|   friend class MxListCursor<T>; | ||||
| private: | ||||
|   T m_obj; | ||||
|   MxListEntry *m_prev; | ||||
|   MxListEntry *m_next; | ||||
| 	T m_obj; | ||||
| 	MxListEntry* m_prev; | ||||
| 	MxListEntry* m_next; | ||||
| }; | ||||
| 
 | ||||
| // SIZE 0x10
 | ||||
| template <class T> | ||||
| class MxListParent : public MxCore | ||||
| { | ||||
| class MxListParent : public MxCore { | ||||
| public: | ||||
|   MxListParent() { | ||||
|     m_count = 0; | ||||
|     m_customDestructor = Destroy; | ||||
|   } | ||||
| 	MxListParent() | ||||
| 	{ | ||||
| 		m_count = 0; | ||||
| 		m_customDestructor = Destroy; | ||||
| 	} | ||||
| 
 | ||||
|   virtual ~MxListParent() {} | ||||
|   virtual MxS8 Compare(T, T) { return 0; }; | ||||
| 	virtual ~MxListParent() {} | ||||
| 	virtual MxS8 Compare(T, T) { return 0; }; | ||||
| 
 | ||||
| 	static void Destroy(T){}; | ||||
| 
 | ||||
|   static void Destroy(T) {}; | ||||
| protected: | ||||
|   MxU32 m_count;                   // +0x8
 | ||||
|   void (*m_customDestructor)(T); // +0xc
 | ||||
| 	MxU32 m_count;                 // +0x8
 | ||||
| 	void (*m_customDestructor)(T); // +0xc
 | ||||
| }; | ||||
| 
 | ||||
| // SIZE 0x18
 | ||||
| template <class T> | ||||
| class MxList : protected MxListParent<T> | ||||
| { | ||||
| class MxList : protected MxListParent<T> { | ||||
| public: | ||||
|   MxList() { | ||||
|     m_last = NULL; | ||||
|     m_first = NULL; | ||||
|   } | ||||
| 	MxList() | ||||
| 	{ | ||||
| 		m_last = NULL; | ||||
| 		m_first = NULL; | ||||
| 	} | ||||
| 
 | ||||
|   virtual ~MxList(); | ||||
| 	virtual ~MxList(); | ||||
| 
 | ||||
|   void Append(T); | ||||
|   void OtherAppend(T p_obj) { _InsertEntry(p_obj, this->m_last, NULL); }; | ||||
|   void DeleteAll(); | ||||
|   MxU32 GetCount() { return m_count; } | ||||
|   void SetDestroy(void (*p_customDestructor)(T)) { this->m_customDestructor = p_customDestructor; } | ||||
| 	void Append(T); | ||||
| 	void OtherAppend(T p_obj) { _InsertEntry(p_obj, this->m_last, NULL); }; | ||||
| 	void DeleteAll(); | ||||
| 	MxU32 GetCount() { return this->m_count; } | ||||
| 	void SetDestroy(void (*p_customDestructor)(T)) { this->m_customDestructor = p_customDestructor; } | ||||
| 
 | ||||
| 	friend class MxListCursor<T>; | ||||
| 
 | ||||
|   friend class MxListCursor<T>; | ||||
| protected: | ||||
|   MxListEntry<T> *m_first; // +0x10
 | ||||
|   MxListEntry<T> *m_last;  // +0x14
 | ||||
| 	MxListEntry<T>* m_first; // +0x10
 | ||||
| 	MxListEntry<T>* m_last;  // +0x14
 | ||||
| 
 | ||||
| private: | ||||
|   void _DeleteEntry(MxListEntry<T> *match); | ||||
|   void _InsertEntry(T, MxListEntry<T> *, MxListEntry<T> *); | ||||
| 	void _DeleteEntry(MxListEntry<T>* match); | ||||
| 	void _InsertEntry(T, MxListEntry<T>*, MxListEntry<T>*); | ||||
| }; | ||||
| 
 | ||||
| template <class T> | ||||
| class MxListCursor : public MxCore | ||||
| { | ||||
| class MxListCursor : public MxCore { | ||||
| public: | ||||
|   MxListCursor(MxList<T> *p_list) { | ||||
|     m_list = p_list; | ||||
|     m_match = NULL; | ||||
|   } | ||||
| 	MxListCursor(MxList<T>* p_list) | ||||
| 	{ | ||||
| 		m_list = p_list; | ||||
| 		m_match = NULL; | ||||
| 	} | ||||
| 
 | ||||
|   MxBool Find(T p_obj); | ||||
|   void Detach(); | ||||
|   void Destroy(); | ||||
|   MxBool Next(T& p_obj); | ||||
|   MxBool Current(T& p_obj); | ||||
|   void Advance(); | ||||
|   MxBool HasMatch() { return m_match != NULL; } | ||||
|   void SetValue(T p_obj); | ||||
|   void Head() { m_match = m_list->m_first; } | ||||
|   void Reset() { m_match = NULL; } | ||||
|   void Prepend(T p_newobj); | ||||
| 	MxBool Find(T p_obj); | ||||
| 	void Detach(); | ||||
| 	void Destroy(); | ||||
| 	MxBool Next(T& p_obj); | ||||
| 	MxBool Current(T& p_obj); | ||||
| 	void Advance(); | ||||
| 	MxBool HasMatch() { return m_match != NULL; } | ||||
| 	void SetValue(T p_obj); | ||||
| 	void Head() { m_match = m_list->m_first; } | ||||
| 	void Reset() { m_match = NULL; } | ||||
| 	void Prepend(T p_newobj); | ||||
| 
 | ||||
| private: | ||||
|   MxList<T> *m_list; | ||||
|   MxListEntry<T> *m_match; | ||||
| 	MxList<T>* m_list; | ||||
| 	MxListEntry<T>* m_match; | ||||
| }; | ||||
| 
 | ||||
| // Unclear purpose
 | ||||
| template <class T> | ||||
| class MxListCursorChild : public MxListCursor<T>  | ||||
| { | ||||
| class MxListCursorChild : public MxListCursor<T> { | ||||
| public: | ||||
|   MxListCursorChild(MxList<T> *p_list) : MxListCursor<T>(p_list) {} | ||||
| 	MxListCursorChild(MxList<T>* p_list) : MxListCursor<T>(p_list) {} | ||||
| }; | ||||
| 
 | ||||
| // Unclear purpose
 | ||||
| template <class T> | ||||
| class MxListCursorChildChild : public MxListCursorChild<T> | ||||
| { | ||||
| class MxListCursorChildChild : public MxListCursorChild<T> { | ||||
| public: | ||||
|   MxListCursorChildChild(MxList<T> *p_list) : MxListCursorChild<T>(p_list) {} | ||||
| 	MxListCursorChildChild(MxList<T>* p_list) : MxListCursorChild<T>(p_list) {} | ||||
| }; | ||||
| 
 | ||||
| template <class T> | ||||
| MxList<T>::~MxList() | ||||
| { | ||||
|   DeleteAll(); | ||||
| 	DeleteAll(); | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxList<T>::DeleteAll() | ||||
| { | ||||
|   for (MxListEntry<T> *t = m_first;;) { | ||||
|     if (!t) | ||||
|       break; | ||||
| 	for (MxListEntry<T>* t = m_first;;) { | ||||
| 		if (!t) | ||||
| 			break; | ||||
| 
 | ||||
|     MxListEntry<T> *next = t->m_next; | ||||
|     m_customDestructor(t->GetValue()); | ||||
|     delete t; | ||||
|     t = next; | ||||
|   } | ||||
| 		MxListEntry<T>* next = t->m_next; | ||||
| 		this->m_customDestructor(t->GetValue()); | ||||
| 		delete t; | ||||
| 		t = next; | ||||
| 	} | ||||
| 
 | ||||
|   m_count = 0; | ||||
|   m_last = NULL; | ||||
|   m_first = NULL; | ||||
| 	this->m_count = 0; | ||||
| 	m_last = NULL; | ||||
| 	m_first = NULL; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxList<T>::Append(T p_newobj) | ||||
| { | ||||
|   MxListEntry<T> *currentLast = this->m_last; | ||||
|   MxListEntry<T> *newEntry = new MxListEntry<T>(p_newobj, currentLast); | ||||
| 	MxListEntry<T>* currentLast = this->m_last; | ||||
| 	MxListEntry<T>* newEntry = new MxListEntry<T>(p_newobj, currentLast); | ||||
| 
 | ||||
|   if (currentLast) | ||||
|     currentLast->m_next = newEntry; | ||||
|   else | ||||
|     this->m_first = newEntry; | ||||
|      | ||||
|   this->m_last = newEntry; | ||||
|   this->m_count++; | ||||
| 	if (currentLast) | ||||
| 		currentLast->m_next = newEntry; | ||||
| 	else | ||||
| 		this->m_first = newEntry; | ||||
| 
 | ||||
| 	this->m_last = newEntry; | ||||
| 	this->m_count++; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxList<T>::_InsertEntry(T p_newobj, MxListEntry<T> *p_prev, MxListEntry<T> *p_next) | ||||
| inline void MxList<T>::_InsertEntry(T p_newobj, MxListEntry<T>* p_prev, MxListEntry<T>* p_next) | ||||
| { | ||||
|   MxListEntry<T> *newEntry = new MxListEntry<T>(p_newobj, p_prev, p_next); | ||||
| 	MxListEntry<T>* newEntry = new MxListEntry<T>(p_newobj, p_prev, p_next); | ||||
| 
 | ||||
|   if (p_prev) | ||||
|     p_prev->m_next = newEntry; | ||||
|   else | ||||
|     this->m_first = newEntry; | ||||
| 	if (p_prev) | ||||
| 		p_prev->m_next = newEntry; | ||||
| 	else | ||||
| 		this->m_first = newEntry; | ||||
| 
 | ||||
|   if (p_next) | ||||
|     p_next->m_prev = newEntry; | ||||
|   else | ||||
|     this->m_last = newEntry; | ||||
| 	if (p_next) | ||||
| 		p_next->m_prev = newEntry; | ||||
| 	else | ||||
| 		this->m_last = newEntry; | ||||
| 
 | ||||
|   this->m_count++; | ||||
| 	this->m_count++; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxList<T>::_DeleteEntry(MxListEntry<T> *match) | ||||
| inline void MxList<T>::_DeleteEntry(MxListEntry<T>* match) | ||||
| { | ||||
|   MxListEntry<T> **pPrev = &match->m_prev; | ||||
|   MxListEntry<T> **pNext = &match->m_next; | ||||
| 	MxListEntry<T>** pPrev = &match->m_prev; | ||||
| 	MxListEntry<T>** pNext = &match->m_next; | ||||
| 
 | ||||
|   if (match->m_prev) | ||||
|     match->m_prev->m_next = *pNext; | ||||
|   else | ||||
|     m_first = *pNext; | ||||
| 	if (match->m_prev) | ||||
| 		match->m_prev->m_next = *pNext; | ||||
| 	else | ||||
| 		m_first = *pNext; | ||||
| 
 | ||||
|   if (*pNext) | ||||
|     (*pNext)->m_prev = *pPrev; | ||||
|   else | ||||
|     m_last = *pPrev; | ||||
| 	if (*pNext) | ||||
| 		(*pNext)->m_prev = *pPrev; | ||||
| 	else | ||||
| 		m_last = *pPrev; | ||||
| 
 | ||||
|   delete match; | ||||
|   m_count--; | ||||
| 	delete match; | ||||
| 	this->m_count--; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline MxBool MxListCursor<T>::Find(T p_obj) | ||||
| { | ||||
|   for (m_match = m_list->m_first; | ||||
|     m_match && m_list->Compare(m_match->m_obj, p_obj); | ||||
|     m_match = m_match->m_next); | ||||
| 	for (m_match = m_list->m_first; m_match && m_list->Compare(m_match->m_obj, p_obj); m_match = m_match->m_next) | ||||
| 		; | ||||
| 
 | ||||
|   return m_match != NULL; | ||||
| 	return m_match != NULL; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxListCursor<T>::Detach() | ||||
| { | ||||
|   m_list->_DeleteEntry(m_match); | ||||
|   m_match = NULL; | ||||
| 	m_list->_DeleteEntry(m_match); | ||||
| 	m_match = NULL; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxListCursor<T>::Destroy() | ||||
| { | ||||
|   m_list->m_customDestructor(m_match->GetValue()); | ||||
| 	m_list->m_customDestructor(m_match->GetValue()); | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline MxBool MxListCursor<T>::Next(T& p_obj) | ||||
| { | ||||
|   if (!m_match) | ||||
|     m_match = m_list->m_first; | ||||
|   else | ||||
|     m_match = m_match->m_next; | ||||
| 	if (!m_match) | ||||
| 		m_match = m_list->m_first; | ||||
| 	else | ||||
| 		m_match = m_match->m_next; | ||||
| 
 | ||||
|   if (m_match) | ||||
|     p_obj = m_match->GetValue(); | ||||
| 	if (m_match) | ||||
| 		p_obj = m_match->GetValue(); | ||||
| 
 | ||||
|   return m_match != NULL; | ||||
| 	return m_match != NULL; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline MxBool MxListCursor<T>::Current(T& p_obj) | ||||
| { | ||||
|   if (m_match) | ||||
|     p_obj = m_match->GetValue(); | ||||
| 	if (m_match) | ||||
| 		p_obj = m_match->GetValue(); | ||||
| 
 | ||||
|   return m_match != NULL; | ||||
| 	return m_match != NULL; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxListCursor<T>::Advance() | ||||
| { | ||||
|   if (!m_match) | ||||
|     m_match = m_list->m_first; | ||||
|   else | ||||
|     m_match = m_match->m_next; | ||||
| 	if (!m_match) | ||||
| 		m_match = m_list->m_first; | ||||
| 	else | ||||
| 		m_match = m_match->m_next; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxListCursor<T>::SetValue(T p_obj) | ||||
| { | ||||
|   if (m_match) | ||||
|     m_match->m_obj = p_obj; | ||||
| 	if (m_match) | ||||
| 		m_match->m_obj = p_obj; | ||||
| } | ||||
| 
 | ||||
| template <class T> | ||||
| inline void MxListCursor<T>::Prepend(T p_newobj) | ||||
| { | ||||
|   if (m_match) | ||||
|     m_list->_InsertEntry(p_newobj, m_match->m_prev, m_match); | ||||
| 	if (m_match) | ||||
| 		m_list->_InsertEntry(p_newobj, m_match->m_prev, m_match); | ||||
| } | ||||
| 
 | ||||
| #endif // MXLIST_H
 | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Christian Semmler
					Christian Semmler