openmw-tes3coop/apps/openmw/mwrender/mwscene.cpp

223 lines
7.4 KiB
C++
Raw Normal View History

#include "mwscene.hpp"
#include <assert.h>
#include "OgreRoot.h"
#include "OgreRenderWindow.h"
#include "OgreSceneManager.h"
#include "OgreViewport.h"
#include "OgreCamera.h"
#include "OgreTextureManager.h"
#include "../mwworld/world.hpp" // these includes can be removed once the static-hack is gone
#include "../mwworld/ptr.hpp"
#include "../mwworld/doingphysics.hpp"
#include <components/esm/loadstat.hpp>
#include "player.hpp"
using namespace MWRender;
using namespace Ogre;
MWScene::MWScene(OEngine::Render::OgreRenderer &_rend , OEngine::Physic::PhysicEngine* physEng)
2011-02-22 13:11:53 +00:00
: rend(_rend)
{
2011-02-22 13:11:53 +00:00
eng = physEng;
rend.createScene("PlayerCam", 55, 5);
2011-02-22 13:11:53 +00:00
// Set default mipmap level (NB some APIs ignore this)
TextureManager::getSingleton().setDefaultNumMipmaps(5);
2011-02-22 13:11:53 +00:00
// Load resources
ResourceGroupManager::getSingleton().initialiseAllResourceGroups();
2011-02-22 13:11:53 +00:00
// Turn the entire scene (represented by the 'root' node) -90
// degrees around the x axis. This makes Z go upwards, and Y go into
// the screen (when x is to the right.) This is the orientation that
// Morrowind uses, and it automagically makes everything work as it
// should.
SceneNode *rt = rend.getScene()->getRootSceneNode();
mwRoot = rt->createChildSceneNode();
mwRoot->pitch(Degree(-90));
2010-08-03 18:17:31 +00:00
2011-02-22 13:11:53 +00:00
//used to obtain ingame information of ogre objects (which are faced or selected)
mRaySceneQuery = rend.getScene()->createRayQuery(Ray());
2011-02-22 13:11:53 +00:00
Ogre::SceneNode *playerNode = mwRoot->createChildSceneNode();
playerNode->pitch(Degree(90));
Ogre::SceneNode *cameraYawNode = playerNode->createChildSceneNode();
Ogre::SceneNode *cameraPitchNode = cameraYawNode->createChildSceneNode();
cameraPitchNode->attachObject(getCamera());
2011-02-10 11:56:19 +00:00
2011-02-22 13:11:53 +00:00
mPlayer = new MWRender::Player (getCamera(), playerNode->getName());
}
MWScene::~MWScene()
{
delete mPlayer;
}
2010-08-03 18:17:31 +00:00
std::pair<std::string, float> MWScene::getFacedHandle (MWWorld::World& world)
2010-08-03 18:17:31 +00:00
{
std::string handle = "";
float distance = -1;
2010-08-03 18:17:31 +00:00
//get a ray pointing to the center of the viewport
Ray centerRay = getCamera()->getCameraToViewportRay(
getViewport()->getWidth()/2,
getViewport()->getHeight()/2);
2011-03-17 09:54:38 +00:00
//let's avoid the capsule shape of the player.
centerRay.setOrigin(centerRay.getOrigin() + 20*centerRay.getDirection());
btVector3 from(centerRay.getOrigin().x,centerRay.getOrigin().y,centerRay.getOrigin().z);
btVector3 to(centerRay.getPoint(1000).x,centerRay.getPoint(1000).x,centerRay.getPoint(1000).x);
2010-08-03 18:17:31 +00:00
// get all objects touched by the ray
/*getRaySceneQuery()->setRay (centerRay );
2010-08-03 18:17:31 +00:00
RaySceneQueryResult &result = getRaySceneQuery()->execute();
RaySceneQueryResult::iterator nearest = result.end();
for (RaySceneQueryResult::iterator itr = result.begin();
itr != result.end(); itr++ )
{
// there seem to be omnipresent objects like the caelum sky dom,
// the distance of these objects is always 0 so this if excludes these
if ( itr->movable && itr->distance >= 0.1)
{
// horrible hack to exclude statics. this must be removed as soon as a replacement for the
// AABB raycasting is implemented (we should not ignore statics)
MWWorld::Ptr ptr = world.getPtrViaHandle (itr->movable->getParentSceneNode()->getName());
if (ptr.getType()==typeid (ESM::Static))
break;
2010-08-03 18:17:31 +00:00
if ( nearest == result.end() ) //if no object is set
{
nearest = itr;
}
else if ( itr->distance < nearest->distance )
{
nearest = itr;
}
}
}
if ( nearest != result.end() )
{
handle = nearest->movable->getParentSceneNode()->getName();
distance = nearest->distance;
}*/
return eng->rayTest(from,to);
2010-08-03 18:17:31 +00:00
}
2011-01-23 14:28:42 +00:00
void MWScene::doPhysics (float duration, MWWorld::World& world,
const std::vector<std::pair<std::string, Ogre::Vector3> >& actors)
2011-01-23 14:28:42 +00:00
{
// stop changes to world from being reported back to the physics system
MWWorld::DoingPhysics scopeGuard;
2011-01-23 14:28:42 +00:00
2011-02-22 13:11:53 +00:00
//set the DebugRenderingMode. To disable it,set it to 0
eng->setDebugRenderingMode(1);
//set the walkdirection to 0 (no movement) for every actor)
for(std::map<std::string,OEngine::Physic::PhysicActor*>::iterator it = eng->PhysicActorMap.begin(); it != eng->PhysicActorMap.end();it++)
{
OEngine::Physic::PhysicActor* act = it->second;
act->setWalkDirection(btVector3(0,0,0));
}
for (std::vector<std::pair<std::string, Ogre::Vector3> >::const_iterator iter (actors.begin());
iter!=actors.end(); ++iter)
{
2011-02-19 10:36:57 +00:00
OEngine::Physic::PhysicActor* act = eng->getCharacter(iter->first);
//dirty stuff to get the camera orientation. Must be changed!
2011-02-23 11:28:13 +00:00
Ogre::SceneNode *sceneNode = rend.getScene()->getSceneNode (iter->first);
2011-02-22 13:11:53 +00:00
Ogre::Quaternion quat = sceneNode->getChildIterator().getNext()->getOrientation();
2011-02-23 11:28:13 +00:00
Ogre::Vector3 dir1(iter->second.x,iter->second.z,-iter->second.y);
2011-03-09 17:29:57 +00:00
Ogre::Vector3 dir = 0.01*(quat*dir1);
//set the walk direction
2011-03-09 17:29:57 +00:00
act->setWalkDirection(btVector3(dir.x,-dir.z,dir.y));
2011-02-19 10:36:57 +00:00
}
2011-02-22 13:11:53 +00:00
eng->stepSimulation(duration);
for(std::map<std::string,OEngine::Physic::PhysicActor*>::iterator it = eng->PhysicActorMap.begin(); it != eng->PhysicActorMap.end();it++)
2011-02-19 10:36:57 +00:00
{
OEngine::Physic::PhysicActor* act = it->second;
2011-02-22 13:11:53 +00:00
btVector3 newPos = act->getPosition();
MWWorld::Ptr ptr = world.getPtrViaHandle (it->first);
2011-02-19 10:36:57 +00:00
world.moveObject (ptr, newPos.x(), newPos.y(), newPos.z());
}
2011-01-23 14:28:42 +00:00
}
void MWScene::addObject (const std::string& handle, const std::string& mesh,
const Ogre::Quaternion& rotation, float scale, const Ogre::Vector3& position)
{
2011-02-19 10:36:57 +00:00
OEngine::Physic::RigidBody* body = eng->createRigidBody(mesh,handle);
eng->addRigidBody(body);
btTransform tr;
tr.setOrigin(btVector3(position.x,position.y,position.z));
tr.setRotation(btQuaternion(rotation.x,rotation.y,rotation.z,rotation.w));
body->setWorldTransform(tr);
}
void MWScene::addActor (const std::string& handle, const std::string& mesh,
const Ogre::Vector3& position)
{
2011-02-19 10:36:57 +00:00
//TODO:optimize this. Searching the std::map isn't very efficient i think.
eng->addCharacter(handle);
OEngine::Physic::PhysicActor* act = eng->getCharacter(handle);
act->setPosition(btVector3(position.x,position.y,position.z));
}
void MWScene::removeObject (const std::string& handle)
{
2011-02-19 10:36:57 +00:00
//TODO:check if actor???
eng->removeRigidBody(handle);
eng->deleteRigidBody(handle);
}
void MWScene::moveObject (const std::string& handle, const Ogre::Vector3& position, bool updatePhysics)
{
2011-02-22 13:11:53 +00:00
rend.getScene()->getSceneNode(handle)->setPosition(position);
2011-02-19 10:36:57 +00:00
if(updatePhysics)//TODO: is it an actor?
{
OEngine::Physic::RigidBody* body = eng->getRigidBody(handle);
btTransform tr = body->getWorldTransform();
tr.setOrigin(btVector3(position.x,position.y,position.z));
body->setWorldTransform(tr);
}
}
void MWScene::rotateObject (const std::string& handle, const Ogre::Quaternion& rotation)
{
}
void MWScene::scaleObject (const std::string& handle, float scale)
{
}
2011-01-25 17:12:13 +00:00
void MWScene::toggleCollisionMode()
2011-01-25 17:12:13 +00:00
{
for(std::map<std::string,OEngine::Physic::PhysicActor*>::iterator it = eng->PhysicActorMap.begin(); it != eng->PhysicActorMap.end();it++)
{
OEngine::Physic::PhysicActor* act = it->second;
bool cmode = act->getCollisionMode();
if(cmode)
{
act->enableCollisions(false);
act->setGravity(0);
}
else
{
act->enableCollisions(true);
act->setGravity(10);
}
}
2011-01-25 17:12:13 +00:00
}