[BACK]Return to SoChildList.c++ CVS log [TXT][DIR] Up to [Development] / inventor / lib / database / src / so

File: [Development] / inventor / lib / database / src / so / SoChildList.c++ (download)

Revision 1.1.1.1 (vendor branch), Tue Aug 15 12:56:17 2000 UTC (17 years, 2 months ago) by naaman
Branch: sgi, MAIN
CVS Tags: start, release-2_1_5-9, release-2_1_5-8, release-2_1_5-10, HEAD
Changes since 1.1: +0 -0 lines

Initial check-in based on 2.1.5 (SGI IRIX) source tree.

/*
 *
 *  Copyright (C) 2000 Silicon Graphics, Inc.  All Rights Reserved. 
 *
 *  This library is free software; you can redistribute it and/or
 *  modify it under the terms of the GNU Lesser General Public
 *  License as published by the Free Software Foundation; either
 *  version 2.1 of the License, or (at your option) any later version.
 *
 *  This library is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 *  Lesser General Public License for more details.
 *
 *  Further, this software is distributed without any warranty that it is
 *  free of the rightful claim of any third person regarding infringement
 *  or the like.  Any license provided herein, whether implied or
 *  otherwise, applies only to this software file.  Patent licenses, if
 *  any, provided herein do not apply to combinations of this program with
 *  other software, or any other product whatsoever.
 * 
 *  You should have received a copy of the GNU Lesser General Public
 *  License along with this library; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 *  Contact information: Silicon Graphics, Inc., 1600 Amphitheatre Pkwy,
 *  Mountain View, CA  94043, or:
 * 
 *  http://www.sgi.com 
 * 
 *  For further information regarding this notice, see: 
 * 
 *  http://oss.sgi.com/projects/GenInfo/NoticeExplan/
 *
 */

/*
 * Copyright (C) 1990,91   Silicon Graphics, Inc.
 *
 _______________________________________________________________________
 ______________  S I L I C O N   G R A P H I C S   I N C .  ____________
 |
 |   $Revision: 1.1.1.1 $
 |
 |   Classes:
 |	SoChildList, SoHiddenChildList
 |
 |   Author(s)		: Paul S. Strauss
 |
 ______________  S I L I C O N   G R A P H I C S   I N C .  ____________
 _______________________________________________________________________
 */

#include <Inventor/misc/SoChildList.h>
#include <Inventor/actions/SoAction.h>

//////////////////////////////////////////////////////////////////////////////
//
// SoChildList class.
//
//////////////////////////////////////////////////////////////////////////////

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Constructor.
//
// Use: public

