mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-15 21:19:57 +00:00
155 lines
4.7 KiB
C++
155 lines
4.7 KiB
C++
#include "scenemanager.hpp"
|
|
|
|
#include <osg/Node>
|
|
#include <osg/Geode>
|
|
#include <osg/UserDataContainer>
|
|
|
|
#include <osgParticle/ParticleSystem>
|
|
|
|
#include <components/nifosg/nifloader.hpp>
|
|
#include <components/nif/niffile.hpp>
|
|
|
|
#include <components/vfs/manager.hpp>
|
|
|
|
#include <components/sceneutil/clone.hpp>
|
|
|
|
namespace
|
|
{
|
|
|
|
class InitWorldSpaceParticlesVisitor : public osg::NodeVisitor
|
|
{
|
|
public:
|
|
InitWorldSpaceParticlesVisitor()
|
|
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
|
|
{
|
|
}
|
|
|
|
void apply(osg::Node& node)
|
|
{
|
|
if (osg::Geode* geode = node.asGeode())
|
|
{
|
|
for (unsigned int i=0;i<geode->getNumDrawables();++i)
|
|
{
|
|
if (osgParticle::ParticleSystem* partsys = dynamic_cast<osgParticle::ParticleSystem*>(geode->getDrawable(i)))
|
|
{
|
|
// HACK: ParticleSystem has no getReferenceFrame()
|
|
if (partsys->getUserDataContainer()
|
|
&& partsys->getUserDataContainer()->getNumDescriptions() > 0
|
|
&& partsys->getUserDataContainer()->getDescriptions()[0] == "worldspace")
|
|
{
|
|
// HACK: Ignore the InverseWorldMatrix transform the geode is attached to
|
|
if (geode->getNumParents() && geode->getParent(0)->getNumParents())
|
|
transformInitialParticles(partsys, geode->getParent(0)->getParent(0));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
traverse(node);
|
|
}
|
|
|
|
void transformInitialParticles(osgParticle::ParticleSystem* partsys, osg::Node* node)
|
|
{
|
|
osg::Matrix worldMat = node->getWorldMatrices()[0];
|
|
worldMat.orthoNormalize(worldMat); // scale is already applied on the particle node
|
|
for (int i=0; i<partsys->numParticles(); ++i)
|
|
{
|
|
partsys->getParticle(i)->transformPositionVelocity(worldMat);
|
|
}
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
namespace Resource
|
|
{
|
|
|
|
SceneManager::SceneManager(const VFS::Manager *vfs, Resource::TextureManager* textureManager)
|
|
: mVFS(vfs)
|
|
, mTextureManager(textureManager)
|
|
{
|
|
}
|
|
|
|
osg::ref_ptr<const osg::Node> SceneManager::getTemplate(const std::string &name)
|
|
{
|
|
std::string normalized = name;
|
|
mVFS->normalizeFilename(normalized);
|
|
|
|
Index::iterator it = mIndex.find(normalized);
|
|
if (it == mIndex.end())
|
|
{
|
|
Files::IStreamPtr file = mVFS->get(normalized);
|
|
|
|
// TODO: add support for non-NIF formats
|
|
|
|
NifOsg::Loader loader;
|
|
osg::ref_ptr<const osg::Node> loaded = loader.load(Nif::NIFFilePtr(new Nif::NIFFile(file, normalized)), mTextureManager);
|
|
|
|
mIndex[normalized] = loaded;
|
|
return loaded;
|
|
}
|
|
else
|
|
return it->second;
|
|
}
|
|
|
|
osg::ref_ptr<osg::Node> SceneManager::createInstance(const std::string &name)
|
|
{
|
|
osg::ref_ptr<const osg::Node> scene = getTemplate(name);
|
|
osg::ref_ptr<osg::Node> cloned = osg::clone(scene.get(), SceneUtil::CopyOp());
|
|
return cloned;
|
|
}
|
|
|
|
osg::ref_ptr<osg::Node> SceneManager::createInstance(const std::string &name, osg::Group* parentNode)
|
|
{
|
|
osg::ref_ptr<osg::Node> cloned = createInstance(name);
|
|
attachTo(cloned, parentNode);
|
|
return cloned;
|
|
}
|
|
|
|
osg::ref_ptr<const NifOsg::KeyframeHolder> SceneManager::getKeyframes(const std::string &name)
|
|
{
|
|
std::string normalized = name;
|
|
mVFS->normalizeFilename(normalized);
|
|
|
|
KeyframeIndex::iterator it = mKeyframeIndex.find(normalized);
|
|
if (it == mKeyframeIndex.end())
|
|
{
|
|
Files::IStreamPtr file = mVFS->get(normalized);
|
|
|
|
NifOsg::Loader loader;
|
|
osg::ref_ptr<NifOsg::KeyframeHolder> loaded (new NifOsg::KeyframeHolder);
|
|
loader.loadKf(Nif::NIFFilePtr(new Nif::NIFFile(file, normalized)), *loaded.get());
|
|
|
|
mKeyframeIndex[normalized] = loaded;
|
|
return loaded;
|
|
}
|
|
else
|
|
return it->second;
|
|
}
|
|
|
|
void SceneManager::attachTo(osg::Node *instance, osg::Group *parentNode) const
|
|
{
|
|
parentNode->addChild(instance);
|
|
InitWorldSpaceParticlesVisitor visitor;
|
|
instance->accept(visitor);
|
|
}
|
|
|
|
void SceneManager::releaseGLObjects(osg::State *state)
|
|
{
|
|
for (Index::iterator it = mIndex.begin(); it != mIndex.end(); ++it)
|
|
{
|
|
it->second->releaseGLObjects(state);
|
|
}
|
|
}
|
|
|
|
const VFS::Manager* SceneManager::getVFS() const
|
|
{
|
|
return mVFS;
|
|
}
|
|
|
|
Resource::TextureManager* SceneManager::getTextureManager()
|
|
{
|
|
return mTextureManager;
|
|
}
|
|
|
|
}
|