2020-09-30 16:53:55 +02:00

298 lines
7.3 KiB
C++

/***************************************************************************\
*
* File: TreeNode.inl
*
* History:
* 1/05/2000: JStall: Created
*
* Copyright (C) 2000 by Microsoft Corporation. All rights reserved.
*
\***************************************************************************/
#if !defined(BASE__TreeNode_inl__INCLUDED)
#define BASE__TreeNode_inl__INCLUDED
#pragma once
/***************************************************************************\
*****************************************************************************
*
* class TreeNode
*
*****************************************************************************
\***************************************************************************/
//------------------------------------------------------------------------------
template <class T>
inline
TreeNodeT<T>::TreeNodeT()
{
}
//------------------------------------------------------------------------------
template <class T>
inline
TreeNodeT<T>::~TreeNodeT()
{
}
//------------------------------------------------------------------------------
template <class T>
inline T *
TreeNodeT<T>::GetTopSibling() const
{
T * pTreeNode = const_cast<T *> (this);
while (pTreeNode->m_ptnPrev != NULL) {
pTreeNode = pTreeNode->m_ptnPrev;
}
return pTreeNode;
}
//------------------------------------------------------------------------------
template <class T>
inline T *
TreeNodeT<T>::GetBottomSibling() const
{
TreeNodeT<T> * pTreeNode = const_cast<TreeNodeT<T> *> (this);
while (pTreeNode->m_ptnNext != NULL) {
pTreeNode = pTreeNode->m_ptnNext;
}
return (T *) pTreeNode;
}
//------------------------------------------------------------------------------
template <class T>
inline T *
TreeNodeT<T>::GetParent() const
{
return (T *) m_ptnParent;
}
//------------------------------------------------------------------------------
template <class T>
inline T *
TreeNodeT<T>::GetPrev() const
{
return (T *) m_ptnPrev;
}
//------------------------------------------------------------------------------
template <class T>
inline T *
TreeNodeT<T>::GetNext() const
{
return (T *) m_ptnNext;
}
//------------------------------------------------------------------------------
template <class T>
inline T *
TreeNodeT<T>::GetTopChild() const
{
return (T *) m_ptnChild;
}
//------------------------------------------------------------------------------
template <class T>
inline T *
TreeNodeT<T>::GetBottomChild() const
{
if (m_ptnChild != NULL) {
return GetTopChild()->GetBottomSibling();
} else {
return NULL;
}
}
/***************************************************************************\
*
* TreeNodeT::DoLink
*
* DoLink chains _this_ node into the tree. A sibling and a parent (may) be
* given as reference, depending on the situation. This node (the one
* being linked) _must_ have already been unlinked previously.
*
\***************************************************************************/
template <class T>
void
TreeNodeT<T>::DoLink(
IN T * ptnParent, // New parent
IN T * ptnSibling, // Node to link to this (unlinked) Node
IN ELinkType lt) // Position of ptn relative to this
{
// Check parameters
AssertWritePtr(ptnParent);
// Verify the TreeNode is unlinked
Assert((m_ptnNext == NULL) && (m_ptnPrev == NULL) && (m_ptnParent == NULL));
//
// Link this TreeNode to the parent.
//
m_ptnParent = ptnParent;
TreeNodeT<T> * ptnOldChild = ptnParent->m_ptnChild;
if (ptnOldChild == NULL) {
//
// Simple case, no siblings so just do it.
//
AssertMsg(ptnSibling == NULL, "Parent doesn't have any children");
ptnParent->m_ptnChild = this;
} else {
//
// Uggh- complex case, so need to Link this TreeNode to its new siblings
//
switch (lt) {
case ltBefore:
AssertWritePtr(ptnSibling);
m_ptnNext = ptnSibling;
m_ptnPrev = ptnSibling->m_ptnPrev;
ptnSibling->m_ptnPrev = this;
if (m_ptnPrev != NULL) {
m_ptnPrev->m_ptnNext = this;
}
break;
case ltBehind:
AssertWritePtr(ptnSibling);
m_ptnPrev = ptnSibling;
m_ptnNext = ptnSibling->m_ptnNext;
ptnSibling->m_ptnNext = this;
if (m_ptnNext != NULL) {
m_ptnNext->m_ptnPrev = this;
}
break;
case ltAny:
case ltTop:
ptnParent->m_ptnChild = this;
m_ptnNext = ptnOldChild;
if (ptnOldChild != NULL) {
ptnOldChild->m_ptnPrev = this;
}
break;
case ltBottom:
ptnOldChild = ptnOldChild->GetBottomSibling();
ptnOldChild->m_ptnNext = this;
m_ptnPrev = ptnOldChild;
break;
default:
AssertMsg(0, "Unknown link type");
}
}
}
/***************************************************************************\
*
* TreeNode::DoUnlink
*
* DoUnlink() removes this TreeNode from the TreeNode tree. The parents and
* siblings of the TreeNode are properly modified.
*
\***************************************************************************/
template <class T>
void
TreeNodeT<T>::DoUnlink()
{
//
// Unlink from the parent
//
if (m_ptnParent != NULL) {
if (m_ptnParent->m_ptnChild == this) {
m_ptnParent->m_ptnChild = m_ptnNext;
}
}
//
// Unlink from siblings
//
if (m_ptnNext != NULL) {
m_ptnNext->m_ptnPrev = m_ptnPrev;
}
if (m_ptnPrev != NULL) {
m_ptnPrev->m_ptnNext = m_ptnNext;
}
m_ptnParent = NULL;
m_ptnNext = NULL;
m_ptnPrev = NULL;
}
#if DBG
template <class T>
inline BOOL
TreeNodeT<T>::DEBUG_IsChild(const TreeNodeT<T> * pChild) const
{
TreeNodeT<T> * pCur = m_ptnChild;
while (pCur != NULL) {
if (pCur == pChild) {
return TRUE;
}
pCur = pCur->m_ptnNext;
}
return FALSE;
}
/***************************************************************************\
*
* TreeNodeT<T>::DEBUG_AssertValid
*
* DEBUG_AssertValid() provides a DEBUG-only mechanism to perform rich
* validation of an object to attempt to determine if the object is still
* valid. This is used during debugging to help track damaged objects
*
\***************************************************************************/
template <class T>
void
TreeNodeT<T>::DEBUG_AssertValid() const
{
if (m_ptnParent != NULL) {
Assert(m_ptnParent->DEBUG_IsChild(this));
}
if (m_ptnNext != NULL) {
Assert(m_ptnNext->m_ptnPrev == this);
}
if (m_ptnPrev != NULL) {
Assert(m_ptnPrev->m_ptnNext == this);
}
if (m_ptnChild != NULL) {
Assert(m_ptnChild->m_ptnParent == this);
}
}
#endif // DBG
#endif // BASE__TreeNode_inl__INCLUDED