From ade7f092032a7e85202b966c103066590b6719ff Mon Sep 17 00:00:00 2001 From: cc9cii Date: Sat, 1 Nov 2014 00:03:09 +1100 Subject: [PATCH] Keep track of terrain in physics engine. Should resolve crash exiting after opening multiple subviews of the same cell. --- apps/opencs/view/render/cell.cpp | 6 ++--- apps/opencs/view/render/cell.hpp | 1 + apps/opencs/view/world/physicssystem.cpp | 30 ++++++++++++++++++++---- apps/opencs/view/world/physicssystem.hpp | 7 +++--- 4 files changed, 33 insertions(+), 11 deletions(-) diff --git a/apps/opencs/view/render/cell.cpp b/apps/opencs/view/render/cell.cpp index f21d6b40b4..52e2335981 100644 --- a/apps/opencs/view/render/cell.cpp +++ b/apps/opencs/view/render/cell.cpp @@ -61,7 +61,7 @@ bool CSVRender::Cell::addObjects (int start, int end) CSVRender::Cell::Cell (CSMWorld::Data& data, Ogre::SceneManager *sceneManager, const std::string& id, const Ogre::Vector3& origin) -: mData (data), mId (Misc::StringUtils::lowerCase (id)) +: mData (data), mId (Misc::StringUtils::lowerCase (id)), mSceneMgr(sceneManager) { mCellNode = sceneManager->getRootSceneNode()->createChildSceneNode(); mCellNode->setPosition (origin); @@ -88,7 +88,7 @@ CSVRender::Cell::Cell (CSMWorld::Data& data, Ogre::SceneManager *sceneManager, { float verts = ESM::Land::LAND_SIZE; float worldsize = ESM::Land::REAL_SIZE; - CSVWorld::PhysicsSystem::instance()->addHeightField( + CSVWorld::PhysicsSystem::instance()->addHeightField(sceneManager, esmLand->mLandData->mHeights, esmLand->mX, esmLand->mY, 0, worldsize / (verts-1), verts); } } @@ -105,7 +105,7 @@ CSVRender::Cell::~Cell() { const ESM::Land* esmLand = land.getRecord(mId).get().mLand.get(); if(esmLand) - CSVWorld::PhysicsSystem::instance()->removeHeightField(esmLand->mX, esmLand->mY); + CSVWorld::PhysicsSystem::instance()->removeHeightField(mSceneMgr, esmLand->mX, esmLand->mY); } } diff --git a/apps/opencs/view/render/cell.hpp b/apps/opencs/view/render/cell.hpp index 98056c3549..2badc84d2f 100644 --- a/apps/opencs/view/render/cell.hpp +++ b/apps/opencs/view/render/cell.hpp @@ -32,6 +32,7 @@ namespace CSVRender Ogre::SceneNode *mCellNode; std::map mObjects; std::auto_ptr mTerrain; + Ogre::SceneManager *mSceneMgr; /// Ignored if cell does not have an object with the given ID. /// diff --git a/apps/opencs/view/world/physicssystem.cpp b/apps/opencs/view/world/physicssystem.cpp index e1acdbbfe7..a8cf5c37b6 100644 --- a/apps/opencs/view/world/physicssystem.cpp +++ b/apps/opencs/view/world/physicssystem.cpp @@ -129,15 +129,35 @@ namespace CSVWorld position, rotation); } - void PhysicsSystem::addHeightField(float* heights, - int x, int y, float yoffset, float triSize, float sqrtVerts) + void PhysicsSystem::addHeightField(Ogre::SceneManager *sceneManager, + float* heights, int x, int y, float yoffset, float triSize, float sqrtVerts) { - mEngine->addHeightField(heights, x, y, yoffset, triSize, sqrtVerts); + std::string name = "HeightField_" + + QString::number(x).toStdString() + "_" + QString::number(y).toStdString(); + + if(mTerrain.find(name) == mTerrain.end()) + mEngine->addHeightField(heights, x, y, yoffset, triSize, sqrtVerts); + + mTerrain.insert(std::pair(name, sceneManager)); } - void PhysicsSystem::removeHeightField(int x, int y) + void PhysicsSystem::removeHeightField(Ogre::SceneManager *sceneManager, int x, int y) { - mEngine->removeHeightField(x, y); + std::string name = "HeightField_" + + QString::number(x).toStdString() + "_" + QString::number(y).toStdString(); + + if(mTerrain.count(name) == 1) + mEngine->removeHeightField(x, y); + + std::multimap::iterator iter = mTerrain.begin(); + for(; iter != mTerrain.end(); ++iter) + { + if((*iter).second == sceneManager) + { + mTerrain.erase(iter); + break; + } + } } // sceneMgr: to lookup the scene node name from the object's referenceId diff --git a/apps/opencs/view/world/physicssystem.hpp b/apps/opencs/view/world/physicssystem.hpp index f746ef531a..c2cb567be5 100644 --- a/apps/opencs/view/world/physicssystem.hpp +++ b/apps/opencs/view/world/physicssystem.hpp @@ -37,6 +37,7 @@ namespace CSVWorld std::list mSceneManagers; // FIXME: change to list per OEngine std::list mSceneWidgets; // FIXME: change to list per OEngine OEngine::Physic::PhysicEngine* mEngine; + std::multimap mTerrain; public: @@ -57,10 +58,10 @@ namespace CSVWorld void moveObject(const std::string &sceneNodeName, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation); - void addHeightField(float* heights, - int x, int y, float yoffset, float triSize, float sqrtVerts); + void addHeightField(Ogre::SceneManager *sceneManager, + float* heights, int x, int y, float yoffset, float triSize, float sqrtVerts); - void removeHeightField(int x, int y); + void removeHeightField(Ogre::SceneManager *sceneManager, int x, int y); void toggleDebugRendering(Ogre::SceneManager *sceneMgr);