@ -35,6 +35,9 @@
# include "aifollow.hpp"
# include "aifollow.hpp"
# include "aipursue.hpp"
# include "aipursue.hpp"
# include "actor.hpp"
# include "summoning.hpp"
namespace
namespace
{
{
@ -71,6 +74,8 @@ bool disintegrateSlot (MWWorld::Ptr ptr, int slot, float disintegrate)
if ( charge = = 0 )
if ( charge = = 0 )
return false ;
return false ;
// FIXME: charge should be a float, not int so that damage < 1 per frame can be applied.
// This was also a bug in the original engine.
charge - =
charge - =
std : : min ( disintegrate ,
std : : min ( disintegrate ,
static_cast < float > ( charge ) ) ;
static_cast < float > ( charge ) ) ;
@ -101,8 +106,8 @@ public:
, mCommanded ( false ) { }
, mCommanded ( false ) { }
virtual void visit ( MWMechanics : : EffectKey key ,
virtual void visit ( MWMechanics : : EffectKey key ,
const std : : string & sourceName , int casterActorId ,
const std : : string & sourceName , const std : : string & sourceId , int casterActorId ,
float magnitude , float remainingTime = - 1 )
float magnitude , float remainingTime = - 1 , float totalTime = - 1 )
{
{
MWWorld : : Ptr player = MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayerPtr ( ) ;
MWWorld : : Ptr player = MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayerPtr ( ) ;
if ( ( ( key . mId = = ESM : : MagicEffect : : CommandHumanoid & & mActor . getClass ( ) . isNpc ( ) )
if ( ( ( key . mId = = ESM : : MagicEffect : : CommandHumanoid & & mActor . getClass ( ) . isNpc ( ) )
@ -161,30 +166,6 @@ void getRestorationPerHourOfSleep (const MWWorld::Ptr& ptr, float& health, float
}
}
}
}
void cleanupSummonedCreature ( MWMechanics : : CreatureStats & casterStats , int creatureActorId )
{
MWWorld : : Ptr ptr = MWBase : : Environment : : get ( ) . getWorld ( ) - > searchPtrViaActorId ( creatureActorId ) ;
if ( ! ptr . isEmpty ( ) )
{
// TODO: Show death animation before deleting? We shouldn't allow looting the corpse while the animation
// plays though, which is a rather lame exploit in vanilla.
MWBase : : Environment : : get ( ) . getWorld ( ) - > deleteObject ( ptr ) ;
const ESM : : Static * fx = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : Static > ( )
. search ( " VFX_Summon_End " ) ;
if ( fx )
MWBase : : Environment : : get ( ) . getWorld ( ) - > spawnEffect ( " meshes \\ " + fx - > mModel ,
" " , Ogre : : Vector3 ( ptr . getRefData ( ) . getPosition ( ) . pos ) ) ;
}
else
{
// We didn't find the creature. It's probably in an inactive cell.
// Add to graveyard so we can delete it when the cell becomes active.
std : : vector < int > & graveyard = casterStats . getSummonedCreatureGraveyard ( ) ;
graveyard . push_back ( creatureActorId ) ;
}
}
}
}
namespace MWMechanics
namespace MWMechanics
@ -199,8 +180,8 @@ namespace MWMechanics
: mCreature ( trappedCreature ) { }
: mCreature ( trappedCreature ) { }
virtual void visit ( MWMechanics : : EffectKey key ,
virtual void visit ( MWMechanics : : EffectKey key ,
const std : : string & sourceName , int casterActorId ,
const std : : string & sourceName , const std : : string & sourceId , int casterActorId ,
float magnitude , float remainingTime = - 1 )
float magnitude , float remainingTime = - 1 , float totalTime = - 1 )
{
{
if ( key . mId ! = ESM : : MagicEffect : : Soultrap )
if ( key . mId ! = ESM : : MagicEffect : : Soultrap )
return ;
return ;
@ -293,6 +274,9 @@ namespace MWMechanics
if ( sqrDist > maxDistance * maxDistance )
if ( sqrDist > maxDistance * maxDistance )
return ;
return ;
if ( targetActor . getClass ( ) . getCreatureStats ( targetActor ) . isDead ( ) )
return ;
// stop tracking when target is behind the actor
// stop tracking when target is behind the actor
Ogre : : Vector3 actorDirection ( actor . getRefData ( ) . getBaseNode ( ) - > getOrientation ( ) . yAxis ( ) ) ;
Ogre : : Vector3 actorDirection ( actor . getRefData ( ) . getBaseNode ( ) - > getOrientation ( ) . yAxis ( ) ) ;
Ogre : : Vector3 targetDirection ( Ogre : : Vector3 ( actor2Pos . pos ) - Ogre : : Vector3 ( actor1Pos . pos ) ) ;
Ogre : : Vector3 targetDirection ( Ogre : : Vector3 ( actor2Pos . pos ) - Ogre : : Vector3 ( actor1Pos . pos ) ) ;
@ -517,6 +501,9 @@ namespace MWMechanics
bool wasDead = creatureStats . isDead ( ) ;
bool wasDead = creatureStats . isDead ( ) ;
// FIXME: effect ticks should go into separate functions so they can be used with either
// magnitude (instant effect) or magnitude*duration
// attributes
// attributes
for ( int i = 0 ; i < ESM : : Attribute : : Length ; + + i )
for ( int i = 0 ; i < ESM : : Attribute : : Length ; + + i )
{
{
@ -726,6 +713,8 @@ namespace MWMechanics
}
}
// Update bound effects
// Update bound effects
// Note: in vanilla MW multiple bound items of the same type can be created by different spells.
// As these extra copies are kinda useless this may or may not be important.
static std : : map < int , std : : string > boundItemsMap ;
static std : : map < int , std : : string > boundItemsMap ;
if ( boundItemsMap . empty ( ) )
if ( boundItemsMap . empty ( ) )
{
{
@ -770,131 +759,11 @@ namespace MWMechanics
}
}
}
}
// Update summon effects
UpdateSummonedCreatures updateSummonedCreatures ( ptr ) ;
static std : : map < int , std : : string > summonMap ;
creatureStats . getActiveSpells ( ) . visitEffectSources ( updateSummonedCreatures ) ;
if ( summonMap . empty ( ) )
if ( ptr . getClass ( ) . hasInventoryStore ( ptr ) )
{
ptr . getClass ( ) . getInventoryStore ( ptr ) . visitEffectSources ( updateSummonedCreatures ) ;
summonMap [ ESM : : MagicEffect : : SummonAncestralGhost ] = " sMagicAncestralGhostID " ;
updateSummonedCreatures . finish ( ) ;
summonMap [ ESM : : MagicEffect : : SummonBonelord ] = " sMagicBonelordID " ;
summonMap [ ESM : : MagicEffect : : SummonBonewalker ] = " sMagicLeastBonewalkerID " ;
summonMap [ ESM : : MagicEffect : : SummonCenturionSphere ] = " sMagicCenturionSphereID " ;
summonMap [ ESM : : MagicEffect : : SummonClannfear ] = " sMagicClannfearID " ;
summonMap [ ESM : : MagicEffect : : SummonDaedroth ] = " sMagicDaedrothID " ;
summonMap [ ESM : : MagicEffect : : SummonDremora ] = " sMagicDremoraID " ;
summonMap [ ESM : : MagicEffect : : SummonFabricant ] = " sMagicFabricantID " ;
summonMap [ ESM : : MagicEffect : : SummonFlameAtronach ] = " sMagicFlameAtronachID " ;
summonMap [ ESM : : MagicEffect : : SummonFrostAtronach ] = " sMagicFrostAtronachID " ;
summonMap [ ESM : : MagicEffect : : SummonGoldenSaint ] = " sMagicGoldenSaintID " ;
summonMap [ ESM : : MagicEffect : : SummonGreaterBonewalker ] = " sMagicGreaterBonewalkerID " ;
summonMap [ ESM : : MagicEffect : : SummonHunger ] = " sMagicHungerID " ;
summonMap [ ESM : : MagicEffect : : SummonScamp ] = " sMagicScampID " ;
summonMap [ ESM : : MagicEffect : : SummonSkeletalMinion ] = " sMagicSkeletalMinionID " ;
summonMap [ ESM : : MagicEffect : : SummonStormAtronach ] = " sMagicStormAtronachID " ;
summonMap [ ESM : : MagicEffect : : SummonWingedTwilight ] = " sMagicWingedTwilightID " ;
summonMap [ ESM : : MagicEffect : : SummonWolf ] = " sMagicCreature01ID " ;
summonMap [ ESM : : MagicEffect : : SummonBear ] = " sMagicCreature02ID " ;
summonMap [ ESM : : MagicEffect : : SummonBonewolf ] = " sMagicCreature03ID " ;
summonMap [ ESM : : MagicEffect : : SummonCreature04 ] = " sMagicCreature04ID " ;
summonMap [ ESM : : MagicEffect : : SummonCreature05 ] = " sMagicCreature05ID " ;
}
std : : map < int , int > & creatureMap = creatureStats . getSummonedCreatureMap ( ) ;
for ( std : : map < int , std : : string > : : iterator it = summonMap . begin ( ) ; it ! = summonMap . end ( ) ; + + it )
{
bool found = creatureMap . find ( it - > first ) ! = creatureMap . end ( ) ;
int magnitude = creatureStats . getMagicEffects ( ) . get ( it - > first ) . getMagnitude ( ) ;
if ( found ! = ( magnitude > 0 ) )
{
if ( magnitude > 0 )
{
ESM : : Position ipos = ptr . getRefData ( ) . getPosition ( ) ;
Ogre : : Vector3 pos ( ipos . pos ) ;
Ogre : : Quaternion rot ( Ogre : : Radian ( - ipos . rot [ 2 ] ) , Ogre : : Vector3 : : UNIT_Z ) ;
const float distance = 50 ;
pos = pos + distance * rot . yAxis ( ) ;
ipos . pos [ 0 ] = pos . x ;
ipos . pos [ 1 ] = pos . y ;
ipos . pos [ 2 ] = pos . z ;
ipos . rot [ 0 ] = 0 ;
ipos . rot [ 1 ] = 0 ;
ipos . rot [ 2 ] = 0 ;
std : : string creatureID =
MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( it - > second ) - > getString ( ) ;
if ( ! creatureID . empty ( ) )
{
MWWorld : : CellStore * store = ptr . getCell ( ) ;
MWWorld : : ManualRef ref ( MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) , creatureID , 1 ) ;
ref . getPtr ( ) . getCellRef ( ) . setPosition ( ipos ) ;
MWMechanics : : CreatureStats & summonedCreatureStats = ref . getPtr ( ) . getClass ( ) . getCreatureStats ( ref . getPtr ( ) ) ;
// Make the summoned creature follow its master and help in fights
AiFollow package ( ptr . getCellRef ( ) . getRefId ( ) ) ;
summonedCreatureStats . getAiSequence ( ) . stack ( package , ref . getPtr ( ) ) ;
int creatureActorId = summonedCreatureStats . getActorId ( ) ;
MWWorld : : Ptr placed = MWBase : : Environment : : get ( ) . getWorld ( ) - > safePlaceObject ( ref . getPtr ( ) , store , ipos ) ;
MWRender : : Animation * anim = MWBase : : Environment : : get ( ) . getWorld ( ) - > getAnimation ( placed ) ;
if ( anim )
{
const ESM : : Static * fx = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : Static > ( )
. search ( " VFX_Summon_Start " ) ;
if ( fx )
anim - > addEffect ( " meshes \\ " + fx - > mModel , - 1 , false ) ;
}
creatureMap . insert ( std : : make_pair ( it - > first , creatureActorId ) ) ;
}
}
else
{
// Effect has ended
std : : map < int , int > : : iterator foundCreature = creatureMap . find ( it - > first ) ;
cleanupSummonedCreature ( creatureStats , foundCreature - > second ) ;
creatureMap . erase ( foundCreature ) ;
}
}
}
for ( std : : map < int , int > : : iterator it = creatureMap . begin ( ) ; it ! = creatureMap . end ( ) ; )
{
MWWorld : : Ptr ptr = MWBase : : Environment : : get ( ) . getWorld ( ) - > searchPtrViaActorId ( it - > second ) ;
if ( ! ptr . isEmpty ( ) & & ptr . getClass ( ) . getCreatureStats ( ptr ) . isDead ( ) )
{
// Purge the magic effect so a new creature can be summoned if desired
creatureStats . getActiveSpells ( ) . purgeEffect ( it - > first ) ;
if ( ptr . getClass ( ) . hasInventoryStore ( ptr ) )
ptr . getClass ( ) . getInventoryStore ( ptr ) . purgeEffect ( it - > first ) ;
cleanupSummonedCreature ( creatureStats , it - > second ) ;
creatureMap . erase ( it + + ) ;
}
else
+ + it ;
}
std : : vector < int > & graveyard = creatureStats . getSummonedCreatureGraveyard ( ) ;
for ( std : : vector < int > : : iterator it = graveyard . begin ( ) ; it ! = graveyard . end ( ) ; )
{
MWWorld : : Ptr ptr = MWBase : : Environment : : get ( ) . getWorld ( ) - > searchPtrViaActorId ( * it ) ;
if ( ! ptr . isEmpty ( ) )
{
it = graveyard . erase ( it ) ;
const ESM : : Static * fx = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : Static > ( )
. search ( " VFX_Summon_End " ) ;
if ( fx )
MWBase : : Environment : : get ( ) . getWorld ( ) - > spawnEffect ( " meshes \\ " + fx - > mModel ,
" " , Ogre : : Vector3 ( ptr . getRefData ( ) . getPosition ( ) . pos ) ) ;
MWBase : : Environment : : get ( ) . getWorld ( ) - > deleteObject ( ptr ) ;
}
else
+ + it ;
}
}
}
void Actors : : calculateNpcStatModifiers ( const MWWorld : : Ptr & ptr , float duration )
void Actors : : calculateNpcStatModifiers ( const MWWorld : : Ptr & ptr , float duration )
@ -917,10 +786,10 @@ namespace MWMechanics
void Actors : : updateDrowning ( const MWWorld : : Ptr & ptr , float duration )
void Actors : : updateDrowning ( const MWWorld : : Ptr & ptr , float duration )
{
{
Ptr Controlle rMap: : iterator it = mActors . find ( ptr ) ;
Ptr Acto rMap: : iterator it = mActors . find ( ptr ) ;
if ( it = = mActors . end ( ) )
if ( it = = mActors . end ( ) )
return ;
return ;
CharacterController * ctrl = it - > second ;
CharacterController * ctrl = it - > second - > getCharacterController ( ) ;
NpcStats & stats = ptr . getClass ( ) . getNpcStats ( ptr ) ;
NpcStats & stats = ptr . getClass ( ) . getNpcStats ( ptr ) ;
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
@ -1125,14 +994,14 @@ namespace MWMechanics
removeActor ( ptr ) ;
removeActor ( ptr ) ;
MWRender : : Animation * anim = MWBase : : Environment : : get ( ) . getWorld ( ) - > getAnimation ( ptr ) ;
MWRender : : Animation * anim = MWBase : : Environment : : get ( ) . getWorld ( ) - > getAnimation ( ptr ) ;
mActors . insert ( std : : make_pair ( ptr , new CharacterControlle r( ptr , anim ) ) ) ;
mActors . insert ( std : : make_pair ( ptr , new Acto r( ptr , anim ) ) ) ;
if ( updateImmediately )
if ( updateImmediately )
mActors [ ptr ] - > update( 0 ) ;
mActors [ ptr ] - > getCharacterController( ) - > update( 0 ) ;
}
}
void Actors : : removeActor ( const MWWorld : : Ptr & ptr )
void Actors : : removeActor ( const MWWorld : : Ptr & ptr )
{
{
Ptr Controlle rMap: : iterator iter = mActors . find ( ptr ) ;
Ptr Acto rMap: : iterator iter = mActors . find ( ptr ) ;
if ( iter ! = mActors . end ( ) )
if ( iter ! = mActors . end ( ) )
{
{
delete iter - > second ;
delete iter - > second ;
@ -1142,20 +1011,20 @@ namespace MWMechanics
void Actors : : updateActor ( const MWWorld : : Ptr & old , const MWWorld : : Ptr & ptr )
void Actors : : updateActor ( const MWWorld : : Ptr & old , const MWWorld : : Ptr & ptr )
{
{
Ptr Controlle rMap: : iterator iter = mActors . find ( old ) ;
Ptr Acto rMap: : iterator iter = mActors . find ( old ) ;
if ( iter ! = mActors . end ( ) )
if ( iter ! = mActors . end ( ) )
{
{
CharacterController * ctrl = iter - > second ;
Actor * actor = iter - > second ;
mActors . erase ( iter ) ;
mActors . erase ( iter ) ;
ctrl - > updatePtr ( ptr ) ;
a cto r- > updatePtr ( ptr ) ;
mActors . insert ( std : : make_pair ( ptr , ctrl ) ) ;
mActors . insert ( std : : make_pair ( ptr , a cto r) ) ;
}
}
}
}
void Actors : : dropActors ( const MWWorld : : CellStore * cellStore , const MWWorld : : Ptr & ignore )
void Actors : : dropActors ( const MWWorld : : CellStore * cellStore , const MWWorld : : Ptr & ignore )
{
{
Ptr Controlle rMap: : iterator iter = mActors . begin ( ) ;
Ptr Acto rMap: : iterator iter = mActors . begin ( ) ;
while ( iter ! = mActors . end ( ) )
while ( iter ! = mActors . end ( ) )
{
{
if ( iter - > first . getCell ( ) = = cellStore & & iter - > first ! = ignore )
if ( iter - > first . getCell ( ) = = cellStore & & iter - > first ! = ignore )
@ -1189,8 +1058,10 @@ namespace MWMechanics
// using higher values will make a quest in Bloodmoon harder or impossible to complete (bug #1876)
// using higher values will make a quest in Bloodmoon harder or impossible to complete (bug #1876)
const float sqrProcessingDistance = 7168 * 7168 ;
const float sqrProcessingDistance = 7168 * 7168 ;
/// \todo move update logic to Actor class where appropriate
// AI and magic effects update
// AI and magic effects update
for ( PtrControllerMap : : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
if ( ! iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . isDead ( ) )
if ( ! iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . isDead ( ) )
{
{
@ -1204,7 +1075,7 @@ namespace MWMechanics
if ( iter - > first ! = player )
if ( iter - > first ! = player )
adjustCommandedActor ( iter - > first ) ;
adjustCommandedActor ( iter - > first ) ;
for ( Ptr Controlle rMap: : iterator it ( mActors . begin ( ) ) ; it ! = mActors . end ( ) ; + + it )
for ( Ptr Acto rMap: : iterator it ( mActors . begin ( ) ) ; it ! = mActors . end ( ) ; + + it )
{
{
if ( it - > first = = iter - > first | | iter - > first = = player ) // player is not AI-controlled
if ( it - > first = = iter - > first | | iter - > first = = player ) // player is not AI-controlled
continue ;
continue ;
@ -1216,13 +1087,13 @@ namespace MWMechanics
float sqrHeadTrackDistance = std : : numeric_limits < float > : : max ( ) ;
float sqrHeadTrackDistance = std : : numeric_limits < float > : : max ( ) ;
MWWorld : : Ptr headTrackTarget ;
MWWorld : : Ptr headTrackTarget ;
for ( Ptr Controlle rMap: : iterator it ( mActors . begin ( ) ) ; it ! = mActors . end ( ) ; + + it )
for ( Ptr Acto rMap: : iterator it ( mActors . begin ( ) ) ; it ! = mActors . end ( ) ; + + it )
{
{
if ( it - > first = = iter - > first )
if ( it - > first = = iter - > first )
continue ;
continue ;
updateHeadTracking ( iter - > first , it - > first , headTrackTarget , sqrHeadTrackDistance ) ;
updateHeadTracking ( iter - > first , it - > first , headTrackTarget , sqrHeadTrackDistance ) ;
}
}
iter - > second - > setHeadTrackTarget( headTrackTarget ) ;
iter - > second - > getCharacterController( ) - > setHeadTrackTarget( headTrackTarget ) ;
}
}
if ( iter - > first . getClass ( ) . isNpc ( ) & & iter - > first ! = player )
if ( iter - > first . getClass ( ) . isNpc ( ) & & iter - > first ! = player )
@ -1251,12 +1122,12 @@ namespace MWMechanics
// Reaching the text keys may trigger Hit / Spellcast (and as such, particles),
// Reaching the text keys may trigger Hit / Spellcast (and as such, particles),
// so updating VFX immediately after that would just remove the particle effects instantly.
// so updating VFX immediately after that would just remove the particle effects instantly.
// There needs to be a magic effect update in between.
// There needs to be a magic effect update in between.
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
iter - > second - > updateContinuousVfx( ) ;
iter - > second - > getCharacterController( ) - > updateContinuousVfx( ) ;
// Animation/movement update
// Animation/movement update
CharacterController * playerCharacter = NULL ;
CharacterController * playerCharacter = NULL ;
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
if ( iter - > first ! = player & &
if ( iter - > first ! = player & &
Ogre : : Vector3 ( player . getRefData ( ) . getPosition ( ) . pos ) . squaredDistance ( Ogre : : Vector3 ( iter - > first . getRefData ( ) . getPosition ( ) . pos ) )
Ogre : : Vector3 ( player . getRefData ( ) . getPosition ( ) . pos ) . squaredDistance ( Ogre : : Vector3 ( iter - > first . getRefData ( ) . getPosition ( ) . pos ) )
@ -1265,22 +1136,22 @@ namespace MWMechanics
if ( iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . getMagicEffects ( ) . get (
if ( iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . getMagicEffects ( ) . get (
ESM : : MagicEffect : : Paralyze ) . getMagnitude ( ) > 0 )
ESM : : MagicEffect : : Paralyze ) . getMagnitude ( ) > 0 )
iter - > second - > skipAnim( ) ;
iter - > second - > getCharacterController( ) - > skipAnim( ) ;
// Handle player last, in case a cell transition occurs by casting a teleportation spell
// Handle player last, in case a cell transition occurs by casting a teleportation spell
// (would invalidate the iterator)
// (would invalidate the iterator)
if ( iter - > first . getCellRef ( ) . getRefId ( ) = = " player " )
if ( iter - > first . getCellRef ( ) . getRefId ( ) = = " player " )
{
{
playerCharacter = iter - > second ;
playerCharacter = iter - > second - > getCharacterController ( ) ;
continue ;
continue ;
}
}
iter - > second - > update( duration ) ;
iter - > second - > getCharacterController( ) - > update( duration ) ;
}
}
if ( playerCharacter )
if ( playerCharacter )
playerCharacter - > update ( duration ) ;
playerCharacter - > update ( duration ) ;
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
@ -1338,7 +1209,7 @@ namespace MWMechanics
bool detected = false ;
bool detected = false ;
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
if ( iter - > first = = player ) // not the player
if ( iter - > first = = player ) // not the player
continue ;
continue ;
@ -1381,32 +1252,32 @@ namespace MWMechanics
void Actors : : killDeadActors ( )
void Actors : : killDeadActors ( )
{
{
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
if ( ! stats . isDead ( ) )
if ( ! stats . isDead ( ) )
{
{
if ( iter - > second - > isDead( ) )
if ( iter - > second - > getCharacterController( ) - > isDead( ) )
{
{
// Actor has been resurrected. Notify the CharacterController and re-enable collision.
// Actor has been resurrected. Notify the CharacterController and re-enable collision.
MWBase : : Environment : : get ( ) . getWorld ( ) - > enableActorCollision ( iter - > first , true ) ;
MWBase : : Environment : : get ( ) . getWorld ( ) - > enableActorCollision ( iter - > first , true ) ;
iter - > second - > resurrect( ) ;
iter - > second - > getCharacterController( ) - > resurrect( ) ;
}
}
if ( ! stats . isDead ( ) )
if ( ! stats . isDead ( ) )
continue ;
continue ;
}
}
if ( iter - > second - > kill( ) )
if ( iter - > second - > getCharacterController( ) - > kill( ) )
{
{
iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . notifyDied ( ) ;
iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . notifyDied ( ) ;
+ + mDeathCount [ Misc : : StringUtils : : lowerCase ( iter - > first . getCellRef ( ) . getRefId ( ) ) ] ;
+ + mDeathCount [ Misc : : StringUtils : : lowerCase ( iter - > first . getCellRef ( ) . getRefId ( ) ) ] ;
// Make sure spell effects with CasterLinked flag are removed
// Make sure spell effects with CasterLinked flag are removed
for ( Ptr Controlle rMap: : iterator iter2 ( mActors . begin ( ) ) ; iter2 ! = mActors . end ( ) ; + + iter2 )
for ( Ptr Acto rMap: : iterator iter2 ( mActors . begin ( ) ) ; iter2 ! = mActors . end ( ) ; + + iter2 )
{
{
MWMechanics : : ActiveSpells & spells = iter2 - > first . getClass ( ) . getCreatureStats ( iter2 - > first ) . getActiveSpells ( ) ;
MWMechanics : : ActiveSpells & spells = iter2 - > first . getClass ( ) . getCreatureStats ( iter2 - > first ) . getActiveSpells ( ) ;
spells . purge ( stats . getActorId ( ) ) ;
spells . purge ( stats . getActorId ( ) ) ;
@ -1435,7 +1306,7 @@ namespace MWMechanics
void Actors : : restoreDynamicStats ( bool sleep )
void Actors : : restoreDynamicStats ( bool sleep )
{
{
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
restoreDynamicStats ( iter - > first , sleep ) ;
restoreDynamicStats ( iter - > first , sleep ) ;
}
}
@ -1467,35 +1338,35 @@ namespace MWMechanics
void Actors : : forceStateUpdate ( const MWWorld : : Ptr & ptr )
void Actors : : forceStateUpdate ( const MWWorld : : Ptr & ptr )
{
{
Ptr Controlle rMap: : iterator iter = mActors . find ( ptr ) ;
Ptr Acto rMap: : iterator iter = mActors . find ( ptr ) ;
if ( iter ! = mActors . end ( ) )
if ( iter ! = mActors . end ( ) )
iter - > second - > forceStateUpdate( ) ;
iter - > second - > getCharacterController( ) - > forceStateUpdate( ) ;
}
}
void Actors : : playAnimationGroup ( const MWWorld : : Ptr & ptr , const std : : string & groupName , int mode , int number )
void Actors : : playAnimationGroup ( const MWWorld : : Ptr & ptr , const std : : string & groupName , int mode , int number )
{
{
Ptr Controlle rMap: : iterator iter = mActors . find ( ptr ) ;
Ptr Acto rMap: : iterator iter = mActors . find ( ptr ) ;
if ( iter ! = mActors . end ( ) )
if ( iter ! = mActors . end ( ) )
iter - > second - > playGroup( groupName , mode , number ) ;
iter - > second - > getCharacterController( ) - > playGroup( groupName , mode , number ) ;
}
}
void Actors : : skipAnimation ( const MWWorld : : Ptr & ptr )
void Actors : : skipAnimation ( const MWWorld : : Ptr & ptr )
{
{
Ptr Controlle rMap: : iterator iter = mActors . find ( ptr ) ;
Ptr Acto rMap: : iterator iter = mActors . find ( ptr ) ;
if ( iter ! = mActors . end ( ) )
if ( iter ! = mActors . end ( ) )
iter - > second - > skipAnim( ) ;
iter - > second - > getCharacterController( ) - > skipAnim( ) ;
}
}
bool Actors : : checkAnimationPlaying ( const MWWorld : : Ptr & ptr , const std : : string & groupName )
bool Actors : : checkAnimationPlaying ( const MWWorld : : Ptr & ptr , const std : : string & groupName )
{
{
Ptr Controlle rMap: : iterator iter = mActors . find ( ptr ) ;
Ptr Acto rMap: : iterator iter = mActors . find ( ptr ) ;
if ( iter ! = mActors . end ( ) )
if ( iter ! = mActors . end ( ) )
return iter - > second - > isAnimPlaying( groupName ) ;
return iter - > second - > getCharacterController( ) - > isAnimPlaying( groupName ) ;
return false ;
return false ;
}
}
void Actors : : getObjectsInRange ( const Ogre : : Vector3 & position , float radius , std : : vector < MWWorld : : Ptr > & out )
void Actors : : getObjectsInRange ( const Ogre : : Vector3 & position , float radius , std : : vector < MWWorld : : Ptr > & out )
{
{
for ( Ptr Controlle rMap: : iterator iter = mActors . begin ( ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter = mActors . begin ( ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
if ( Ogre : : Vector3 ( iter - > first . getRefData ( ) . getPosition ( ) . pos ) . squaredDistance ( position ) < = radius * radius )
if ( Ogre : : Vector3 ( iter - > first . getRefData ( ) . getPosition ( ) . pos ) . squaredDistance ( position ) < = radius * radius )
out . push_back ( iter - > first ) ;
out . push_back ( iter - > first ) ;
@ -1505,7 +1376,7 @@ namespace MWMechanics
std : : list < MWWorld : : Ptr > Actors : : getActorsFollowing ( const MWWorld : : Ptr & actor )
std : : list < MWWorld : : Ptr > Actors : : getActorsFollowing ( const MWWorld : : Ptr & actor )
{
{
std : : list < MWWorld : : Ptr > list ;
std : : list < MWWorld : : Ptr > list ;
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
@ -1535,7 +1406,7 @@ namespace MWMechanics
std : : list < int > Actors : : getActorsFollowingIndices ( const MWWorld : : Ptr & actor )
std : : list < int > Actors : : getActorsFollowingIndices ( const MWWorld : : Ptr & actor )
{
{
std : : list < int > list ;
std : : list < int > list ;
for ( Ptr Controlle rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
for ( Ptr Acto rMap: : iterator iter ( mActors . begin ( ) ) ; iter ! = mActors . end ( ) ; + + iter )
{
{
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
const MWWorld : : Class & cls = iter - > first . getClass ( ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
CreatureStats & stats = cls . getCreatureStats ( iter - > first ) ;
@ -1608,7 +1479,7 @@ namespace MWMechanics
void Actors : : clear ( )
void Actors : : clear ( )
{
{
Ptr Controlle rMap: : iterator it ( mActors . begin ( ) ) ;
Ptr Acto rMap: : iterator it ( mActors . begin ( ) ) ;
for ( ; it ! = mActors . end ( ) ; + + it )
for ( ; it ! = mActors . end ( ) ; + + it )
{
{
delete it - > second ;
delete it - > second ;
@ -1628,10 +1499,27 @@ namespace MWMechanics
bool Actors : : isReadyToBlock ( const MWWorld : : Ptr & ptr ) const
bool Actors : : isReadyToBlock ( const MWWorld : : Ptr & ptr ) const
{
{
Ptr Controlle rMap: : const_iterator it = mActors . find ( ptr ) ;
Ptr Acto rMap: : const_iterator it = mActors . find ( ptr ) ;
if ( it = = mActors . end ( ) )
if ( it = = mActors . end ( ) )
return false ;
return false ;
return it - > second - > isReadyToBlock ( ) ;
return it - > second - > getCharacterController ( ) - > isReadyToBlock ( ) ;
}
void Actors : : fastForwardAi ( )
{
if ( ! MWBase : : Environment : : get ( ) . getMechanicsManager ( ) - > isAIActive ( ) )
return ;
// making a copy since fast-forward could move actor to a different cell and invalidate the mActors iterator
PtrActorMap map = mActors ;
for ( PtrActorMap : : iterator it = map . begin ( ) ; it ! = map . end ( ) ; + + it )
{
MWWorld : : Ptr ptr = it - > first ;
if ( ptr = = MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayerPtr ( ) )
continue ;
MWMechanics : : AiSequence & seq = ptr . getClass ( ) . getCreatureStats ( ptr ) . getAiSequence ( ) ;
seq . fastForward ( ptr , it - > second - > getAiState ( ) ) ;
}
}
}
}
}