2011-10-25 18:06:44 +00:00
|
|
|
#include "debugging.hpp"
|
2010-06-05 18:37:01 +00:00
|
|
|
|
2012-07-03 10:30:50 +00:00
|
|
|
#include <cassert>
|
2010-06-05 18:37:01 +00:00
|
|
|
|
2012-03-11 22:12:56 +00:00
|
|
|
#include <OgreNode.h>
|
|
|
|
#include <OgreSceneManager.h>
|
2012-03-14 11:03:04 +00:00
|
|
|
#include <OgreMaterial.h>
|
|
|
|
#include <OgreMaterialManager.h>
|
2012-07-03 13:32:38 +00:00
|
|
|
#include <OgreManualObject.h>
|
2014-02-19 17:40:29 +00:00
|
|
|
#include <OgreTechnique.h>
|
|
|
|
#include <OgreSceneNode.h>
|
2010-06-06 11:36:45 +00:00
|
|
|
|
2013-02-07 05:47:09 +00:00
|
|
|
#include <openengine/bullet/physic.hpp>
|
|
|
|
|
2010-08-25 07:19:15 +00:00
|
|
|
#include <components/esm/loadstat.hpp>
|
2012-03-08 06:46:34 +00:00
|
|
|
#include <components/esm/loadpgrd.hpp>
|
2010-08-25 07:19:15 +00:00
|
|
|
|
2012-07-03 10:30:50 +00:00
|
|
|
#include "../mwbase/world.hpp" // these includes can be removed once the static-hack is gone
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
|
|
|
|
#include "../mwworld/ptr.hpp"
|
2014-02-23 19:11:05 +00:00
|
|
|
#include "../mwworld/cellstore.hpp"
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
2015-03-08 05:11:54 +00:00
|
|
|
#include "../mwmechanics/pathfinding.hpp"
|
2012-07-03 10:30:50 +00:00
|
|
|
|
2012-09-28 15:02:18 +00:00
|
|
|
#include "renderconst.hpp"
|
2011-01-08 14:11:37 +00:00
|
|
|
|
2010-06-05 18:37:01 +00:00
|
|
|
using namespace Ogre;
|
|
|
|
|
2012-03-13 23:06:56 +00:00
|
|
|
namespace MWRender
|
|
|
|
{
|
|
|
|
|
2012-03-14 11:03:04 +00:00
|
|
|
static const std::string PATHGRID_POINT_MATERIAL = "pathgridPointMaterial";
|
2012-03-13 23:06:56 +00:00
|
|
|
static const std::string PATHGRID_LINE_MATERIAL = "pathgridLineMaterial";
|
|
|
|
static const std::string DEBUGGING_GROUP = "debugging";
|
2012-04-01 13:51:37 +00:00
|
|
|
static const int POINT_MESH_BASE = 35;
|
2012-03-13 23:06:56 +00:00
|
|
|
|
2012-03-14 11:03:04 +00:00
|
|
|
void Debugging::createGridMaterials()
|
2012-03-13 23:06:56 +00:00
|
|
|
{
|
2012-03-14 11:03:04 +00:00
|
|
|
if (mGridMatsCreated) return;
|
|
|
|
|
2012-03-13 23:06:56 +00:00
|
|
|
if (MaterialManager::getSingleton().getByName(PATHGRID_LINE_MATERIAL, DEBUGGING_GROUP).isNull())
|
|
|
|
{
|
|
|
|
MaterialPtr lineMatPtr = MaterialManager::getSingleton().create(PATHGRID_LINE_MATERIAL, DEBUGGING_GROUP);
|
|
|
|
lineMatPtr->setReceiveShadows(false);
|
|
|
|
lineMatPtr->getTechnique(0)->setLightingEnabled(true);
|
|
|
|
lineMatPtr->getTechnique(0)->getPass(0)->setDiffuse(1,1,0,0);
|
|
|
|
lineMatPtr->getTechnique(0)->getPass(0)->setAmbient(1,1,0);
|
|
|
|
lineMatPtr->getTechnique(0)->getPass(0)->setSelfIllumination(1,1,0);
|
|
|
|
}
|
|
|
|
|
2012-03-14 11:03:04 +00:00
|
|
|
if (MaterialManager::getSingleton().getByName(PATHGRID_POINT_MATERIAL, DEBUGGING_GROUP).isNull())
|
|
|
|
{
|
|
|
|
MaterialPtr pointMatPtr = MaterialManager::getSingleton().create(PATHGRID_POINT_MATERIAL, DEBUGGING_GROUP);
|
|
|
|
pointMatPtr->setReceiveShadows(false);
|
|
|
|
pointMatPtr->getTechnique(0)->setLightingEnabled(true);
|
|
|
|
pointMatPtr->getTechnique(0)->getPass(0)->setDiffuse(1,0,0,0);
|
|
|
|
pointMatPtr->getTechnique(0)->getPass(0)->setAmbient(1,0,0);
|
|
|
|
pointMatPtr->getTechnique(0)->getPass(0)->setSelfIllumination(1,0,0);
|
|
|
|
}
|
|
|
|
mGridMatsCreated = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Debugging::destroyGridMaterials()
|
|
|
|
{
|
|
|
|
if (mGridMatsCreated)
|
|
|
|
{
|
|
|
|
MaterialManager::getSingleton().remove(PATHGRID_POINT_MATERIAL);
|
|
|
|
MaterialManager::getSingleton().remove(PATHGRID_LINE_MATERIAL);
|
|
|
|
mGridMatsCreated = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-01 13:27:18 +00:00
|
|
|
ManualObject *Debugging::createPathgridLines(const ESM::Pathgrid *pathgrid)
|
2012-03-14 11:03:04 +00:00
|
|
|
{
|
2012-04-01 13:27:18 +00:00
|
|
|
ManualObject *result = mSceneMgr->createManualObject();
|
2012-03-13 23:06:56 +00:00
|
|
|
|
2012-04-01 13:27:18 +00:00
|
|
|
result->begin(PATHGRID_LINE_MATERIAL, RenderOperation::OT_LINE_LIST);
|
2012-09-17 07:37:50 +00:00
|
|
|
for(ESM::Pathgrid::EdgeList::const_iterator it = pathgrid->mEdges.begin();
|
|
|
|
it != pathgrid->mEdges.end();
|
2012-06-06 18:29:30 +00:00
|
|
|
++it)
|
2012-04-01 13:27:18 +00:00
|
|
|
{
|
|
|
|
const ESM::Pathgrid::Edge &edge = *it;
|
2012-09-17 07:37:50 +00:00
|
|
|
const ESM::Pathgrid::Point &p1 = pathgrid->mPoints[edge.mV0], &p2 = pathgrid->mPoints[edge.mV1];
|
2015-03-08 05:11:54 +00:00
|
|
|
Vector3 direction = (MWMechanics::PathFinder::MakeOgreVector3(p2) - MWMechanics::PathFinder::MakeOgreVector3(p1));
|
2012-04-01 13:27:18 +00:00
|
|
|
Vector3 lineDisplacement = direction.crossProduct(Vector3::UNIT_Z).normalisedCopy();
|
2012-04-01 13:51:37 +00:00
|
|
|
lineDisplacement = lineDisplacement * POINT_MESH_BASE +
|
|
|
|
Vector3(0, 0, 10); // move lines up a little, so they will be less covered by meshes/landscape
|
2015-03-08 05:11:54 +00:00
|
|
|
result->position(MWMechanics::PathFinder::MakeOgreVector3(p1) + lineDisplacement);
|
|
|
|
result->position(MWMechanics::PathFinder::MakeOgreVector3(p2) + lineDisplacement);
|
2012-04-01 13:27:18 +00:00
|
|
|
}
|
|
|
|
result->end();
|
|
|
|
|
2012-09-28 15:02:18 +00:00
|
|
|
result->setVisibilityFlags (RV_Debug);
|
|
|
|
|
2012-04-01 13:27:18 +00:00
|
|
|
return result;
|
2012-03-13 23:06:56 +00:00
|
|
|
}
|
|
|
|
|
2012-04-01 13:27:18 +00:00
|
|
|
ManualObject *Debugging::createPathgridPoints(const ESM::Pathgrid *pathgrid)
|
2012-03-13 23:06:56 +00:00
|
|
|
{
|
2012-04-01 13:27:18 +00:00
|
|
|
ManualObject *result = mSceneMgr->createManualObject();
|
2012-04-01 13:51:37 +00:00
|
|
|
const float height = POINT_MESH_BASE * sqrtf(2);
|
2012-04-01 13:27:18 +00:00
|
|
|
|
|
|
|
result->begin(PATHGRID_POINT_MATERIAL, RenderOperation::OT_TRIANGLE_STRIP);
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
uint32 startIndex = 0;
|
2012-09-17 07:37:50 +00:00
|
|
|
for(ESM::Pathgrid::PointList::const_iterator it = pathgrid->mPoints.begin();
|
|
|
|
it != pathgrid->mPoints.end();
|
2014-04-27 17:03:33 +00:00
|
|
|
++it, startIndex += 6)
|
2012-04-01 13:27:18 +00:00
|
|
|
{
|
2015-03-08 05:11:54 +00:00
|
|
|
Vector3 pointPos(MWMechanics::PathFinder::MakeOgreVector3(*it));
|
2012-04-01 13:27:18 +00:00
|
|
|
|
|
|
|
if (!first)
|
|
|
|
{
|
|
|
|
// degenerate triangle from previous octahedron
|
|
|
|
result->index(startIndex - 4); // 2nd point of previous octahedron
|
|
|
|
result->index(startIndex); // start point of current octahedron
|
|
|
|
}
|
|
|
|
|
2015-03-08 05:11:54 +00:00
|
|
|
Ogre::Real pointMeshBase = static_cast<Ogre::Real>(POINT_MESH_BASE);
|
|
|
|
|
2012-04-01 13:27:18 +00:00
|
|
|
result->position(pointPos + Vector3(0, 0, height)); // 0
|
2015-03-08 05:11:54 +00:00
|
|
|
result->position(pointPos + Vector3(-pointMeshBase, -pointMeshBase, 0)); // 1
|
|
|
|
result->position(pointPos + Vector3(pointMeshBase, -pointMeshBase, 0)); // 2
|
|
|
|
result->position(pointPos + Vector3(pointMeshBase, pointMeshBase, 0)); // 3
|
|
|
|
result->position(pointPos + Vector3(-pointMeshBase, pointMeshBase, 0)); // 4
|
2012-04-01 13:27:18 +00:00
|
|
|
result->position(pointPos + Vector3(0, 0, -height)); // 5
|
|
|
|
|
|
|
|
result->index(startIndex + 0);
|
|
|
|
result->index(startIndex + 1);
|
|
|
|
result->index(startIndex + 2);
|
|
|
|
result->index(startIndex + 5);
|
|
|
|
result->index(startIndex + 3);
|
|
|
|
result->index(startIndex + 4);
|
|
|
|
// degenerates
|
|
|
|
result->index(startIndex + 4);
|
|
|
|
result->index(startIndex + 5);
|
|
|
|
result->index(startIndex + 5);
|
|
|
|
// end degenerates
|
|
|
|
result->index(startIndex + 1);
|
|
|
|
result->index(startIndex + 4);
|
|
|
|
result->index(startIndex + 0);
|
|
|
|
result->index(startIndex + 3);
|
|
|
|
result->index(startIndex + 2);
|
|
|
|
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
result->end();
|
|
|
|
|
2012-09-28 15:02:18 +00:00
|
|
|
result->setVisibilityFlags (RV_Debug);
|
|
|
|
|
2012-04-01 13:27:18 +00:00
|
|
|
return result;
|
2012-03-13 23:06:56 +00:00
|
|
|
}
|
|
|
|
|
2013-02-26 13:01:10 +00:00
|
|
|
Debugging::Debugging(SceneNode *root, OEngine::Physic::PhysicEngine *engine) :
|
|
|
|
mRootNode(root), mEngine(engine),
|
|
|
|
mSceneMgr(root->getCreator()),
|
2012-03-14 11:03:04 +00:00
|
|
|
mPathgridEnabled(false),
|
|
|
|
mInteriorPathgridNode(NULL), mPathGridRoot(NULL),
|
|
|
|
mGridMatsCreated(false)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2012-03-13 23:06:56 +00:00
|
|
|
ResourceGroupManager::getSingleton().createResourceGroup(DEBUGGING_GROUP);
|
|
|
|
}
|
|
|
|
|
|
|
|
Debugging::~Debugging()
|
|
|
|
{
|
2012-03-14 11:03:04 +00:00
|
|
|
if (mPathgridEnabled)
|
|
|
|
{
|
|
|
|
togglePathgrid();
|
|
|
|
}
|
|
|
|
|
2012-03-13 23:06:56 +00:00
|
|
|
ResourceGroupManager::getSingleton().destroyResourceGroup(DEBUGGING_GROUP);
|
2011-10-22 04:15:15 +00:00
|
|
|
}
|
|
|
|
|
2011-10-24 17:42:36 +00:00
|
|
|
|
2011-10-22 04:15:15 +00:00
|
|
|
bool Debugging::toggleRenderMode (int mode){
|
2012-03-08 06:46:34 +00:00
|
|
|
switch (mode)
|
2011-10-22 04:15:15 +00:00
|
|
|
{
|
2012-07-03 10:30:50 +00:00
|
|
|
case MWBase::World::Render_CollisionDebug:
|
2012-04-02 10:04:47 +00:00
|
|
|
|
2012-03-31 10:50:10 +00:00
|
|
|
return mEngine->toggleDebugRendering();
|
2012-04-02 10:04:47 +00:00
|
|
|
|
2012-07-03 10:30:50 +00:00
|
|
|
case MWBase::World::Render_Pathgrid:
|
2012-03-08 06:46:34 +00:00
|
|
|
togglePathgrid();
|
2012-03-14 11:03:04 +00:00
|
|
|
return mPathgridEnabled;
|
2011-10-22 04:15:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-08 06:46:34 +00:00
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
void Debugging::cellAdded(MWWorld::CellStore *store)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
|
|
|
mActiveCells.push_back(store);
|
2012-03-14 11:03:04 +00:00
|
|
|
if (mPathgridEnabled)
|
2012-03-13 23:06:56 +00:00
|
|
|
enableCellPathgrid(store);
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
void Debugging::cellRemoved(MWWorld::CellStore *store)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
|
|
|
mActiveCells.erase(std::remove(mActiveCells.begin(), mActiveCells.end(), store), mActiveCells.end());
|
2012-03-14 11:03:04 +00:00
|
|
|
if (mPathgridEnabled)
|
2012-03-13 23:06:56 +00:00
|
|
|
disableCellPathgrid(store);
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Debugging::togglePathgrid()
|
|
|
|
{
|
2012-03-14 11:03:04 +00:00
|
|
|
mPathgridEnabled = !mPathgridEnabled;
|
|
|
|
if (mPathgridEnabled)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2012-03-14 11:03:04 +00:00
|
|
|
createGridMaterials();
|
|
|
|
|
2012-03-08 06:46:34 +00:00
|
|
|
// add path grid meshes to already loaded cells
|
2013-02-26 13:01:10 +00:00
|
|
|
mPathGridRoot = mRootNode->createChildSceneNode();
|
2012-06-06 18:29:30 +00:00
|
|
|
for(CellList::iterator it = mActiveCells.begin(); it != mActiveCells.end(); ++it)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2012-03-13 23:06:56 +00:00
|
|
|
enableCellPathgrid(*it);
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
|
|
|
}
|
2012-03-14 11:03:04 +00:00
|
|
|
else
|
|
|
|
{
|
2012-03-08 06:46:34 +00:00
|
|
|
// remove path grid meshes from already loaded cells
|
2012-06-06 18:29:30 +00:00
|
|
|
for(CellList::iterator it = mActiveCells.begin(); it != mActiveCells.end(); ++it)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2012-03-13 23:06:56 +00:00
|
|
|
disableCellPathgrid(*it);
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
|
|
|
mPathGridRoot->removeAndDestroyAllChildren();
|
|
|
|
mSceneMgr->destroySceneNode(mPathGridRoot);
|
2012-03-11 22:12:56 +00:00
|
|
|
mPathGridRoot = NULL;
|
2012-03-14 11:03:04 +00:00
|
|
|
destroyGridMaterials();
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
void Debugging::enableCellPathgrid(MWWorld::CellStore *store)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2014-12-08 23:13:56 +00:00
|
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
2012-11-06 07:53:00 +00:00
|
|
|
const ESM::Pathgrid *pathgrid =
|
2014-12-11 14:19:48 +00:00
|
|
|
world->getStore().get<ESM::Pathgrid>().search(*store->getCell());
|
2012-03-14 11:03:04 +00:00
|
|
|
if (!pathgrid) return;
|
2012-03-08 06:46:34 +00:00
|
|
|
|
2012-04-01 13:27:18 +00:00
|
|
|
Vector3 cellPathGridPos(0, 0, 0);
|
2014-02-21 10:35:46 +00:00
|
|
|
if (store->getCell()->isExterior())
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2015-03-08 05:11:54 +00:00
|
|
|
cellPathGridPos.x = static_cast<Ogre::Real>(store->getCell()->mData.mX * ESM::Land::REAL_SIZE);
|
|
|
|
cellPathGridPos.y = static_cast<Ogre::Real>(store->getCell()->mData.mY * ESM::Land::REAL_SIZE);
|
2012-03-13 23:06:56 +00:00
|
|
|
}
|
|
|
|
SceneNode *cellPathGrid = mPathGridRoot->createChildSceneNode(cellPathGridPos);
|
2012-04-01 13:27:18 +00:00
|
|
|
cellPathGrid->attachObject(createPathgridLines(pathgrid));
|
|
|
|
cellPathGrid->attachObject(createPathgridPoints(pathgrid));
|
2012-03-13 23:06:56 +00:00
|
|
|
|
2014-02-21 10:35:46 +00:00
|
|
|
if (store->getCell()->isExterior())
|
2012-03-13 23:06:56 +00:00
|
|
|
{
|
2014-02-21 10:35:46 +00:00
|
|
|
mExteriorPathgridNodes[std::make_pair(store->getCell()->getGridX(), store->getCell()->getGridY())] = cellPathGrid;
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-03-13 23:06:56 +00:00
|
|
|
assert(mInteriorPathgridNode == NULL);
|
|
|
|
mInteriorPathgridNode = cellPathGrid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
void Debugging::disableCellPathgrid(MWWorld::CellStore *store)
|
2012-03-13 23:06:56 +00:00
|
|
|
{
|
2014-02-21 10:35:46 +00:00
|
|
|
if (store->getCell()->isExterior())
|
2012-03-13 23:06:56 +00:00
|
|
|
{
|
|
|
|
ExteriorPathgridNodes::iterator it =
|
2014-02-21 10:35:46 +00:00
|
|
|
mExteriorPathgridNodes.find(std::make_pair(store->getCell()->getGridX(), store->getCell()->getGridY()));
|
2012-03-13 23:06:56 +00:00
|
|
|
if (it != mExteriorPathgridNodes.end())
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2012-03-13 23:06:56 +00:00
|
|
|
destroyCellPathgridNode(it->second);
|
|
|
|
mExteriorPathgridNodes.erase(it);
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
2012-03-13 23:06:56 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (mInteriorPathgridNode)
|
2012-03-08 06:46:34 +00:00
|
|
|
{
|
2012-03-13 23:06:56 +00:00
|
|
|
destroyCellPathgridNode(mInteriorPathgridNode);
|
|
|
|
mInteriorPathgridNode = NULL;
|
2012-03-08 06:46:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-03-11 22:12:56 +00:00
|
|
|
|
|
|
|
void Debugging::destroyCellPathgridNode(SceneNode *node)
|
|
|
|
{
|
|
|
|
mPathGridRoot->removeChild(node);
|
2012-03-13 23:06:56 +00:00
|
|
|
destroyAttachedObjects(node);
|
2012-03-11 22:12:56 +00:00
|
|
|
mSceneMgr->destroySceneNode(node);
|
|
|
|
}
|
2012-03-13 23:06:56 +00:00
|
|
|
|
|
|
|
void Debugging::destroyAttachedObjects(SceneNode *node)
|
|
|
|
{
|
|
|
|
SceneNode::ObjectIterator objIt = node->getAttachedObjectIterator();
|
|
|
|
while (objIt.hasMoreElements())
|
|
|
|
{
|
|
|
|
MovableObject *mesh = static_cast<MovableObject *>(objIt.getNext());
|
2012-03-14 11:03:04 +00:00
|
|
|
mSceneMgr->destroyMovableObject(mesh);
|
2012-03-13 23:06:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|