/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2003 Robert Osfield * * This library is open source and may be redistributed and/or modified under * the terms of the OpenSceneGraph Public License (OSGPL) version 0.0 or * (at your option) any later version. The full license is in LICENSE file * included with this distribution, and on the openscenegraph.org website. * * 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 * OpenSceneGraph Public License for more details. */ #include #include #include #include #define square(x) ((x)*(x)) using namespace osg; Geode::Geode() { } Geode::Geode(const Geode& geode,const CopyOp& copyop): Node(geode,copyop) { for(DrawableList::const_iterator itr=geode._drawables.begin(); itr!=geode._drawables.end(); ++itr) { Drawable* drawable = copyop(itr->get()); if (drawable) addDrawable(drawable); } } Geode::~Geode() { // remove reference to this from children's parent lists. for(DrawableList::iterator itr=_drawables.begin(); itr!=_drawables.end(); ++itr) { (*itr)->removeParent(this); } } bool Geode::addDrawable( Drawable *drawable ) { if (drawable && !containsDrawable(drawable)) { // note ref_ptr<> automatically handles incrementing drawable's reference count. _drawables.push_back(drawable); // register as parent of drawable. drawable->addParent(this); if (drawable->getUpdateCallback()) { setNumChildrenRequiringUpdateTraversal(getNumChildrenRequiringUpdateTraversal()+1); } dirtyBound(); return true; } else return false; } bool Geode::removeDrawable( Drawable *drawable ) { return removeDrawable(getDrawableIndex(drawable)); } bool Geode::removeDrawable(unsigned int pos,unsigned int numDrawablesToRemove) { if (pos<_drawables.size() && numDrawablesToRemove>0) { unsigned int endOfRemoveRange = pos+numDrawablesToRemove; if (endOfRemoveRange>_drawables.size()) { notify(DEBUG_INFO)<<"Warning: Geode::removeDrawable(i,numDrawablesToRemove) has been passed an excessive number"<removeParent(this); // update the number of app calbacks removed if (_drawables[i]->getUpdateCallback()) ++updateCallbackRemoved; } _drawables.erase(_drawables.begin()+pos,_drawables.begin()+endOfRemoveRange); if (updateCallbackRemoved) { setNumChildrenRequiringUpdateTraversal(getNumChildrenRequiringUpdateTraversal()-updateCallbackRemoved); } dirtyBound(); return true; } else return false; } bool Geode::replaceDrawable( Drawable *origDrawable, Drawable *newDrawable ) { if (newDrawable==NULL || origDrawable==newDrawable) return false; unsigned int pos = getDrawableIndex(origDrawable); if (pos<_drawables.size()) { return setDrawable(pos,newDrawable); } return false; } bool Geode::setDrawable( unsigned int i, Drawable* newDrawable ) { if (i<_drawables.size() && newDrawable) { Drawable* origDrawable = _drawables[i].get(); int delta = 0; if (origDrawable->getUpdateCallback()) --delta; if (newDrawable->getUpdateCallback()) ++delta; if (delta!=0) { setNumChildrenRequiringUpdateTraversal(getNumChildrenRequiringUpdateTraversal()+delta); } // remove from origDrawable's parent list. origDrawable->removeParent(this); // note ref_ptr<> automatically handles decrementing origGset's reference count, // and inccrementing newGset's reference count. _drawables[i] = newDrawable; // register as parent of child. newDrawable->addParent(this); dirtyBound(); return true; } else return false; } bool Geode::computeBound() const { _bsphere.init(); BoundingBox bb; DrawableList::const_iterator itr; for(itr=_drawables.begin(); itr!=_drawables.end(); ++itr) { bb.expandBy((*itr)->getBound()); } if (bb.valid()) { _bsphere.expandBy(bb); _bsphere_computed=true; return true; } else { _bsphere_computed=true; return false; } } void Geode::compileDrawables(State& state) { for(DrawableList::iterator itr = _drawables.begin(); itr!=_drawables.end(); ++itr) { (*itr)->compile(state); } }