1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-27 03:35:27 +00:00
OpenMW/apps/openmw/mwrender/objects.cpp

267 lines
7.4 KiB
C++
Raw Normal View History

2011-10-31 23:59:16 -04:00
#include "objects.hpp"
2012-02-06 10:41:13 +01:00
2013-03-29 09:04:02 +01:00
#include <cmath>
#include <osg/Group>
#include <osg/Geode>
#include <osg/UserDataContainer>
#include <osg/Version>
2012-02-06 10:41:13 +01:00
#include <osgParticle/ParticleSystem>
#include <osgParticle/ParticleProcessor>
2014-02-23 20:11:05 +01:00
#include <components/resource/scenemanager.hpp>
#include <components/sceneutil/visitor.hpp>
#include <components/sceneutil/positionattitudetransform.hpp>
#include "../mwworld/ptr.hpp"
2012-07-30 23:28:14 +04:00
#include "../mwworld/class.hpp"
#include "animation.hpp"
2015-04-15 22:11:38 +02:00
#include "npcanimation.hpp"
2015-04-19 03:05:47 +02:00
#include "creatureanimation.hpp"
2015-05-23 05:42:37 +02:00
#include "vismask.hpp"
namespace
{
/// Removes all particle systems and related nodes in a subgraph.
class RemoveParticlesVisitor : public osg::NodeVisitor
{
public:
RemoveParticlesVisitor()
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
{ }
virtual void apply(osg::Node &node)
{
2015-06-17 15:13:17 +02:00
if (dynamic_cast<osgParticle::ParticleProcessor*>(&node))
mToRemove.push_back(&node);
traverse(node);
}
2011-10-31 23:59:16 -04:00
2015-06-17 15:13:17 +02:00
virtual void apply(osg::Geode& geode)
{
std::vector<osgParticle::ParticleSystem*> partsysVector;
for (unsigned int i=0; i<geode.getNumDrawables(); ++i)
{
osg::Drawable* drw = geode.getDrawable(i);
if (osgParticle::ParticleSystem* partsys = dynamic_cast<osgParticle::ParticleSystem*>(drw))
partsysVector.push_back(partsys);
}
for (std::vector<osgParticle::ParticleSystem*>::iterator it = partsysVector.begin(); it != partsysVector.end(); ++it)
geode.removeDrawable(*it);
}
#if OSG_VERSION_GREATER_OR_EQUAL(3,3,3)
virtual void apply(osg::Drawable& drw)
{
if (osgParticle::ParticleSystem* partsys = dynamic_cast<osgParticle::ParticleSystem*>(&drw))
mToRemove.push_back(partsys);
}
#endif
2015-06-17 15:13:17 +02:00
void remove()
{
for (std::vector<osg::ref_ptr<osg::Node> >::iterator it = mToRemove.begin(); it != mToRemove.end(); ++it)
{
// FIXME: a Drawable might have more than one parent
osg::Node* node = *it;
if (node->getNumParents())
node->getParent(0)->removeChild(node);
}
mToRemove.clear();
}
private:
std::vector<osg::ref_ptr<osg::Node> > mToRemove;
};
}
2011-10-31 23:59:16 -04:00
namespace MWRender
{
Objects::Objects(Resource::ResourceSystem* resourceSystem, osg::ref_ptr<osg::Group> rootNode)
2015-05-23 22:44:00 +02:00
: mRootNode(rootNode)
, mResourceSystem(resourceSystem)
2012-02-06 10:41:13 +01:00
{
}
Objects::~Objects()
{
for(PtrAnimationMap::iterator iter = mObjects.begin();iter != mObjects.end();++iter)
delete iter->second;
mObjects.clear();
for (CellMap::iterator iter = mCellSceneNodes.begin(); iter != mCellSceneNodes.end(); ++iter)
iter->second->getParent(0)->removeChild(iter->second);
mCellSceneNodes.clear();
2011-11-17 17:10:27 -05:00
}
2012-02-06 10:41:13 +01:00
2013-08-07 03:51:57 -07:00
void Objects::insertBegin(const MWWorld::Ptr& ptr)
2012-02-06 10:41:13 +01:00
{
osg::ref_ptr<osg::Group> cellnode;
CellMap::iterator found = mCellSceneNodes.find(ptr.getCell());
if (found == mCellSceneNodes.end())
2011-11-04 21:57:39 -04:00
{
cellnode = new osg::Group;
mRootNode->addChild(cellnode);
2011-11-04 21:57:39 -04:00
mCellSceneNodes[ptr.getCell()] = cellnode;
}
else
cellnode = found->second;
2011-11-04 21:48:52 -04:00
osg::ref_ptr<SceneUtil::PositionAttitudeTransform> insert (new SceneUtil::PositionAttitudeTransform);
cellnode->addChild(insert);
2012-03-24 22:03:08 -04:00
insert->getOrCreateUserDataContainer()->addUserObject(new PtrHolder(ptr));
const float *f = ptr.getRefData().getPosition().pos;
insert->setPosition(osg::Vec3(f[0], f[1], f[2]));
const float scale = ptr.getCellRef().getScale();
osg::Vec3f scaleVec(scale, scale, scale);
ptr.getClass().adjustScale(ptr, scaleVec, true);
insert->setScale(scaleVec);
2011-11-02 22:41:48 -04:00
ptr.getRefData().setBaseNode(insert);
2011-10-31 23:59:16 -04:00
}
2012-02-06 10:41:13 +01:00
void Objects::insertModel(const MWWorld::Ptr &ptr, const std::string &mesh, bool animated, bool allowLight)
2012-02-06 10:41:13 +01:00
{
2013-08-07 03:51:57 -07:00
insertBegin(ptr);
2015-05-28 15:44:58 +02:00
std::auto_ptr<ObjectAnimation> anim (new ObjectAnimation(ptr, mesh, mResourceSystem, animated, allowLight));
if (!allowLight)
{
RemoveParticlesVisitor visitor;
anim->getObjectRoot()->accept(visitor);
visitor.remove();
2011-11-04 21:57:39 -04:00
}
2013-02-23 04:06:05 +01:00
mObjects.insert(std::make_pair(ptr, anim.release()));
}
void Objects::insertCreature(const MWWorld::Ptr &ptr, const std::string &mesh, bool weaponsShields)
{
insertBegin(ptr);
2015-05-23 05:42:37 +02:00
ptr.getRefData().getBaseNode()->setNodeMask(Mask_Actor);
// CreatureAnimation
2015-04-19 03:05:47 +02:00
std::auto_ptr<Animation> anim;
if (weaponsShields)
anim.reset(new CreatureWeaponAnimation(ptr, mesh, mResourceSystem));
else
anim.reset(new CreatureAnimation(ptr, mesh, mResourceSystem));
mObjects.insert(std::make_pair(ptr, anim.release()));
}
void Objects::insertNPC(const MWWorld::Ptr &ptr)
{
2015-04-15 22:11:38 +02:00
insertBegin(ptr);
2015-05-23 05:42:37 +02:00
ptr.getRefData().getBaseNode()->setNodeMask(Mask_Actor);
std::auto_ptr<NpcAnimation> anim (new NpcAnimation(ptr, osg::ref_ptr<osg::Group>(ptr.getRefData().getBaseNode()), mResourceSystem));
2015-04-19 15:03:08 +02:00
2015-04-15 22:11:38 +02:00
mObjects.insert(std::make_pair(ptr, anim.release()));
2011-10-31 23:59:16 -04:00
}
2012-02-06 10:41:13 +01:00
2015-05-22 00:55:43 +02:00
bool Objects::removeObject (const MWWorld::Ptr& ptr)
2012-02-06 10:41:13 +01:00
{
if(!ptr.getRefData().getBaseNode())
return true;
2012-04-28 20:42:53 +02:00
PtrAnimationMap::iterator iter = mObjects.find(ptr);
if(iter != mObjects.end())
2012-02-06 10:41:13 +01:00
{
delete iter->second;
mObjects.erase(iter);
2012-02-06 10:41:13 +01:00
ptr.getRefData().getBaseNode()->getParent(0)->removeChild(ptr.getRefData().getBaseNode());
ptr.getRefData().setBaseNode(NULL);
return true;
}
return false;
2011-10-31 23:59:16 -04:00
}
2011-11-02 00:13:33 -04:00
2011-11-04 21:48:52 -04:00
void Objects::removeCell(const MWWorld::CellStore* store)
2012-02-06 10:41:13 +01:00
{
for(PtrAnimationMap::iterator iter = mObjects.begin();iter != mObjects.end();)
2011-11-04 21:48:52 -04:00
{
if(iter->first.getCell() == store)
{
delete iter->second;
mObjects.erase(iter++);
}
else
++iter;
2011-11-04 21:48:52 -04:00
}
CellMap::iterator cell = mCellSceneNodes.find(store);
if(cell != mCellSceneNodes.end())
{
cell->second->getParent(0)->removeChild(cell->second);
mCellSceneNodes.erase(cell);
}
}
2012-04-02 19:37:24 +02:00
2015-05-14 17:34:55 +02:00
void Objects::updatePtr(const MWWorld::Ptr &old, const MWWorld::Ptr &cur)
2012-07-30 23:28:14 +04:00
{
2015-05-26 16:40:44 +02:00
osg::Node* objectNode = cur.getRefData().getBaseNode();
if (!objectNode)
return;
MWWorld::CellStore *newCell = cur.getCell();
2012-07-30 23:28:14 +04:00
2015-05-14 17:34:55 +02:00
osg::Group* cellnode;
2012-07-30 23:28:14 +04:00
if(mCellSceneNodes.find(newCell) == mCellSceneNodes.end()) {
2015-05-14 17:34:55 +02:00
cellnode = new osg::Group;
mRootNode->addChild(cellnode);
mCellSceneNodes[newCell] = cellnode;
2012-07-30 23:28:14 +04:00
} else {
2015-05-14 17:34:55 +02:00
cellnode = mCellSceneNodes[newCell];
2012-07-30 23:28:14 +04:00
}
osg::UserDataContainer* userDataContainer = objectNode->getUserDataContainer();
if (userDataContainer)
for (unsigned int i=0; i<userDataContainer->getNumUserObjects(); ++i)
{
if (dynamic_cast<PtrHolder*>(userDataContainer->getUserObject(i)))
userDataContainer->setUserObject(i, new PtrHolder(cur));
}
2015-05-14 17:34:55 +02:00
if (objectNode->getNumParents())
objectNode->getParent(0)->removeChild(objectNode);
cellnode->addChild(objectNode);
PtrAnimationMap::iterator iter = mObjects.find(old);
if(iter != mObjects.end())
{
2015-05-14 17:34:55 +02:00
Animation *anim = iter->second;
mObjects.erase(iter);
anim->updatePtr(cur);
mObjects[cur] = anim;
}
2012-07-30 23:28:14 +04:00
}
Animation* Objects::getAnimation(const MWWorld::Ptr &ptr)
{
PtrAnimationMap::const_iterator iter = mObjects.find(ptr);
if(iter != mObjects.end())
return iter->second;
return NULL;
2012-07-30 23:28:14 +04:00
}
}