@ -170,10 +170,10 @@ namespace MWWorld
World : : World ( OEngine : : Render : : OgreRenderer & renderer ,
const Files : : Collections & fileCollections ,
const std : : string & master , const boost : : filesystem : : path & resDir , const boost : : filesystem : : path & cacheDir , bool newGame ,
ToUTF8 : : Utf8Encoder * encoder , std : : map < std : : string , std : : string > fallbackMap )
ToUTF8 : : Utf8Encoder * encoder , std : : map < std : : string , std : : string > fallbackMap , int mActivationDistanceOverride )
: mPlayer ( 0 ) , mLocalScripts ( mStore ) , mGlobalVariables ( 0 ) ,
mSky ( true ) , mCells ( mStore , mEsm ) ,
mNumFacing ( 0 )
mNumFacing ( 0 ) , mActivationDistanceOverride ( mActivationDistanceOverride )
{
mPhysics = new PhysicsSystem ( renderer ) ;
mPhysEngine = mPhysics - > getEngine ( ) ;
@ -573,23 +573,55 @@ namespace MWWorld
return mWorldScene - > markCellAsUnchanged ( ) ;
}
std : : string World : : getFacedHandle ( )
float World : : getMaxActivationDistance ( )
{
if ( ! mRendering - > occlusionQuerySupported ( ) )
{
std : : pair < std : : string , float > result = mPhysics - > getFacedHandle ( * this ) ;
if ( mActivationDistanceOverride > = 0 )
return mActivationDistanceOverride ;
if ( result . first . empty ( ) | |
result . second > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " iMaxActivateDist " ) - > getInt ( ) )
return " " ;
return ( std : : max ) ( getNpcActivationDistance ( ) , getObjectActivationDistance ( ) ) ;
}
return result . first ;
float World : : getNpcActivationDistance ( )
{
if ( mActivationDistanceOverride > = 0 )
return mActivationDistanceOverride ;
return getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " iMaxActivateDist " ) - > getInt ( ) * 5 / 4 ;
}
else
float World : : getObjectActivationDistance ( )
{
// updated every few frames in update()
return mFacedHandle ;
if ( mActivationDistanceOverride > = 0 )
return mActivationDistanceOverride ;
return getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " iMaxActivateDist " ) - > getInt ( ) ;
}
MWWorld : : Ptr World : : getFacedObject ( )
{
std : : pair < float , std : : string > result ;
if ( ! mRendering - > occlusionQuerySupported ( ) )
result = mPhysics - > getFacedHandle ( * this , getMaxActivationDistance ( ) ) ;
else
result = std : : make_pair ( mFacedDistance , mFacedHandle ) ;
if ( result . second . empty ( ) )
return MWWorld : : Ptr ( ) ;
MWWorld : : Ptr object = searchPtrViaHandle ( result . second ) ;
float ActivationDistance ;
if ( object . getTypeName ( ) . find ( " NPC " ) ! = std : : string : : npos )
ActivationDistance = getNpcActivationDistance ( ) ;
else
ActivationDistance = getObjectActivationDistance ( ) ;
if ( result . first > ActivationDistance )
return MWWorld : : Ptr ( ) ;
return object ;
}
void World : : deleteObject ( const Ptr & ptr )
@ -884,8 +916,6 @@ namespace MWWorld
void World : : update ( float duration , bool paused )
{
/// \todo split this function up into subfunctions
mWorldScene - > update ( duration , paused ) ;
float pitch , yaw ;
@ -895,8 +925,15 @@ namespace MWWorld
mWeatherManager - > update ( duration ) ;
performUpdateSceneQueries ( ) ;
updateWindowManager ( ) ;
}
void World : : updateWindowManager ( )
{
// inform the GUI about focused object
MWWorld : : Ptr object = searchPtrViaHandle ( mFacedHandle ) ;
MWWorld : : Ptr object = getFacedObject ( ) ;
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > setFocusObject ( object ) ;
@ -918,7 +955,10 @@ namespace MWWorld
screenCoords [ 0 ] , screenCoords [ 1 ] , screenCoords [ 2 ] , screenCoords [ 3 ] ) ;
}
}
}
void World : : performUpdateSceneQueries ( )
{
if ( ! mRendering - > occlusionQuerySupported ( ) )
{
// cast a ray from player to sun to detect if the sun is visible
@ -936,20 +976,37 @@ namespace MWWorld
{
MWRender : : OcclusionQuery * query = mRendering - > getOcclusionQuery ( ) ;
if ( ! query - > occlusionTestPending ( ) )
{
processFacedQueryResults ( query ) ;
beginFacedQueryProcess ( query ) ;
}
}
}
void World : : processFacedQueryResults ( MWRender : : OcclusionQuery * query )
{
// get result of last query
if ( mNumFacing = = 0 ) mFacedHandle = " " ;
if ( mNumFacing = = 0 )
{
mFacedHandle = " " ;
mFacedDistance = FLT_MAX ;
}
else if ( mNumFacing = = 1 )
{
bool result = query - > getTestResult ( ) ;
mFacedHandle = result ? mFaced1Name : " " ;
mFacedDistance = result ? mFaced1Distance : FLT_MAX ;
}
else if ( mNumFacing = = 2 )
{
bool result = query - > getTestResult ( ) ;
mFacedHandle = result ? mFaced2Name : mFaced1Name ;
mFacedDistance = result ? mFaced1Distance : mFaced1Distance ;
}
}
void World : : beginFacedQueryProcess ( MWRender : : OcclusionQuery * query )
{
// send new query
// figure out which object we want to test against
std : : vector < std : : pair < float , std : : string > > results ;
@ -957,10 +1014,12 @@ namespace MWWorld
{
float x , y ;
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > getMousePosition ( x , y ) ;
results = mPhysics - > getFaced Objects( x , y ) ;
results = mPhysics - > getFaced Handles( x , y , getMaxActivationDistance ( ) ) ;
}
else
results = mPhysics - > getFacedObjects ( ) ;
{
results = mPhysics - > getFacedHandles ( getMaxActivationDistance ( ) ) ;
}
// ignore the player and other things we're not interested in
std : : vector < std : : pair < float , std : : string > > : : iterator it = results . begin ( ) ;
@ -980,9 +1039,20 @@ namespace MWWorld
mNumFacing = 0 ;
}
else if ( results . size ( ) = = 1 )
{
beginSingleFacedQueryProcess ( query , results ) ;
}
else
{
beginDoubleFacedQueryProcess ( query , results ) ;
}
}
void World : : beginSingleFacedQueryProcess ( MWRender : : OcclusionQuery * query , std : : vector < std : : pair < float , std : : string > > const & results )
{
mFaced1 = getPtrViaHandle ( results . front ( ) . second ) ;
mFaced1Name = results . front ( ) . second ;
mFaced1Distance = results . front ( ) . first ;
mNumFacing = 1 ;
btVector3 p ;
@ -1002,12 +1072,15 @@ namespace MWWorld
query - > occlusionTest ( pos , node ) ;
}
else
void World : : beginDoubleFacedQueryProcess ( MWRender : : OcclusionQuery * query , std : : vector < std : : pair < float , std : : string > > const & results )
{
mFaced1Name = results . front ( ) . second ;
mFaced2Name = results [ 1 ] . second ;
mFaced1 = getPtrViaHandle ( results . front ( ) . second ) ;
mFaced2 = getPtrViaHandle ( results [ 1 ] . second ) ;
mFaced1Name = results . at ( 0 ) . second ;
mFaced2Name = results . at ( 1 ) . second ;
mFaced1Distance = results . at ( 0 ) . first ;
mFaced2Distance = results . at ( 1 ) . first ;
mFaced1 = getPtrViaHandle ( results . at ( 0 ) . second ) ;
mFaced2 = getPtrViaHandle ( results . at ( 1 ) . second ) ;
mNumFacing = 2 ;
btVector3 p ;
@ -1015,10 +1088,10 @@ namespace MWWorld
{
float x , y ;
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > getMousePosition ( x , y ) ;
p = mPhysics - > getRayPoint ( results [1 ] . first , x , y ) ;
p = mPhysics - > getRayPoint ( results .at ( 1 ) . first , x , y ) ;
}
else
p = mPhysics - > getRayPoint ( results [1 ] . first ) ;
p = mPhysics - > getRayPoint ( results .at ( 1 ) . first ) ;
Ogre : : Vector3 pos ( p . x ( ) , p . z ( ) , - p . y ( ) ) ;
Ogre : : SceneNode * node1 = mFaced1 . getRefData ( ) . getBaseNode ( ) ;
Ogre : : SceneNode * node2 = mFaced2 . getRefData ( ) . getBaseNode ( ) ;
@ -1027,6 +1100,7 @@ namespace MWWorld
if ( ! query - > isPotentialOccluder ( node1 ) & & ( mFaced1 . getTypeName ( ) . find ( " Static " ) = = std : : string : : npos ) )
{
mFacedHandle = mFaced1Name ;
mFacedDistance = mFaced1Distance ;
//std::cout << "node1 Not an occluder" << std::endl;
return ;
}
@ -1035,6 +1109,7 @@ namespace MWWorld
if ( mFaced2 . getTypeName ( ) . find ( " Static " ) ! = std : : string : : npos )
{
mFacedHandle = mFaced1Name ;
mFacedDistance = mFaced1Distance ;
return ;
}
@ -1043,14 +1118,12 @@ namespace MWWorld
& & mFaced2 . getTypeName ( ) . find ( " Door " ) ! = std : : string : : npos )
{
mFacedHandle = mFaced2Name ;
mFacedDistance = mFaced2Distance ;
return ;
}
query - > occlusionTest ( pos , node2 ) ;
}
}
}
}
bool World : : isCellExterior ( ) const
{