mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 09:56:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			125 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			125 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "clone.hpp"
 | |
| 
 | |
| #include <osg/StateSet>
 | |
| #include <osg/Version>
 | |
| 
 | |
| #include <osgParticle/ParticleProcessor>
 | |
| #include <osgParticle/ParticleSystemUpdater>
 | |
| #include <osgParticle/Emitter>
 | |
| #include <osgParticle/Program>
 | |
| 
 | |
| #include <osgAnimation/MorphGeometry>
 | |
| 
 | |
| #include <components/sceneutil/riggeometry.hpp>
 | |
| 
 | |
| namespace SceneUtil
 | |
| {
 | |
| 
 | |
|     CopyOp::CopyOp()
 | |
|     {
 | |
|         setCopyFlags(osg::CopyOp::DEEP_COPY_NODES
 | |
|                      // Controller might need different inputs per scene instance
 | |
|                      | osg::CopyOp::DEEP_COPY_CALLBACKS
 | |
|                      | osg::CopyOp::DEEP_COPY_USERDATA);
 | |
|     }
 | |
| 
 | |
|     osg::StateSet* CopyOp::operator ()(const osg::StateSet* stateset) const
 | |
|     {
 | |
|         if (!stateset)
 | |
|             return NULL;
 | |
|         if (stateset->getDataVariance() == osg::StateSet::DYNAMIC)
 | |
|             return osg::clone(stateset, *this);
 | |
|         return const_cast<osg::StateSet*>(stateset);
 | |
|     }
 | |
| 
 | |
|     osg::Node* CopyOp::operator ()(const osg::Node* node) const
 | |
|     {
 | |
|         if (const osgParticle::ParticleProcessor* processor = dynamic_cast<const osgParticle::ParticleProcessor*>(node))
 | |
|             return operator()(processor);
 | |
|         if (const osgParticle::ParticleSystemUpdater* updater = dynamic_cast<const osgParticle::ParticleSystemUpdater*>(node))
 | |
|         {
 | |
|             osgParticle::ParticleSystemUpdater* cloned = new osgParticle::ParticleSystemUpdater(*updater, osg::CopyOp::SHALLOW_COPY);
 | |
|             mMap2[cloned] = updater->getParticleSystem(0);
 | |
|             return cloned;
 | |
|         }
 | |
|         return osg::CopyOp::operator()(node);
 | |
|     }
 | |
| 
 | |
|     osg::Drawable* CopyOp::operator ()(const osg::Drawable* drawable) const
 | |
|     {
 | |
|         if (const osgParticle::ParticleSystem* partsys = dynamic_cast<const osgParticle::ParticleSystem*>(drawable))
 | |
|             return operator()(partsys);
 | |
|         if (dynamic_cast<const osgAnimation::MorphGeometry*>(drawable))
 | |
|         {
 | |
|             osg::CopyOp copyop = *this;
 | |
|             copyop.setCopyFlags(copyop.getCopyFlags()|osg::CopyOp::DEEP_COPY_ARRAYS);
 | |
| 
 | |
| #if OSG_VERSION_LESS_THAN(3,5,0)
 | |
|             /*
 | |
| 
 | |
|             Deep copy of primitives required to work around the following (bad?) code in osg::Geometry copy constructor:
 | |
| 
 | |
|             if ((copyop.getCopyFlags() & osg::CopyOp::DEEP_COPY_ARRAYS))
 | |
|             {
 | |
|                 if (_useVertexBufferObjects)
 | |
|                 {
 | |
|                     // copying of arrays doesn't set up buffer objects so we'll need to force
 | |
|                     // Geometry to assign these, we'll do this by switching off VBO's then renabling them.
 | |
|                     setUseVertexBufferObjects(false);
 | |
|                     setUseVertexBufferObjects(true);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             In case of DEEP_COPY_PRIMITIVES=Off, DEEP_COPY_ARRAYS=On, the above code makes a modification to the original const Geometry& we copied from,
 | |
|             causing problems if we relied on the original Geometry to remain static such as when it was added to an osgUtil::IncrementalCompileOperation.
 | |
| 
 | |
|             Fixed in OSG 3.5 ( http://forum.openscenegraph.org/viewtopic.php?t=15217 ).
 | |
| 
 | |
|             */
 | |
| 
 | |
|             copyop.setCopyFlags(copyop.getCopyFlags()|osg::CopyOp::DEEP_COPY_PRIMITIVES);
 | |
| #endif
 | |
| 
 | |
|             osg::Drawable* cloned = osg::clone(drawable, copyop);
 | |
|             return cloned;
 | |
|         }
 | |
|         if (dynamic_cast<const SceneUtil::RigGeometry*>(drawable))
 | |
|         {
 | |
|             return osg::clone(drawable, *this);
 | |
|         }
 | |
| 
 | |
| 
 | |
|         return osg::CopyOp::operator()(drawable);
 | |
|     }
 | |
| 
 | |
|     osgParticle::ParticleProcessor* CopyOp::operator() (const osgParticle::ParticleProcessor* processor) const
 | |
|     {
 | |
|         osgParticle::ParticleProcessor* cloned = osg::clone(processor, osg::CopyOp::DEEP_COPY_CALLBACKS);
 | |
|         mMap[cloned] = processor->getParticleSystem();
 | |
|         return cloned;
 | |
|     }
 | |
| 
 | |
|     osgParticle::ParticleSystem* CopyOp::operator ()(const osgParticle::ParticleSystem* partsys) const
 | |
|     {
 | |
|         osgParticle::ParticleSystem* cloned = osg::clone(partsys, *this);
 | |
| 
 | |
|         for (std::map<osgParticle::ParticleProcessor*, const osgParticle::ParticleSystem*>::const_iterator it = mMap.begin(); it != mMap.end(); ++it)
 | |
|         {
 | |
|             if (it->second == partsys)
 | |
|             {
 | |
|                 it->first->setParticleSystem(cloned);
 | |
|             }
 | |
|         }
 | |
|         for (std::map<osgParticle::ParticleSystemUpdater*, const osgParticle::ParticleSystem*>::const_iterator it = mMap2.begin(); it != mMap2.end(); ++it)
 | |
|         {
 | |
|             if (it->second == partsys)
 | |
|             {
 | |
|                 osgParticle::ParticleSystemUpdater* updater = it->first;
 | |
|                 updater->removeParticleSystem(updater->getParticleSystem(0));
 | |
|                 updater->addParticleSystem(cloned);
 | |
|             }
 | |
|         }
 | |
|         return cloned;
 | |
|     }
 | |
| 
 | |
| }
 |