SoChildList::SoChildList(SoNode *parentNode) : SoNodeList()
//
////////////////////////////////////////////////////////////////////////
{
    parent = parentNode;
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Constructor.
//
// Use: public

SoChildList::SoChildList(SoNode *parentNode, int size) : SoNodeList(size)
//
////////////////////////////////////////////////////////////////////////
{
    parent = parentNode;
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Constructor.
//
// Use: public

SoChildList::SoChildList(SoNode *parentNode, const SoChildList &l) :
	SoNodeList(l)
//
////////////////////////////////////////////////////////////////////////
{
    parent = parentNode;
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Destructor.
//
// Use: public

SoChildList::~SoChildList()
//
////////////////////////////////////////////////////////////////////////
{
    truncate(0);
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Adds a child to the end of the list.
//
// Use: public

void
SoChildList::append(SoNode *child)
//
////////////////////////////////////////////////////////////////////////
{
    SoNodeList::append(child);

    // Express interest by parent in child's modification
    child->addAuditor(parent, SoNotRec::PARENT);

    // the parent has changed; notify its auditors
    parent->startNotify();
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Insert given child into list before child with given index
//
// Use: public

void
SoChildList::insert(SoNode *child, int addBefore)
//
////////////////////////////////////////////////////////////////////////
{
    int i;

    SoNodeList::insert(child, addBefore);

    // If any paths go through the parent, make sure they get updated
    for (i = 0; i < auditors.getLength(); i++)
	((SoPath *)auditors[i])->insertIndex(parent, addBefore);

    // Express interest by parent in child's modification
    child->addAuditor(parent, SoNotRec::PARENT);

    // the parent has changed; notify its auditors
    parent->startNotify();
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Removes child with given index.
//
// Use: public

void
SoChildList::remove(int which)
//
////////////////////////////////////////////////////////////////////////
{
    int i;

    // Remove interest of parent in child
    (*this)[which]->removeAuditor(parent, SoNotRec::PARENT);

    // If any paths go through the parent, make sure they get updated
    for (i = 0; i < auditors.getLength(); i++)
	((SoPath *)auditors[i])->removeIndex(parent, which);

    SoNodeList::remove(which);

    // the parent has changed; notify its auditors
    parent->startNotify();
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Removes all children from one with given index on, inclusive.
//
// Use: public

void
SoChildList::truncate(int start)
//
////////////////////////////////////////////////////////////////////////
{
    int which;

    // Remove children from right to left. This allows the indices to
    // remain correct during the removal and also keeps the paths that
    // audit the group from doing too much work. (A path has to adjust
    // indices if we remove a child before the one in the path.) It
    // also minimizes shuffling of entries in the children array.

    for (which = getLength() - 1; which >= start; --which) {
	int i;

	// Remove interest of parent in child
	(*this)[which]->removeAuditor(parent, SoNotRec::PARENT);

	// If any paths go through the parent, make sure they get updated
	for (i = 0; i < auditors.getLength(); i++)
	    ((SoPath *)auditors[i])->removeIndex(parent, which);

	SoNodeList::remove(which);
    }

    // the parent has changed; notify its auditors
    parent->startNotify();
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Copies a list of children.
//
// Use: public

void
SoChildList::copy(const SoChildList &cList)
//
////////////////////////////////////////////////////////////////////////
{
    int i;

    // Truncate to get rid of old children
    truncate(0);

    // Copy children
    SoNodeList::copy(cList);

    // Express parent's interest in all children
    for (i = 0; i < getLength(); i++)
	(*this)[i]->addAuditor(parent, SoNotRec::PARENT);

    // the parent has changed; notify its auditors
    parent->startNotify();
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Replaces child with given index with new child.
//
// Use: public

void
SoChildList::set(int which, SoNode *child)
//
////////////////////////////////////////////////////////////////////////
{
    int i;

    // Remove interest of parent in old child
    (*this)[which]->removeAuditor(parent, SoNotRec::PARENT);

    // If any paths go through the parent, make sure they get updated
    for (i = 0; i < auditors.getLength(); i++)
	((SoPath *)auditors[i])->replaceIndex(parent, which, child);

    // Replace child
    SoNodeList::set(which, child);

    // Express parent's interest in new child
    child->addAuditor(parent, SoNotRec::PARENT);

    // the parent has changed; notify its auditors
    parent->startNotify();
}

////////////////////////////////////////////////////////////////////////
//
// Description:
//    Traverses all children between the two given indices, inclusive,
//    to apply the given action to them. Stops if the termination
//    condition for the action has been reached.
//
// Use: public

void
SoChildList::traverse(SoAction *action, int firstChild, int lastChild)
//
////////////////////////////////////////////////////////////////////////
{
    int i;

    // Optimization:  If the node this childList is part of is not in
    // the path, then the path code can't change as we traverse its
    // children, and we can use a more efficient traversal:
    SoAction::PathCode	pc = action->getCurPathCode();

    if (pc == SoAction::NO_PATH || pc == SoAction::BELOW_PATH) {
	//begin by pushing nonexistent child:
	action->pushCurPath();
	for (i = firstChild; i <= lastChild; i++) {
	    SoNode *child = (*this)[i];
	    // Now pop and push in one move:
	    action->popPushCurPath(i);
	    action->traverse(child);

	    // Stop if action has reached termination condition. (For
	    // example, search has found what it was looking for, or event
	    // was handled.)
	    if (action->hasTerminated())
		break;
	}	
        action->popCurPath();
    }
    else {
	for (i = firstChild; i <= lastChild; i++) {
	    SoNode	*child = (*this)[i];

	    // Never traverse a child that does not affect the state
	    // if we are off the path
	    if (pc == SoAction::OFF_PATH && ! child->affectsState())
		continue;

	    // If we are in the path, don't traverse a child that does
	    // not affect the state if it is not on a path
	    action->pushCurPath(i);

	    // If action code is now OFF_PATH, we know that the child
	    // was not on a path. Don't traverse it if it does not
	    // affect the state
	    if (action->getCurPathCode() != SoAction::OFF_PATH ||
		child->affectsState())
		action->traverse(child);

	    action->popCurPath(pc);

	    // Stop if action has reached termination condition. (For
	    // example, search has found what it was looking for, or event
	    // was handled.)
	    if (action->hasTerminated())
		break;
	}
    }
}