2013-01-10 16:35:24 +00:00
/*
* OpenMW - The completely unofficial reimplementation of Morrowind
*
* This file ( character . cpp ) is part of the OpenMW package .
*
* OpenMW is distributed as free software : you can redistribute it
* and / or modify it under the terms of the GNU General Public License
* version 3 , as published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* version 3 along with this program . If not , see
* http : //www.gnu.org/licenses/ .
*/
# include "character.hpp"
2015-06-14 21:13:26 +00:00
# include <iostream>
2015-04-24 23:20:07 +00:00
# include <components/misc/rng.hpp>
2015-03-15 01:07:47 +00:00
2015-01-31 22:27:34 +00:00
# include <components/settings/settings.hpp>
2015-11-20 20:57:04 +00:00
# include <components/sceneutil/positionattitudetransform.hpp>
2013-01-16 18:45:18 +00:00
# include "../mwrender/animation.hpp"
2013-01-10 16:35:24 +00:00
2013-01-16 23:52:03 +00:00
# include "../mwbase/environment.hpp"
2013-02-04 15:10:14 +00:00
# include "../mwbase/world.hpp"
2013-05-15 08:45:47 +00:00
# include "../mwbase/soundmanager.hpp"
2013-08-03 01:21:42 +00:00
# include "../mwbase/windowmanager.hpp"
2013-01-16 23:52:03 +00:00
2013-01-16 22:37:32 +00:00
# include "../mwworld/class.hpp"
2013-05-01 17:19:16 +00:00
# include "../mwworld/inventorystore.hpp"
2014-02-23 19:11:05 +00:00
# include "../mwworld/esmstore.hpp"
2015-07-16 17:56:09 +00:00
# include "../mwworld/player.hpp"
2013-01-16 21:09:21 +00:00
2016-06-17 14:07:16 +00:00
# include "movement.hpp"
# include "npcstats.hpp"
# include "creaturestats.hpp"
# include "security.hpp"
# include "actorutil.hpp"
2016-07-16 13:47:33 +00:00
# include "spellcasting.hpp"
2016-06-17 14:07:16 +00:00
2013-07-16 21:32:41 +00:00
namespace
{
2014-12-16 19:47:45 +00:00
// Wraps a value to (-PI, PI]
2015-05-31 16:04:14 +00:00
void wrap ( float & rad )
2014-12-16 19:47:45 +00:00
{
2015-05-31 16:04:14 +00:00
if ( rad > 0 )
rad = std : : fmod ( rad + osg : : PI , 2.0f * osg : : PI ) - osg : : PI ;
2014-12-16 19:47:45 +00:00
else
2015-05-31 16:04:14 +00:00
rad = std : : fmod ( rad - osg : : PI , 2.0f * osg : : PI ) + osg : : PI ;
2014-12-16 19:47:45 +00:00
}
2015-06-03 17:59:54 +00:00
std : : string toString ( int num )
{
std : : ostringstream stream ;
stream < < num ;
return stream . str ( ) ;
}
2014-01-27 21:05:17 +00:00
std : : string getBestAttack ( const ESM : : Weapon * weapon )
2013-07-16 21:32:41 +00:00
{
int slash = ( weapon - > mData . mSlash [ 0 ] + weapon - > mData . mSlash [ 1 ] ) / 2 ;
int chop = ( weapon - > mData . mChop [ 0 ] + weapon - > mData . mChop [ 1 ] ) / 2 ;
int thrust = ( weapon - > mData . mThrust [ 0 ] + weapon - > mData . mThrust [ 1 ] ) / 2 ;
2017-01-11 13:09:26 +00:00
if ( slash = = chop & & slash = = thrust )
2014-01-27 21:05:17 +00:00
return " slash " ;
2017-01-11 13:09:26 +00:00
else if ( thrust > = chop & & thrust > = slash )
2014-01-27 21:05:17 +00:00
return " thrust " ;
2017-01-11 13:09:26 +00:00
else if ( slash > = chop & & slash > = thrust )
return " slash " ;
else
return " chop " ;
2013-07-16 21:32:41 +00:00
}
2014-06-24 22:51:02 +00:00
// Converts a movement Run state to its equivalent Walk state.
MWMechanics : : CharacterState runStateToWalkState ( MWMechanics : : CharacterState state )
{
using namespace MWMechanics ;
CharacterState ret = state ;
switch ( state )
{
case CharState_RunForward :
ret = CharState_WalkForward ;
break ;
case CharState_RunBack :
ret = CharState_WalkBack ;
break ;
case CharState_RunLeft :
ret = CharState_WalkLeft ;
break ;
case CharState_RunRight :
ret = CharState_WalkRight ;
break ;
case CharState_SwimRunForward :
ret = CharState_SwimWalkForward ;
break ;
case CharState_SwimRunBack :
ret = CharState_SwimWalkBack ;
break ;
case CharState_SwimRunLeft :
ret = CharState_SwimWalkLeft ;
break ;
case CharState_SwimRunRight :
ret = CharState_SwimWalkRight ;
break ;
default :
break ;
}
return ret ;
}
2014-12-10 16:21:34 +00:00
float getFallDamage ( const MWWorld : : Ptr & ptr , float fallHeight )
{
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
const MWWorld : : Store < ESM : : GameSetting > & store = world - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
const float fallDistanceMin = store . find ( " fFallDamageDistanceMin " ) - > getFloat ( ) ;
if ( fallHeight > = fallDistanceMin )
{
2015-03-08 04:42:07 +00:00
const float acrobaticsSkill = static_cast < float > ( ptr . getClass ( ) . getSkill ( ptr , ESM : : Skill : : Acrobatics ) ) ;
2014-12-10 16:21:34 +00:00
const float jumpSpellBonus = ptr . getClass ( ) . getCreatureStats ( ptr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Jump ) . getMagnitude ( ) ;
const float fallAcroBase = store . find ( " fFallAcroBase " ) - > getFloat ( ) ;
const float fallAcroMult = store . find ( " fFallAcroMult " ) - > getFloat ( ) ;
const float fallDistanceBase = store . find ( " fFallDistanceBase " ) - > getFloat ( ) ;
const float fallDistanceMult = store . find ( " fFallDistanceMult " ) - > getFloat ( ) ;
float x = fallHeight - fallDistanceMin ;
2015-03-08 04:42:07 +00:00
x - = ( 1.5f * acrobaticsSkill ) + jumpSpellBonus ;
2014-12-10 16:21:34 +00:00
x = std : : max ( 0.0f , x ) ;
float a = fallAcroBase + fallAcroMult * ( 100 - acrobaticsSkill ) ;
x = fallDistanceBase + fallDistanceMult * x ;
x * = a ;
return x ;
}
return 0.f ;
}
2013-07-16 21:32:41 +00:00
}
2013-02-04 15:10:14 +00:00
2013-01-10 16:35:24 +00:00
namespace MWMechanics
{
2013-07-16 05:56:23 +00:00
struct StateInfo {
2013-01-19 22:56:24 +00:00
CharacterState state ;
const char groupname [ 32 ] ;
2013-07-16 05:56:23 +00:00
} ;
static const StateInfo sMovementList [ ] = {
{ CharState_WalkForward , " walkforward " } ,
{ CharState_WalkBack , " walkback " } ,
{ CharState_WalkLeft , " walkleft " } ,
{ CharState_WalkRight , " walkright " } ,
{ CharState_SwimWalkForward , " swimwalkforward " } ,
{ CharState_SwimWalkBack , " swimwalkback " } ,
{ CharState_SwimWalkLeft , " swimwalkleft " } ,
{ CharState_SwimWalkRight , " swimwalkright " } ,
{ CharState_RunForward , " runforward " } ,
{ CharState_RunBack , " runback " } ,
{ CharState_RunLeft , " runleft " } ,
{ CharState_RunRight , " runright " } ,
{ CharState_SwimRunForward , " swimrunforward " } ,
{ CharState_SwimRunBack , " swimrunback " } ,
{ CharState_SwimRunLeft , " swimrunleft " } ,
{ CharState_SwimRunRight , " swimrunright " } ,
{ CharState_SneakForward , " sneakforward " } ,
{ CharState_SneakBack , " sneakback " } ,
{ CharState_SneakLeft , " sneakleft " } ,
{ CharState_SneakRight , " sneakright " } ,
2013-08-19 06:42:56 +00:00
{ CharState_Jump , " jump " } ,
2013-07-16 05:56:23 +00:00
{ CharState_TurnLeft , " turnleft " } ,
{ CharState_TurnRight , " turnright " } ,
} ;
static const StateInfo * sMovementListEnd = & sMovementList [ sizeof ( sMovementList ) / sizeof ( sMovementList [ 0 ] ) ] ;
2013-05-13 11:08:36 +00:00
class FindCharState {
CharacterState state ;
public :
FindCharState ( CharacterState _state ) : state ( _state ) { }
bool operator ( ) ( const StateInfo & info ) const
{ return info . state = = state ; }
} ;
static const struct WeaponInfo {
2013-05-11 05:22:39 +00:00
WeaponType type ;
2013-07-16 05:56:23 +00:00
const char shortgroup [ 16 ] ;
const char longgroup [ 16 ] ;
2013-05-11 05:22:39 +00:00
} sWeaponTypeList [ ] = {
2013-07-16 05:56:23 +00:00
{ WeapType_HandToHand , " hh " , " handtohand " } ,
{ WeapType_OneHand , " 1h " , " weapononehand " } ,
{ WeapType_TwoHand , " 2c " , " weapontwohand " } ,
{ WeapType_TwoWide , " 2w " , " weapontwowide " } ,
{ WeapType_BowAndArrow , " 1h " , " bowandarrow " } ,
{ WeapType_Crossbow , " crossbow " , " crossbow " } ,
2014-02-04 02:55:40 +00:00
{ WeapType_Thrown , " 1h " , " throwweapon " } ,
2013-07-16 05:56:23 +00:00
{ WeapType_PickProbe , " 1h " , " pickprobe " } ,
{ WeapType_Spell , " spell " , " spellcast " } ,
2013-05-01 17:19:16 +00:00
} ;
2013-05-13 11:08:36 +00:00
static const WeaponInfo * sWeaponTypeListEnd = & sWeaponTypeList [ sizeof ( sWeaponTypeList ) / sizeof ( sWeaponTypeList [ 0 ] ) ] ;
class FindWeaponType {
WeaponType type ;
public :
FindWeaponType ( WeaponType _type ) : type ( _type ) { }
bool operator ( ) ( const WeaponInfo & weap ) const
{ return weap . type = = type ; }
} ;
2013-05-01 17:19:16 +00:00
2016-05-19 20:30:14 +00:00
std : : string CharacterController : : chooseRandomGroup ( const std : : string & prefix , int * num ) const
2014-01-19 20:11:48 +00:00
{
int numAnims = 0 ;
2015-06-03 17:59:54 +00:00
while ( mAnimation - > hasAnimation ( prefix + toString ( numAnims + 1 ) ) )
2014-01-19 20:11:48 +00:00
+ + numAnims ;
2015-04-24 23:20:07 +00:00
int roll = Misc : : Rng : : rollDice ( numAnims ) + 1 ; // [1, numAnims]
2014-01-19 20:11:48 +00:00
if ( num )
* num = roll ;
2015-06-03 17:59:54 +00:00
return prefix + toString ( roll ) ;
2014-01-19 20:11:48 +00:00
}
2013-05-01 17:19:16 +00:00
2016-06-15 01:14:44 +00:00
void CharacterController : : refreshHitRecoilAnims ( )
2013-01-19 22:56:24 +00:00
{
2016-06-15 01:14:44 +00:00
bool recovery = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getHitRecovery ( ) ;
bool knockdown = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getKnockedDown ( ) ;
bool block = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getBlock ( ) ;
if ( mHitState = = CharState_None )
2013-12-31 11:24:20 +00:00
{
2016-06-15 01:14:44 +00:00
if ( ( mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getFatigue ( ) . getCurrent ( ) < 0
| | mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getFatigue ( ) . getBase ( ) = = 0 )
& & mAnimation - > hasAnimation ( " knockout " ) )
2013-12-31 11:24:20 +00:00
{
2016-06-15 01:14:44 +00:00
mHitState = CharState_KnockOut ;
mCurrentHit = " knockout " ;
mAnimation - > play ( mCurrentHit , Priority_Knockdown , MWRender : : Animation : : BlendMask_All , false , 1 , " start " , " stop " , 0.0f , ~ 0ul ) ;
mPtr . getClass ( ) . getCreatureStats ( mPtr ) . setKnockedDown ( true ) ;
}
else if ( knockdown & & mAnimation - > hasAnimation ( " knockdown " ) )
{
mHitState = CharState_KnockDown ;
mCurrentHit = " knockdown " ;
mAnimation - > play ( mCurrentHit , Priority_Knockdown , MWRender : : Animation : : BlendMask_All , true , 1 , " start " , " stop " , 0.0f , 0 ) ;
}
else if ( recovery )
{
std : : string anim = chooseRandomGroup ( " hit " ) ;
if ( mAnimation - > hasAnimation ( anim ) )
2014-01-21 00:01:21 +00:00
{
2016-06-15 01:14:44 +00:00
mHitState = CharState_Hit ;
mCurrentHit = anim ;
mAnimation - > play ( mCurrentHit , Priority_Hit , MWRender : : Animation : : BlendMask_All , true , 1 , " start " , " stop " , 0.0f , 0 ) ;
2014-07-28 17:40:56 +00:00
}
2013-12-31 11:24:20 +00:00
}
2016-06-15 01:14:44 +00:00
else if ( block & & mAnimation - > hasAnimation ( " shield " ) )
2014-01-02 19:54:41 +00:00
{
2016-06-15 01:14:44 +00:00
mHitState = CharState_Block ;
mCurrentHit = " shield " ;
MWRender : : Animation : : AnimPriority priorityBlock ( Priority_Hit ) ;
priorityBlock [ MWRender : : Animation : : BoneGroup_LeftArm ] = Priority_Block ;
mAnimation - > play ( mCurrentHit , priorityBlock , MWRender : : Animation : : BlendMask_All , true , 1 , " block start " , " block stop " , 0.0f , 0 ) ;
2014-01-02 19:54:41 +00:00
}
2016-06-15 01:14:44 +00:00
// Cancel upper body animations
if ( mHitState = = CharState_KnockDown | | mHitState = = CharState_KnockOut )
2014-02-02 02:24:40 +00:00
{
2016-06-15 01:14:44 +00:00
if ( mUpperBodyState > UpperCharState_WeapEquiped )
{
mAnimation - > disable ( mCurrentWeapon ) ;
mUpperBodyState = UpperCharState_WeapEquiped ;
}
else if ( mUpperBodyState > UpperCharState_Nothing & & mUpperBodyState < UpperCharState_WeapEquiped )
{
mAnimation - > disable ( mCurrentWeapon ) ;
mUpperBodyState = UpperCharState_Nothing ;
}
2014-02-02 02:24:40 +00:00
}
2013-12-31 11:24:20 +00:00
}
2016-06-15 01:14:44 +00:00
else if ( ! mAnimation - > isPlaying ( mCurrentHit ) )
{
mCurrentHit . erase ( ) ;
if ( knockdown )
mPtr . getClass ( ) . getCreatureStats ( mPtr ) . setKnockedDown ( false ) ;
if ( recovery )
mPtr . getClass ( ) . getCreatureStats ( mPtr ) . setHitRecovery ( false ) ;
if ( block )
mPtr . getClass ( ) . getCreatureStats ( mPtr ) . setBlock ( false ) ;
mHitState = CharState_None ;
}
else if ( mHitState = = CharState_KnockOut & & mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getFatigue ( ) . getCurrent ( ) > 0 )
{
mHitState = CharState_KnockDown ;
mAnimation - > disable ( mCurrentHit ) ;
mAnimation - > play ( mCurrentHit , Priority_Knockdown , MWRender : : Animation : : BlendMask_All , true , 1 , " loop stop " , " stop " , 0.0f , 0 ) ;
}
2016-08-25 13:19:49 +00:00
if ( mHitState ! = CharState_None )
mIdleState = CharState_None ;
2016-06-15 01:14:44 +00:00
}
2013-12-31 11:24:20 +00:00
2016-06-15 01:14:44 +00:00
void CharacterController : : refreshJumpAnims ( const WeaponInfo * weap , JumpingState jump , bool force )
{
2015-07-16 18:03:16 +00:00
if ( force | | jump ! = mJumpState )
2013-08-19 15:10:18 +00:00
{
2016-08-25 13:19:49 +00:00
mIdleState = CharState_None ;
2015-07-16 18:03:16 +00:00
bool startAtLoop = ( jump = = mJumpState ) ;
mJumpState = jump ;
std : : string jumpAnimName ;
2015-07-09 16:47:11 +00:00
MWRender : : Animation : : BlendMask jumpmask = MWRender : : Animation : : BlendMask_All ;
2013-08-19 15:10:18 +00:00
if ( mJumpState ! = JumpState_None )
{
2015-07-16 18:03:16 +00:00
jumpAnimName = " jump " ;
2013-08-19 15:10:18 +00:00
if ( weap ! = sWeaponTypeListEnd )
{
2015-07-16 18:03:16 +00:00
jumpAnimName + = weap - > shortgroup ;
if ( ! mAnimation - > hasAnimation ( jumpAnimName ) )
2013-08-19 15:10:18 +00:00
{
2015-07-09 16:47:11 +00:00
jumpmask = MWRender : : Animation : : BlendMask_LowerBody ;
2015-07-16 18:03:16 +00:00
jumpAnimName = " jump " ;
2013-08-19 15:10:18 +00:00
}
}
}
2014-09-17 00:20:46 +00:00
if ( mJumpState = = JumpState_InAir )
2013-08-19 15:10:18 +00:00
{
mAnimation - > disable ( mCurrentJump ) ;
2015-07-16 18:03:16 +00:00
mCurrentJump = jumpAnimName ;
2014-01-19 12:46:40 +00:00
if ( mAnimation - > hasAnimation ( " jump " ) )
2015-07-09 16:47:11 +00:00
mAnimation - > play ( mCurrentJump , Priority_Jump , jumpmask , false ,
2015-07-16 18:03:16 +00:00
1.0f , ( startAtLoop ? " loop start " : " start " ) , " stop " , 0.0f , ~ 0ul ) ;
2013-08-19 15:10:18 +00:00
}
else
{
mAnimation - > disable ( mCurrentJump ) ;
2014-01-08 14:05:14 +00:00
mCurrentJump . clear ( ) ;
2014-01-19 12:46:40 +00:00
if ( mAnimation - > hasAnimation ( " jump " ) )
2015-07-16 18:03:16 +00:00
mAnimation - > play ( jumpAnimName , Priority_Jump , jumpmask , true ,
2013-08-19 15:10:18 +00:00
1.0f , " loop stop " , " stop " , 0.0f , 0 ) ;
}
}
2016-06-15 01:14:44 +00:00
}
2013-08-19 15:10:18 +00:00
2016-06-15 01:14:44 +00:00
void CharacterController : : refreshMovementAnims ( const WeaponInfo * weap , CharacterState movement , bool force )
{
2013-07-16 05:56:23 +00:00
if ( force | | movement ! = mMovementState )
{
mMovementState = movement ;
2017-03-23 18:36:34 +00:00
if ( movement ! = CharState_None )
mIdleState = CharState_None ;
2015-07-15 13:46:31 +00:00
std : : string movementAnimName ;
2015-07-09 16:47:11 +00:00
MWRender : : Animation : : BlendMask movemask = MWRender : : Animation : : BlendMask_All ;
2013-07-16 05:56:23 +00:00
const StateInfo * movestate = std : : find_if ( sMovementList , sMovementListEnd , FindCharState ( mMovementState ) ) ;
if ( movestate ! = sMovementListEnd )
{
2015-07-15 13:46:31 +00:00
movementAnimName = movestate - > groupname ;
if ( weap ! = sWeaponTypeListEnd & & movementAnimName . find ( " swim " ) = = std : : string : : npos )
2013-07-16 05:56:23 +00:00
{
2015-07-15 13:46:31 +00:00
movementAnimName + = weap - > shortgroup ;
if ( ! mAnimation - > hasAnimation ( movementAnimName ) )
2013-07-16 05:56:23 +00:00
{
2015-07-09 16:47:11 +00:00
movemask = MWRender : : Animation : : BlendMask_LowerBody ;
2015-07-15 13:46:31 +00:00
movementAnimName = movestate - > groupname ;
2013-07-16 05:56:23 +00:00
}
}
2015-07-15 13:46:31 +00:00
if ( ! mAnimation - > hasAnimation ( movementAnimName ) )
2013-07-16 05:56:23 +00:00
{
2015-07-15 13:46:31 +00:00
std : : string : : size_type swimpos = movementAnimName . find ( " swim " ) ;
2013-07-18 07:22:40 +00:00
if ( swimpos = = std : : string : : npos )
2014-06-27 22:07:13 +00:00
{
2015-07-15 13:46:31 +00:00
std : : string : : size_type runpos = movementAnimName . find ( " run " ) ;
2014-06-27 22:07:13 +00:00
if ( runpos ! = std : : string : : npos )
{
2015-07-15 13:46:31 +00:00
movementAnimName . replace ( runpos , runpos + 3 , " walk " ) ;
if ( ! mAnimation - > hasAnimation ( movementAnimName ) )
movementAnimName . clear ( ) ;
2014-06-27 22:07:13 +00:00
}
else
2015-07-15 13:46:31 +00:00
movementAnimName . clear ( ) ;
2014-06-27 22:07:13 +00:00
}
2013-07-16 05:56:23 +00:00
else
{
2015-07-15 13:46:31 +00:00
movementAnimName . erase ( swimpos , 4 ) ;
2015-12-11 00:24:33 +00:00
if ( weap ! = sWeaponTypeListEnd )
{
std : : string weapMovementAnimName = movementAnimName + weap - > shortgroup ;
if ( mAnimation - > hasAnimation ( weapMovementAnimName ) )
movementAnimName = weapMovementAnimName ;
else
movemask = MWRender : : Animation : : BlendMask_LowerBody ;
}
if ( ! mAnimation - > hasAnimation ( movementAnimName ) )
2015-07-15 13:46:31 +00:00
movementAnimName . clear ( ) ;
2013-07-16 05:56:23 +00:00
}
}
}
2013-08-10 04:25:28 +00:00
/* If we're playing the same animation, restart from the loop start instead of the
* beginning . */
2015-07-15 13:46:31 +00:00
int mode = ( ( movementAnimName = = mCurrentMovement ) ? 2 : 1 ) ;
2013-08-10 04:25:28 +00:00
2014-07-03 15:40:44 +00:00
mMovementAnimationControlled = true ;
2013-07-16 05:56:23 +00:00
mAnimation - > disable ( mCurrentMovement ) ;
2015-07-15 13:46:31 +00:00
mCurrentMovement = movementAnimName ;
2013-07-16 05:56:23 +00:00
if ( ! mCurrentMovement . empty ( ) )
2013-07-16 08:30:03 +00:00
{
2014-10-14 23:06:16 +00:00
bool isrunning = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getStance ( MWMechanics : : CreatureStats : : Stance_Run )
& & ! MWBase : : Environment : : get ( ) . getWorld ( ) - > isFlying ( mPtr ) ;
2014-02-06 02:58:07 +00:00
2014-06-24 22:51:02 +00:00
// For non-flying creatures, MW uses the Walk animation to calculate the animation velocity
// even if we are running. This must be replicated, otherwise the observed speed would differ drastically.
std : : string anim = mCurrentMovement ;
2015-07-25 16:22:48 +00:00
mAdjustMovementAnimSpeed = true ;
2014-06-24 22:51:02 +00:00
if ( mPtr . getClass ( ) . getTypeName ( ) = = typeid ( ESM : : Creature ) . name ( )
& & ! ( mPtr . get < ESM : : Creature > ( ) - > mBase - > mFlags & ESM : : Creature : : Flies ) )
{
CharacterState walkState = runStateToWalkState ( mMovementState ) ;
const StateInfo * stateinfo = std : : find_if ( sMovementList , sMovementListEnd , FindCharState ( walkState ) ) ;
anim = stateinfo - > groupname ;
2015-07-25 16:22:48 +00:00
mMovementAnimSpeed = mAnimation - > getVelocity ( anim ) ;
if ( mMovementAnimSpeed < = 1.0f )
{
2014-08-12 23:53:56 +00:00
// Another bug: when using a fallback animation (e.g. RunForward as fallback to SwimRunForward),
// then the equivalent Walk animation will not use a fallback, and if that animation doesn't exist
// we will play without any scaling.
// Makes the speed attribute of most water creatures totally useless.
// And again, this can not be fixed without patching game data.
2015-07-25 16:22:48 +00:00
mAdjustMovementAnimSpeed = false ;
mMovementAnimSpeed = 1.f ;
}
2014-03-07 05:11:00 +00:00
}
2014-08-12 23:53:56 +00:00
else
2014-07-03 15:40:44 +00:00
{
2015-07-25 16:22:48 +00:00
mMovementAnimSpeed = mAnimation - > getVelocity ( anim ) ;
if ( mMovementAnimSpeed < = 1.0f )
2014-08-12 23:53:56 +00:00
{
// The first person anims don't have any velocity to calculate a speed multiplier from.
// We use the third person velocities instead.
// FIXME: should be pulled from the actual animation, but it is not presently loaded.
2015-07-25 16:22:48 +00:00
mMovementAnimSpeed = ( isrunning ? 222.857f : 154.064f ) ;
2014-08-12 23:53:56 +00:00
mMovementAnimationControlled = false ;
}
2014-07-03 15:40:44 +00:00
}
2014-08-12 23:53:56 +00:00
2015-07-25 16:59:16 +00:00
mAnimation - > play ( mCurrentMovement , Priority_Movement , movemask , false ,
2016-10-03 20:33:16 +00:00
1.f , ( ( mode ! = 2 ) ? " start " : " loop start " ) , " stop " , 0.0f , ~ 0ul , true ) ;
2013-07-16 08:30:03 +00:00
}
2013-07-16 05:56:23 +00:00
}
2016-06-15 01:14:44 +00:00
}
2014-12-31 22:01:31 +00:00
2016-06-15 01:14:44 +00:00
void CharacterController : : refreshIdleAnims ( const WeaponInfo * weap , CharacterState idle , bool force )
{
2016-08-25 12:35:03 +00:00
if ( force | | idle ! = mIdleState | |
( ( idle = = mIdleState ) & & ! mAnimation - > isPlaying ( mCurrentIdle ) & & mAnimQueue . empty ( ) ) )
2014-12-31 22:01:31 +00:00
{
mIdleState = idle ;
2016-08-25 12:17:40 +00:00
size_t numLoops = ~ 0ul ;
2014-12-31 22:01:31 +00:00
2016-10-05 16:12:06 +00:00
std : : string idleGroup ;
2015-07-30 22:52:34 +00:00
MWRender : : Animation : : AnimPriority idlePriority ( Priority_Default ) ;
2014-12-31 22:01:31 +00:00
// Only play "idleswim" or "idlesneak" if they exist. Otherwise, fallback to
// "idle"+weapon or "idle".
if ( mIdleState = = CharState_IdleSwim & & mAnimation - > hasAnimation ( " idleswim " ) )
2015-07-30 22:52:34 +00:00
{
2016-10-05 16:12:06 +00:00
idleGroup = " idleswim " ;
2015-07-30 22:52:34 +00:00
idlePriority = Priority_SwimIdle ;
}
2014-12-31 22:01:31 +00:00
else if ( mIdleState = = CharState_IdleSneak & & mAnimation - > hasAnimation ( " idlesneak " ) )
2015-09-16 14:14:17 +00:00
{
2016-10-05 16:12:06 +00:00
idleGroup = " idlesneak " ;
2015-09-16 14:14:17 +00:00
idlePriority [ MWRender : : Animation : : BoneGroup_LowerBody ] = Priority_SneakIdleLowerBody ;
}
2014-12-31 22:01:31 +00:00
else if ( mIdleState ! = CharState_None )
{
2016-10-05 16:12:06 +00:00
idleGroup = " idle " ;
2014-12-31 22:01:31 +00:00
if ( weap ! = sWeaponTypeListEnd )
{
2016-10-05 16:12:06 +00:00
idleGroup + = weap - > shortgroup ;
if ( ! mAnimation - > hasAnimation ( idleGroup ) )
idleGroup = " idle " ;
2016-08-25 14:42:24 +00:00
// play until the Loop Stop key 2 to 5 times, then play until the Stop key
// this replicates original engine behavior for the "Idle1h" 1st-person animation
numLoops = 1 + Misc : : Rng : : rollDice ( 4 ) ;
2016-08-25 12:17:40 +00:00
}
2014-12-31 22:01:31 +00:00
}
mAnimation - > disable ( mCurrentIdle ) ;
2016-10-05 16:12:06 +00:00
mCurrentIdle = idleGroup ;
2014-12-31 22:01:31 +00:00
if ( ! mCurrentIdle . empty ( ) )
2015-07-30 22:52:34 +00:00
mAnimation - > play ( mCurrentIdle , idlePriority , MWRender : : Animation : : BlendMask_All , false ,
2016-08-25 12:17:40 +00:00
1.0f , " start " , " stop " , 0.0f , numLoops , true ) ;
2014-12-31 22:01:31 +00:00
}
2013-01-19 22:56:24 +00:00
}
2016-06-15 01:14:44 +00:00
void CharacterController : : refreshCurrentAnims ( CharacterState idle , CharacterState movement , JumpingState jump , bool force )
{
if ( mPtr . getClass ( ) . isActor ( ) )
refreshHitRecoilAnims ( ) ;
const WeaponInfo * weap = std : : find_if ( sWeaponTypeList , sWeaponTypeListEnd , FindWeaponType ( mWeaponType ) ) ;
if ( ! mPtr . getClass ( ) . isBipedal ( mPtr ) )
weap = sWeaponTypeListEnd ;
refreshJumpAnims ( weap , jump , force ) ;
refreshMovementAnims ( weap , movement , force ) ;
// idle handled last as it can depend on the other states
// FIXME: if one of the below states is close to their last animation frame (i.e. will be disabled in the coming update),
// the idle animation should be displayed
if ( ( mUpperBodyState ! = UpperCharState_Nothing
| | ( mMovementState ! = CharState_None & & mMovementState ! = CharState_TurnLeft & & mMovementState ! = CharState_TurnRight )
| | mHitState ! = CharState_None )
& & ! mPtr . getClass ( ) . isBipedal ( mPtr ) )
idle = CharState_None ;
refreshIdleAnims ( weap , idle , force ) ;
}
2013-01-22 06:51:13 +00:00
2014-01-15 20:56:55 +00:00
void getWeaponGroup ( WeaponType weaptype , std : : string & group )
2013-05-13 10:32:00 +00:00
{
2013-05-13 11:08:36 +00:00
const WeaponInfo * info = std : : find_if ( sWeaponTypeList , sWeaponTypeListEnd , FindWeaponType ( weaptype ) ) ;
if ( info ! = sWeaponTypeListEnd )
2013-07-16 05:56:23 +00:00
group = info - > longgroup ;
2015-12-26 17:45:09 +00:00
else
group . clear ( ) ;
2013-05-13 10:32:00 +00:00
}
2014-01-27 20:38:01 +00:00
MWWorld : : ContainerStoreIterator getActiveWeapon ( CreatureStats & stats , MWWorld : : InventoryStore & inv , WeaponType * weaptype )
2013-07-25 04:08:16 +00:00
{
if ( stats . getDrawState ( ) = = DrawState_Spell )
{
* weaptype = WeapType_Spell ;
return inv . end ( ) ;
}
if ( stats . getDrawState ( ) = = MWMechanics : : DrawState_Weapon )
{
MWWorld : : ContainerStoreIterator weapon = inv . getSlot ( MWWorld : : InventoryStore : : Slot_CarriedRight ) ;
if ( weapon = = inv . end ( ) )
* weaptype = WeapType_HandToHand ;
else
{
const std : : string & type = weapon - > getTypeName ( ) ;
if ( type = = typeid ( ESM : : Lockpick ) . name ( ) | | type = = typeid ( ESM : : Probe ) . name ( ) )
* weaptype = WeapType_PickProbe ;
else if ( type = = typeid ( ESM : : Weapon ) . name ( ) )
{
MWWorld : : LiveCellRef < ESM : : Weapon > * ref = weapon - > get < ESM : : Weapon > ( ) ;
2016-10-05 16:12:06 +00:00
ESM : : Weapon : : Type weaponType = ( ESM : : Weapon : : Type ) ref - > mBase - > mData . mType ;
switch ( weaponType )
2013-07-25 04:08:16 +00:00
{
case ESM : : Weapon : : ShortBladeOneHand :
case ESM : : Weapon : : LongBladeOneHand :
case ESM : : Weapon : : BluntOneHand :
case ESM : : Weapon : : AxeOneHand :
case ESM : : Weapon : : Arrow :
case ESM : : Weapon : : Bolt :
* weaptype = WeapType_OneHand ;
break ;
case ESM : : Weapon : : LongBladeTwoHand :
case ESM : : Weapon : : BluntTwoClose :
case ESM : : Weapon : : AxeTwoHand :
* weaptype = WeapType_TwoHand ;
break ;
case ESM : : Weapon : : BluntTwoWide :
case ESM : : Weapon : : SpearTwoWide :
* weaptype = WeapType_TwoWide ;
break ;
case ESM : : Weapon : : MarksmanBow :
* weaptype = WeapType_BowAndArrow ;
break ;
case ESM : : Weapon : : MarksmanCrossbow :
* weaptype = WeapType_Crossbow ;
break ;
case ESM : : Weapon : : MarksmanThrown :
2014-02-04 02:55:40 +00:00
* weaptype = WeapType_Thrown ;
2013-07-25 04:08:16 +00:00
break ;
}
}
}
return weapon ;
}
return inv . getSlot ( MWWorld : : InventoryStore : : Slot_CarriedRight ) ;
}
2014-05-26 17:56:32 +00:00
void CharacterController : : playDeath ( float startpoint , CharacterState death )
{
switch ( death )
{
case CharState_SwimDeath :
mCurrentDeath = " swimdeath " ;
break ;
case CharState_DeathKnockDown :
mCurrentDeath = " deathknockdown " ;
break ;
case CharState_DeathKnockOut :
mCurrentDeath = " deathknockout " ;
break ;
default :
2015-06-03 17:59:54 +00:00
mCurrentDeath = " death " + toString ( death - CharState_Death1 + 1 ) ;
2014-05-26 17:56:32 +00:00
}
mDeathState = death ;
mPtr . getClass ( ) . getCreatureStats ( mPtr ) . setDeathAnimation ( mDeathState - CharState_Death1 ) ;
// For dead actors, refreshCurrentAnims is no longer called, so we need to disable the movement state manually.
2014-06-14 19:27:16 +00:00
// Note that these animations wouldn't actually be visible (due to the Death animation's priority being higher).
// However, they could still trigger text keys, such as Hit events, or sounds.
2014-05-26 17:56:32 +00:00
mMovementState = CharState_None ;
mAnimation - > disable ( mCurrentMovement ) ;
mCurrentMovement = " " ;
2014-06-14 19:27:16 +00:00
mUpperBodyState = UpperCharState_Nothing ;
mAnimation - > disable ( mCurrentWeapon ) ;
mCurrentWeapon = " " ;
mHitState = CharState_None ;
mAnimation - > disable ( mCurrentHit ) ;
mCurrentHit = " " ;
mIdleState = CharState_None ;
mAnimation - > disable ( mCurrentIdle ) ;
mCurrentIdle = " " ;
mJumpState = JumpState_None ;
mAnimation - > disable ( mCurrentJump ) ;
mCurrentJump = " " ;
2014-07-03 15:40:44 +00:00
mMovementAnimationControlled = true ;
2014-05-26 17:56:32 +00:00
2015-07-09 16:47:11 +00:00
mAnimation - > play ( mCurrentDeath , Priority_Death , MWRender : : Animation : : BlendMask_All ,
2014-05-26 17:56:32 +00:00
false , 1.0f , " start " , " stop " , startpoint , 0 ) ;
}
2016-05-19 20:30:14 +00:00
CharacterState CharacterController : : chooseRandomDeathState ( ) const
2016-05-19 19:37:24 +00:00
{
int selected = 0 ;
chooseRandomGroup ( " death " , & selected ) ;
return static_cast < CharacterState > ( CharState_Death1 + ( selected - 1 ) ) ;
}
2013-12-31 11:24:20 +00:00
void CharacterController : : playRandomDeath ( float startpoint )
{
2015-08-21 09:12:39 +00:00
if ( mPtr = = getPlayer ( ) )
2014-06-18 13:33:09 +00:00
{
// The first-person animations do not include death, so we need to
// force-switch to third person before playing the death animation.
MWBase : : Environment : : get ( ) . getWorld ( ) - > useDeathCamera ( ) ;
}
2014-01-19 20:11:48 +00:00
if ( MWBase : : Environment : : get ( ) . getWorld ( ) - > isSwimming ( mPtr ) & & mAnimation - > hasAnimation ( " swimdeath " ) )
2013-12-31 11:24:20 +00:00
{
2014-01-19 20:11:48 +00:00
mDeathState = CharState_SwimDeath ;
2013-12-31 11:24:20 +00:00
}
2014-09-02 13:14:23 +00:00
else if ( mHitState = = CharState_KnockDown & & mAnimation - > hasAnimation ( " deathknockdown " ) )
2014-03-26 17:55:16 +00:00
{
mDeathState = CharState_DeathKnockDown ;
}
2014-09-02 13:14:23 +00:00
else if ( mHitState = = CharState_KnockOut & & mAnimation - > hasAnimation ( " deathknockout " ) )
2014-03-26 17:55:16 +00:00
{
mDeathState = CharState_DeathKnockOut ;
}
2013-12-31 11:24:20 +00:00
else
{
2016-05-19 19:37:24 +00:00
mDeathState = chooseRandomDeathState ( ) ;
2013-12-31 11:24:20 +00:00
}
2014-05-26 17:56:32 +00:00
playDeath ( startpoint , mDeathState ) ;
2013-12-31 11:24:20 +00:00
}
2013-07-25 04:08:16 +00:00
2013-07-16 06:43:33 +00:00
CharacterController : : CharacterController ( const MWWorld : : Ptr & ptr , MWRender : : Animation * anim )
2013-04-29 20:07:49 +00:00
: mPtr ( ptr )
, mAnimation ( anim )
2013-07-18 07:35:03 +00:00
, mIdleState ( CharState_None )
2013-07-16 05:56:23 +00:00
, mMovementState ( CharState_None )
2016-10-05 14:32:26 +00:00
, mMovementAnimSpeed ( 0.f )
2015-11-04 19:34:50 +00:00
, mAdjustMovementAnimSpeed ( false )
2014-09-06 19:31:48 +00:00
, mHasMovedInXY ( false )
2014-07-03 15:40:44 +00:00
, mMovementAnimationControlled ( true )
2013-07-16 06:43:33 +00:00
, mDeathState ( CharState_None )
2016-02-01 22:13:43 +00:00
, mFloatToSurface ( true )
2013-12-31 11:24:20 +00:00
, mHitState ( CharState_None )
2013-07-13 21:24:52 +00:00
, mUpperBodyState ( UpperCharState_Nothing )
2013-08-19 06:42:56 +00:00
, mJumpState ( JumpState_None )
2013-05-11 05:22:39 +00:00
, mWeaponType ( WeapType_None )
2015-06-26 03:15:07 +00:00
, mAttackStrength ( 0.f )
2013-04-29 20:07:49 +00:00
, mSkipAnim ( false )
, mSecondsOfSwimming ( 0 )
2015-05-01 00:24:27 +00:00
, mSecondsOfRunning ( 0 )
2014-09-17 03:20:10 +00:00
, mTurnAnimationThreshold ( 0 )
2015-07-02 17:14:28 +00:00
, mAttackingOrSpell ( false )
2013-01-16 18:45:18 +00:00
{
2013-01-20 05:55:04 +00:00
if ( ! mAnimation )
2013-01-16 18:45:18 +00:00
return ;
2015-05-21 22:55:43 +00:00
mAnimation - > setTextKeyListener ( this ) ;
2014-05-22 18:37:22 +00:00
const MWWorld : : Class & cls = mPtr . getClass ( ) ;
2013-07-25 04:08:16 +00:00
if ( cls . isActor ( ) )
2013-02-05 20:55:06 +00:00
{
/* Accumulate along X/Y only for now, until we can figure out how we should
* handle knockout and death which moves the character down . */
2015-04-24 23:20:07 +00:00
mAnimation - > setAccumulation ( osg : : Vec3f ( 1.0f , 1.0f , 0.0f ) ) ;
2013-07-16 06:43:33 +00:00
2014-01-19 12:31:17 +00:00
if ( cls . hasInventoryStore ( mPtr ) )
2013-07-25 04:08:16 +00:00
{
2014-01-19 12:31:17 +00:00
getActiveWeapon ( cls . getCreatureStats ( mPtr ) , cls . getInventoryStore ( mPtr ) , & mWeaponType ) ;
2014-07-03 18:28:52 +00:00
if ( mWeaponType ! = WeapType_None )
2013-07-25 04:08:16 +00:00
{
mUpperBodyState = UpperCharState_WeapEquiped ;
2014-07-03 18:28:52 +00:00
getWeaponGroup ( mWeaponType , mCurrentWeapon ) ;
}
if ( mWeaponType ! = WeapType_None & & mWeaponType ! = WeapType_Spell & & mWeaponType ! = WeapType_HandToHand )
{
2014-01-15 20:56:55 +00:00
mAnimation - > showWeapons ( true ) ;
2014-06-15 23:58:13 +00:00
mAnimation - > setWeaponGroup ( mCurrentWeapon ) ;
2013-07-25 04:08:16 +00:00
}
2014-12-12 15:49:22 +00:00
mAnimation - > showCarriedLeft ( updateCarriedLeftVisible ( mWeaponType ) ) ;
2013-07-25 04:08:16 +00:00
}
if ( ! cls . getCreatureStats ( mPtr ) . isDead ( ) )
2013-07-18 07:35:03 +00:00
mIdleState = CharState_Idle ;
else
2013-07-16 06:43:33 +00:00
{
2016-06-11 22:04:50 +00:00
const MWMechanics : : CreatureStats & cStats = mPtr . getClass ( ) . getCreatureStats ( mPtr ) ;
if ( cStats . isDeathAnimationFinished ( ) )
{
// Set the death state, but don't play it yet
// We will play it in the first frame, but only if no script set the skipAnim flag
signed char deathanim = cStats . getDeathAnimation ( ) ;
if ( deathanim = = - 1 )
mDeathState = chooseRandomDeathState ( ) ;
else
mDeathState = static_cast < CharacterState > ( CharState_Death1 + deathanim ) ;
2016-05-19 19:37:24 +00:00
2016-06-11 22:04:50 +00:00
mFloatToSurface = false ;
}
// else: nothing to do, will detect death in the next frame and start playing death animation
2013-07-16 06:43:33 +00:00
}
2013-02-05 20:55:06 +00:00
}
2013-03-31 23:12:02 +00:00
else
{
/* Don't accumulate with non-actors. */
2015-04-24 23:20:07 +00:00
mAnimation - > setAccumulation ( osg : : Vec3f ( 0.f , 0.f , 0.f ) ) ;
2013-07-18 07:35:03 +00:00
mIdleState = CharState_Idle ;
2013-03-31 23:12:02 +00:00
}
2013-05-01 02:26:41 +00:00
2014-01-02 19:54:41 +00:00
2014-06-15 23:46:10 +00:00
if ( mDeathState = = CharState_None )
2015-07-16 18:03:16 +00:00
refreshCurrentAnims ( mIdleState , mMovementState , mJumpState , true ) ;
2014-07-13 08:35:11 +00:00
mAnimation - > runAnimation ( 0.f ) ;
2016-07-30 17:24:03 +00:00
unpersistAnimationState ( ) ;
2013-01-16 18:45:18 +00:00
}
2013-02-04 15:10:14 +00:00
CharacterController : : ~ CharacterController ( )
{
2015-05-21 22:55:43 +00:00
if ( mAnimation )
2016-07-30 17:24:03 +00:00
{
persistAnimationState ( ) ;
2015-05-21 22:55:43 +00:00
mAnimation - > setTextKeyListener ( NULL ) ;
2016-07-30 17:24:03 +00:00
}
2013-02-04 15:10:14 +00:00
}
2015-05-21 22:55:43 +00:00
void split ( const std : : string & s , char delim , std : : vector < std : : string > & elems ) {
std : : stringstream ss ( s ) ;
std : : string item ;
while ( std : : getline ( ss , item , delim ) ) {
elems . push_back ( item ) ;
}
}
void CharacterController : : handleTextKey ( const std : : string & groupname , const std : : multimap < float , std : : string > : : const_iterator & key , const std : : multimap < float , std : : string > & map )
{
const std : : string & evt = key - > second ;
if ( evt . compare ( 0 , 7 , " sound: " ) = = 0 )
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
2016-03-24 17:31:37 +00:00
sndMgr - > stopSound3D ( mPtr , evt . substr ( 7 ) ) ;
2015-05-21 22:55:43 +00:00
sndMgr - > playSound3D ( mPtr , evt . substr ( 7 ) , 1.0f , 1.0f ) ;
return ;
}
if ( evt . compare ( 0 , 10 , " soundgen: " ) = = 0 )
{
std : : string soundgen = evt . substr ( 10 ) ;
// The event can optionally contain volume and pitch modifiers
float volume = 1.f , pitch = 1.f ;
if ( soundgen . find ( " " ) ! = std : : string : : npos )
{
std : : vector < std : : string > tokens ;
split ( soundgen , ' ' , tokens ) ;
soundgen = tokens [ 0 ] ;
if ( tokens . size ( ) > = 2 )
2015-06-03 17:59:54 +00:00
{
std : : stringstream stream ;
stream < < tokens [ 1 ] ;
stream > > volume ;
}
2015-05-21 22:55:43 +00:00
if ( tokens . size ( ) > = 3 )
2015-06-03 17:59:54 +00:00
{
std : : stringstream stream ;
stream < < tokens [ 2 ] ;
stream > > pitch ;
}
2015-05-21 22:55:43 +00:00
}
std : : string sound = mPtr . getClass ( ) . getSoundIdFromSndGen ( mPtr , soundgen ) ;
if ( ! sound . empty ( ) )
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
if ( evt . compare ( 10 , evt . size ( ) - 10 , " left " ) = = 0 | | evt . compare ( 10 , evt . size ( ) - 10 , " right " ) = = 0 | | evt . compare ( 10 , evt . size ( ) - 10 , " land " ) = = 0 )
2015-11-27 09:02:53 +00:00
{
// Don't make foot sounds local for the player, it makes sense to keep them
// positioned on the ground.
sndMgr - > playSound3D ( mPtr , sound , volume , pitch , MWBase : : SoundManager : : Play_TypeFoot ,
MWBase : : SoundManager : : Play_NoPlayerLocal ) ;
}
else
2016-03-24 17:31:37 +00:00
{
sndMgr - > stopSound3D ( mPtr , sound ) ;
2015-11-27 09:02:53 +00:00
sndMgr - > playSound3D ( mPtr , sound , volume , pitch ) ;
2016-03-24 17:31:37 +00:00
}
2015-05-21 22:55:43 +00:00
}
return ;
}
if ( evt . compare ( 0 , groupname . size ( ) , groupname ) ! = 0 | |
evt . compare ( groupname . size ( ) , 2 , " : " ) ! = 0 )
{
// Not ours, skip it
return ;
}
size_t off = groupname . size ( ) + 2 ;
size_t len = evt . size ( ) - off ;
if ( evt . compare ( off , len , " equip attach " ) = = 0 )
mAnimation - > showWeapons ( true ) ;
else if ( evt . compare ( off , len , " unequip detach " ) = = 0 )
mAnimation - > showWeapons ( false ) ;
else if ( evt . compare ( off , len , " chop hit " ) = = 0 )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Chop ) ;
2015-05-21 22:55:43 +00:00
else if ( evt . compare ( off , len , " slash hit " ) = = 0 )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Slash ) ;
2015-05-21 22:55:43 +00:00
else if ( evt . compare ( off , len , " thrust hit " ) = = 0 )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Thrust ) ;
2015-05-21 22:55:43 +00:00
else if ( evt . compare ( off , len , " hit " ) = = 0 )
{
if ( groupname = = " attack1 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Chop ) ;
2015-05-21 22:55:43 +00:00
else if ( groupname = = " attack2 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Slash ) ;
2015-05-21 22:55:43 +00:00
else if ( groupname = = " attack3 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Thrust ) ;
2015-05-21 22:55:43 +00:00
else
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength ) ;
2015-05-21 22:55:43 +00:00
}
else if ( ! groupname . empty ( ) & & groupname . compare ( 0 , groupname . size ( ) - 1 , " attack " ) = = 0
& & evt . compare ( off , len , " start " ) = = 0 )
{
std : : multimap < float , std : : string > : : const_iterator hitKey = key ;
// Not all animations have a hit key defined. If there is none, the hit happens with the start key.
bool hasHitKey = false ;
while ( hitKey ! = map . end ( ) )
{
if ( hitKey - > second = = groupname + " : hit " )
{
hasHitKey = true ;
break ;
}
if ( hitKey - > second = = groupname + " : stop " )
break ;
+ + hitKey ;
}
if ( ! hasHitKey )
{
if ( groupname = = " attack1 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Chop ) ;
2015-05-21 22:55:43 +00:00
else if ( groupname = = " attack2 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Slash ) ;
2015-05-21 22:55:43 +00:00
else if ( groupname = = " attack3 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Thrust ) ;
2015-05-21 22:55:43 +00:00
}
}
else if ( evt . compare ( off , len , " shoot attach " ) = = 0 )
mAnimation - > attachArrow ( ) ;
else if ( evt . compare ( off , len , " shoot release " ) = = 0 )
2015-06-26 03:15:07 +00:00
mAnimation - > releaseArrow ( mAttackStrength ) ;
2015-05-21 22:55:43 +00:00
else if ( evt . compare ( off , len , " shoot follow attach " ) = = 0 )
mAnimation - > attachArrow ( ) ;
2015-05-29 23:00:24 +00:00
else if ( groupname = = " spellcast " & & evt . substr ( evt . size ( ) - 7 , 7 ) = = " release "
// Make sure this key is actually for the RangeType we are casting. The flame atronach has
// the same animation for all range types, so there are 3 "release" keys on the same time, one for each range type.
& & evt . compare ( off , len , mAttackType + " release " ) = = 0 )
2015-05-21 22:55:43 +00:00
{
2015-05-29 23:00:24 +00:00
MWBase : : Environment : : get ( ) . getWorld ( ) - > castSpell ( mPtr ) ;
2015-05-21 22:55:43 +00:00
}
else if ( groupname = = " shield " & & evt . compare ( off , len , " block hit " ) = = 0 )
mPtr . getClass ( ) . block ( mPtr ) ;
}
2013-01-16 21:09:21 +00:00
2013-02-25 17:57:34 +00:00
void CharacterController : : updatePtr ( const MWWorld : : Ptr & ptr )
{
mPtr = ptr ;
}
2015-11-03 16:48:35 +00:00
void CharacterController : : updateIdleStormState ( bool inwater )
2014-06-24 16:37:38 +00:00
{
bool inStormDirection = false ;
if ( MWBase : : Environment : : get ( ) . getWorld ( ) - > isInStorm ( ) )
{
2015-05-31 23:57:15 +00:00
osg : : Vec3f stormDirection = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStormDirection ( ) ;
osg : : Vec3f characterDirection = mPtr . getRefData ( ) . getBaseNode ( ) - > getAttitude ( ) * osg : : Vec3f ( 0 , 1 , 0 ) ;
2015-06-05 01:41:10 +00:00
inStormDirection = std : : acos ( stormDirection * characterDirection / ( stormDirection . length ( ) * characterDirection . length ( ) ) )
> osg : : DegreesToRadians ( 120.f ) ;
2014-06-24 16:37:38 +00:00
}
2015-11-03 16:48:35 +00:00
if ( inStormDirection & & ! inwater & & mUpperBodyState = = UpperCharState_Nothing & & mAnimation - > hasAnimation ( " idlestorm " ) )
2014-06-24 16:37:38 +00:00
{
float complete = 0 ;
mAnimation - > getInfo ( " idlestorm " , & complete ) ;
if ( complete = = 0 )
2015-07-09 16:47:11 +00:00
mAnimation - > play ( " idlestorm " , Priority_Storm , MWRender : : Animation : : BlendMask_RightArm , false ,
2014-06-24 16:37:38 +00:00
1.0f , " start " , " loop start " , 0.0f , 0 ) ;
else if ( complete = = 1 )
2015-07-09 16:47:11 +00:00
mAnimation - > play ( " idlestorm " , Priority_Storm , MWRender : : Animation : : BlendMask_RightArm , false ,
2014-06-24 16:37:38 +00:00
1.0f , " loop start " , " loop stop " , 0.0f , ~ 0ul ) ;
}
else
{
if ( mUpperBodyState = = UpperCharState_Nothing )
{
if ( mAnimation - > isPlaying ( " idlestorm " ) )
{
if ( mAnimation - > getCurrentTime ( " idlestorm " ) < mAnimation - > getTextKeyTime ( " idlestorm: loop stop " ) )
{
2015-07-09 16:47:11 +00:00
mAnimation - > play ( " idlestorm " , Priority_Storm , MWRender : : Animation : : BlendMask_RightArm , true ,
2014-06-24 16:37:38 +00:00
1.0f , " loop stop " , " stop " , 0.0f , 0 ) ;
}
}
}
else
mAnimation - > disable ( " idlestorm " ) ;
}
}
2014-01-17 15:31:27 +00:00
bool CharacterController : : updateCreatureState ( )
2013-07-23 10:26:24 +00:00
{
2014-01-17 15:31:27 +00:00
const MWWorld : : Class & cls = mPtr . getClass ( ) ;
CreatureStats & stats = cls . getCreatureStats ( mPtr ) ;
2014-09-14 22:29:21 +00:00
WeaponType weapType = WeapType_None ;
if ( stats . getDrawState ( ) = = DrawState_Weapon )
weapType = WeapType_HandToHand ;
else if ( stats . getDrawState ( ) = = DrawState_Spell )
weapType = WeapType_Spell ;
if ( weapType ! = mWeaponType )
{
mWeaponType = weapType ;
if ( mAnimation - > isPlaying ( mCurrentWeapon ) )
mAnimation - > disable ( mCurrentWeapon ) ;
}
2015-07-02 17:14:28 +00:00
if ( mAttackingOrSpell )
2014-01-17 15:31:27 +00:00
{
if ( mUpperBodyState = = UpperCharState_Nothing & & mHitState = = CharState_None )
{
MWBase : : Environment : : get ( ) . getWorld ( ) - > breakInvisibility ( mPtr ) ;
2014-09-18 01:24:47 +00:00
std : : string startKey = " start " ;
std : : string stopKey = " stop " ;
2014-09-14 22:29:21 +00:00
if ( weapType = = WeapType_Spell )
{
const std : : string spellid = stats . getSpells ( ) . getSelectedSpell ( ) ;
if ( ! spellid . empty ( ) & & MWBase : : Environment : : get ( ) . getWorld ( ) - > startSpellCast ( mPtr ) )
{
2016-07-16 13:47:33 +00:00
MWMechanics : : CastSpell cast ( mPtr , NULL ) ;
cast . playSpellCastingEffects ( spellid ) ;
2014-09-18 01:24:47 +00:00
if ( ! mAnimation - > hasAnimation ( " spellcast " ) )
MWBase : : Environment : : get ( ) . getWorld ( ) - > castSpell ( mPtr ) ; // No "release" text key to use, so cast immediately
else
{
const ESM : : Spell * spell = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : Spell > ( ) . find ( spellid ) ;
const ESM : : ENAMstruct & effectentry = spell - > mEffects . mList . at ( 0 ) ;
switch ( effectentry . mRange )
{
case 0 : mAttackType = " self " ; break ;
case 1 : mAttackType = " touch " ; break ;
case 2 : mAttackType = " target " ; break ;
}
startKey = mAttackType + " " + startKey ;
stopKey = mAttackType + " " + stopKey ;
mCurrentWeapon = " spellcast " ;
}
2014-09-14 22:29:21 +00:00
}
2014-09-18 01:24:47 +00:00
else
mCurrentWeapon = " " ;
}
if ( weapType ! = WeapType_Spell | | ! mAnimation - > hasAnimation ( " spellcast " ) ) // Not all creatures have a dedicated spellcast animation
{
2015-04-24 23:20:07 +00:00
int roll = Misc : : Rng : : rollDice ( 3 ) ; // [0, 2]
2014-09-18 01:24:47 +00:00
if ( roll = = 0 )
mCurrentWeapon = " attack1 " ;
else if ( roll = = 1 )
mCurrentWeapon = " attack2 " ;
else
mCurrentWeapon = " attack3 " ;
}
if ( ! mCurrentWeapon . empty ( ) )
{
mAnimation - > play ( mCurrentWeapon , Priority_Weapon ,
2015-07-09 16:47:11 +00:00
MWRender : : Animation : : BlendMask_All , true ,
2014-09-18 01:24:47 +00:00
1 , startKey , stopKey ,
0.0f , 0 ) ;
mUpperBodyState = UpperCharState_StartToMinAttack ;
2015-06-26 02:21:10 +00:00
2015-06-26 03:15:07 +00:00
mAttackStrength = std : : min ( 1.f , 0.1f + Misc : : Rng : : rollClosedProbability ( ) ) ;
2016-12-11 18:35:53 +00:00
if ( weapType = = WeapType_HandToHand )
playSwishSound ( 0.0f ) ;
2014-09-14 22:29:21 +00:00
}
2014-01-17 15:31:27 +00:00
}
2014-09-14 22:29:21 +00:00
2015-07-02 17:14:28 +00:00
mAttackingOrSpell = false ;
2014-01-17 15:31:27 +00:00
}
bool animPlaying = mAnimation - > getInfo ( mCurrentWeapon ) ;
if ( ! animPlaying )
mUpperBodyState = UpperCharState_Nothing ;
return false ;
}
2014-12-12 15:49:22 +00:00
bool CharacterController : : updateCarriedLeftVisible ( WeaponType weaptype ) const
{
// Shields/torches shouldn't be visible during any operation involving two hands
// There seems to be no text keys for this purpose, except maybe for "[un]equip start/stop",
// but they are also present in weapon drawing animation.
switch ( weaptype )
{
case WeapType_Spell :
case WeapType_BowAndArrow :
case WeapType_Crossbow :
case WeapType_HandToHand :
case WeapType_TwoHand :
case WeapType_TwoWide :
return false ;
default :
return true ;
}
}
2014-01-27 20:38:01 +00:00
bool CharacterController : : updateWeaponState ( )
2014-01-17 15:31:27 +00:00
{
2014-05-22 18:37:22 +00:00
const MWWorld : : Class & cls = mPtr . getClass ( ) ;
2014-01-19 12:31:17 +00:00
CreatureStats & stats = cls . getCreatureStats ( mPtr ) ;
2013-07-23 10:26:24 +00:00
WeaponType weaptype = WeapType_None ;
2014-12-31 15:59:21 +00:00
if ( stats . getDrawState ( ) = = DrawState_Weapon )
weaptype = WeapType_HandToHand ;
else if ( stats . getDrawState ( ) = = DrawState_Spell )
weaptype = WeapType_Spell ;
2014-01-19 12:31:17 +00:00
const bool isWerewolf = cls . isNpc ( ) & & cls . getNpcStats ( mPtr ) . isWerewolf ( ) ;
2013-07-23 10:26:24 +00:00
2014-12-31 15:59:21 +00:00
std : : string soundid ;
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) )
{
MWWorld : : InventoryStore & inv = cls . getInventoryStore ( mPtr ) ;
2017-02-27 21:50:10 +00:00
MWWorld : : ConstContainerStoreIterator weapon = getActiveWeapon ( stats , inv , & weaptype ) ;
2014-12-31 15:59:21 +00:00
if ( weapon ! = inv . end ( ) & & ! ( weaptype = = WeapType_None & & mWeaponType = = WeapType_Spell ) )
{
soundid = ( weaptype = = WeapType_None ) ?
weapon - > getClass ( ) . getDownSoundId ( * weapon ) :
weapon - > getClass ( ) . getUpSoundId ( * weapon ) ;
}
}
2015-07-15 12:40:36 +00:00
MWRender : : Animation : : AnimPriority priorityWeapon ( Priority_Weapon ) ;
2015-09-16 13:43:42 +00:00
priorityWeapon [ MWRender : : Animation : : BoneGroup_LowerBody ] = Priority_WeaponLowerBody ;
2015-07-15 12:40:36 +00:00
2013-07-23 10:26:24 +00:00
bool forcestateupdate = false ;
2015-01-04 18:50:46 +00:00
if ( weaptype ! = mWeaponType & & mHitState ! = CharState_KnockDown & & mHitState ! = CharState_KnockOut
& & mHitState ! = CharState_Hit )
2013-07-23 10:26:24 +00:00
{
forcestateupdate = true ;
2014-12-12 15:49:22 +00:00
mAnimation - > showCarriedLeft ( updateCarriedLeftVisible ( weaptype ) ) ;
2013-11-23 19:24:52 +00:00
2013-07-23 10:26:24 +00:00
std : : string weapgroup ;
if ( weaptype = = WeapType_None )
{
2015-12-26 17:45:09 +00:00
if ( ( ! isWerewolf | | mWeaponType ! = WeapType_Spell ) )
{
getWeaponGroup ( mWeaponType , weapgroup ) ;
mAnimation - > play ( weapgroup , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , true ,
1.0f , " unequip start " , " unequip stop " , 0.0f , 0 ) ;
mUpperBodyState = UpperCharState_UnEquipingWeap ;
}
2013-07-23 10:26:24 +00:00
}
else
{
getWeaponGroup ( weaptype , weapgroup ) ;
mAnimation - > showWeapons ( false ) ;
2014-02-04 02:46:15 +00:00
mAnimation - > setWeaponGroup ( weapgroup ) ;
2014-02-23 19:11:05 +00:00
2015-07-15 12:40:36 +00:00
mAnimation - > play ( weapgroup , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , true ,
2013-07-23 10:26:24 +00:00
1.0f , " equip start " , " equip stop " , 0.0f , 0 ) ;
mUpperBodyState = UpperCharState_EquipingWeap ;
2013-11-23 19:24:52 +00:00
2013-08-09 13:40:16 +00:00
if ( isWerewolf )
{
2013-08-11 07:35:19 +00:00
const MWWorld : : ESMStore & store = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
const ESM : : Sound * sound = store . get < ESM : : Sound > ( ) . searchRandom ( " WolfEquip " ) ;
if ( sound )
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
sndMgr - > playSound3D ( mPtr , sound - > mId , 1.0f , 1.0f ) ;
}
2013-08-09 13:40:16 +00:00
}
2013-07-23 10:26:24 +00:00
}
2014-12-31 15:59:21 +00:00
if ( ! soundid . empty ( ) )
2013-07-23 10:26:24 +00:00
{
2014-12-31 15:59:21 +00:00
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
sndMgr - > playSound3D ( mPtr , soundid , 1.0f , 1.0f ) ;
2013-07-23 10:26:24 +00:00
}
2013-07-25 07:46:20 +00:00
mWeaponType = weaptype ;
getWeaponGroup ( mWeaponType , mCurrentWeapon ) ;
2013-07-23 10:26:24 +00:00
}
2013-08-09 13:40:16 +00:00
if ( isWerewolf )
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
2014-01-23 21:14:20 +00:00
if ( cls . getCreatureStats ( mPtr ) . getStance ( MWMechanics : : CreatureStats : : Stance_Run )
2014-09-06 03:52:47 +00:00
& & mHasMovedInXY
2014-01-23 21:14:20 +00:00
& & ! MWBase : : Environment : : get ( ) . getWorld ( ) - > isSwimming ( mPtr )
& & mWeaponType = = WeapType_None )
2013-08-09 13:40:16 +00:00
{
if ( ! sndMgr - > getSoundPlaying ( mPtr , " WolfRun " ) )
sndMgr - > playSound3D ( mPtr , " WolfRun " , 1.0f , 1.0f , MWBase : : SoundManager : : Play_TypeSfx ,
MWBase : : SoundManager : : Play_Loop ) ;
}
else
sndMgr - > stopSound3D ( mPtr , " WolfRun " ) ;
}
2013-07-23 10:26:24 +00:00
2014-02-04 03:05:01 +00:00
// Cancel attack if we no longer have ammunition
bool ammunition = true ;
2014-12-31 15:59:21 +00:00
bool isWeapon = false ;
float weapSpeed = 1.f ;
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) )
2014-02-04 03:05:01 +00:00
{
2014-12-31 15:59:21 +00:00
MWWorld : : InventoryStore & inv = cls . getInventoryStore ( mPtr ) ;
2017-02-27 21:50:10 +00:00
MWWorld : : ConstContainerStoreIterator weapon = getActiveWeapon ( stats , inv , & weaptype ) ;
2014-12-31 15:59:21 +00:00
isWeapon = ( weapon ! = inv . end ( ) & & weapon - > getTypeName ( ) = = typeid ( ESM : : Weapon ) . name ( ) ) ;
if ( isWeapon )
weapSpeed = weapon - > get < ESM : : Weapon > ( ) - > mBase - > mData . mSpeed ;
2017-02-28 14:31:51 +00:00
MWWorld : : ConstContainerStoreIterator ammo = inv . getSlot ( MWWorld : : InventoryStore : : Slot_Ammunition ) ;
2014-12-31 15:59:21 +00:00
if ( mWeaponType = = WeapType_Crossbow )
ammunition = ( ammo ! = inv . end ( ) & & ammo - > get < ESM : : Weapon > ( ) - > mBase - > mData . mType = = ESM : : Weapon : : Bolt ) ;
else if ( mWeaponType = = WeapType_BowAndArrow )
ammunition = ( ammo ! = inv . end ( ) & & ammo - > get < ESM : : Weapon > ( ) - > mBase - > mData . mType = = ESM : : Weapon : : Arrow ) ;
if ( ! ammunition & & mUpperBodyState > UpperCharState_WeapEquiped )
{
mAnimation - > disable ( mCurrentWeapon ) ;
mUpperBodyState = UpperCharState_WeapEquiped ;
}
2014-02-04 03:05:01 +00:00
}
2013-07-23 10:26:24 +00:00
float complete ;
2013-07-23 18:12:19 +00:00
bool animPlaying ;
2015-07-02 17:14:28 +00:00
if ( mAttackingOrSpell )
2013-07-23 10:26:24 +00:00
{
2016-08-25 13:19:49 +00:00
mIdleState = CharState_None ;
2015-07-15 12:54:37 +00:00
if ( mUpperBodyState = = UpperCharState_WeapEquiped & & ( mHitState = = CharState_None | | mHitState = = CharState_Block ) )
2013-07-23 10:26:24 +00:00
{
2013-12-08 22:36:37 +00:00
MWBase : : Environment : : get ( ) . getWorld ( ) - > breakInvisibility ( mPtr ) ;
2013-07-23 14:30:54 +00:00
if ( mWeaponType = = WeapType_Spell )
2013-07-23 13:13:08 +00:00
{
2013-12-26 21:06:13 +00:00
// Unset casting flag, otherwise pressing the mouse button down would
// continue casting every frame if there is no animation
2015-07-02 17:14:28 +00:00
mAttackingOrSpell = false ;
2015-08-21 09:12:39 +00:00
if ( mPtr = = getPlayer ( ) )
2015-07-16 17:56:09 +00:00
{
MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . setAttackingOrSpell ( false ) ;
}
2013-12-26 21:06:13 +00:00
2013-07-23 18:12:19 +00:00
const MWWorld : : ESMStore & store = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
2013-12-26 21:32:39 +00:00
// For the player, set the spell we want to cast
// This has to be done at the start of the casting animation,
// *not* when selecting a spell in the GUI (otherwise you could change the spell mid-animation)
2015-08-21 09:12:39 +00:00
if ( mPtr = = getPlayer ( ) )
2014-01-17 12:13:58 +00:00
{
std : : string selectedSpell = MWBase : : Environment : : get ( ) . getWindowManager ( ) - > getSelectedSpell ( ) ;
stats . getSpells ( ) . setSelectedSpell ( selectedSpell ) ;
}
2013-12-26 21:32:39 +00:00
std : : string spellid = stats . getSpells ( ) . getSelectedSpell ( ) ;
2013-12-26 21:06:13 +00:00
if ( ! spellid . empty ( ) & & MWBase : : Environment : : get ( ) . getWorld ( ) - > startSpellCast ( mPtr ) )
2013-07-23 14:30:54 +00:00
{
2016-07-16 13:47:33 +00:00
MWMechanics : : CastSpell cast ( mPtr , NULL ) ;
cast . playSpellCastingEffects ( spellid ) ;
2013-07-23 18:12:19 +00:00
2016-09-12 10:20:33 +00:00
const ESM : : Spell * spell = store . get < ESM : : Spell > ( ) . find ( spellid ) ;
const ESM : : ENAMstruct & lastEffect = spell - > mEffects . mList . back ( ) ;
2013-07-24 12:39:15 +00:00
const ESM : : MagicEffect * effect ;
2016-09-02 11:10:13 +00:00
effect = store . get < ESM : : MagicEffect > ( ) . find ( lastEffect . mEffectID ) ; // use last effect of list for color of VFX_Hands
2013-07-24 12:39:15 +00:00
2014-09-14 22:29:21 +00:00
const ESM : : Static * castStatic = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : Static > ( ) . find ( " VFX_Hands " ) ;
2014-12-31 15:59:21 +00:00
2016-09-02 15:16:24 +00:00
for ( size_t iter = 0 ; iter < spell - > mEffects . mList . size ( ) ; + + iter ) // play hands vfx for each effect
2016-09-02 11:10:13 +00:00
{
if ( mAnimation - > getNode ( " Bip01 L Hand " ) )
mAnimation - > addEffect ( " meshes \\ " + castStatic - > mModel , - 1 , false , " Bip01 L Hand " , effect - > mParticle ) ;
if ( mAnimation - > getNode ( " Bip01 R Hand " ) )
mAnimation - > addEffect ( " meshes \\ " + castStatic - > mModel , - 1 , false , " Bip01 R Hand " , effect - > mParticle ) ;
}
const ESM : : ENAMstruct & firstEffect = spell - > mEffects . mList . at ( 0 ) ; // first effect used for casting animation
2013-11-17 22:15:57 +00:00
2016-09-02 11:10:13 +00:00
switch ( firstEffect . mRange )
2014-01-26 21:32:57 +00:00
{
case 0 : mAttackType = " self " ; break ;
case 1 : mAttackType = " touch " ; break ;
case 2 : mAttackType = " target " ; break ;
}
2013-07-23 14:30:54 +00:00
2015-07-15 12:40:36 +00:00
mAnimation - > play ( mCurrentWeapon , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , true ,
2013-07-23 14:30:54 +00:00
weapSpeed , mAttackType + " start " , mAttackType + " stop " ,
0.0f , 0 ) ;
mUpperBodyState = UpperCharState_CastingSpell ;
}
2014-12-31 15:59:21 +00:00
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) )
2013-11-09 06:51:46 +00:00
{
2014-12-31 15:59:21 +00:00
MWWorld : : InventoryStore & inv = mPtr . getClass ( ) . getInventoryStore ( mPtr ) ;
if ( inv . getSelectedEnchantItem ( ) ! = inv . end ( ) )
{
// Enchanted items cast immediately (no animation)
MWBase : : Environment : : get ( ) . getWorld ( ) - > castSpell ( mPtr ) ;
}
2013-11-09 06:51:46 +00:00
}
2014-12-31 15:59:21 +00:00
2013-07-23 13:13:08 +00:00
}
2013-08-03 01:21:42 +00:00
else if ( mWeaponType = = WeapType_PickProbe )
{
2014-12-31 15:59:21 +00:00
MWWorld : : ContainerStoreIterator weapon = mPtr . getClass ( ) . getInventoryStore ( mPtr ) . getSlot ( MWWorld : : InventoryStore : : Slot_CarriedRight ) ;
2013-08-03 01:21:42 +00:00
MWWorld : : Ptr item = * weapon ;
2014-09-09 03:05:07 +00:00
// TODO: this will only work for the player, and needs to be fixed if NPCs should ever use lockpicks/probes.
2013-08-03 01:21:42 +00:00
MWWorld : : Ptr target = MWBase : : Environment : : get ( ) . getWorld ( ) - > getFacedObject ( ) ;
std : : string resultMessage , resultSound ;
if ( ! target . isEmpty ( ) )
{
if ( item . getTypeName ( ) = = typeid ( ESM : : Lockpick ) . name ( ) )
Security ( mPtr ) . pickLock ( target , item , resultMessage , resultSound ) ;
else if ( item . getTypeName ( ) = = typeid ( ESM : : Probe ) . name ( ) )
Security ( mPtr ) . probeTrap ( target , item , resultMessage , resultSound ) ;
}
2015-07-15 12:40:36 +00:00
mAnimation - > play ( mCurrentWeapon , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , true ,
2013-08-03 01:21:42 +00:00
1.0f , " start " , " stop " , 0.0 , 0 ) ;
mUpperBodyState = UpperCharState_FollowStartToFollowStop ;
if ( ! resultMessage . empty ( ) )
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > messageBox ( resultMessage ) ;
if ( ! resultSound . empty ( ) )
2016-07-10 13:08:42 +00:00
MWBase : : Environment : : get ( ) . getSoundManager ( ) - > playSound3D ( target ,
resultSound , 1.0f , 1.0f , MWBase : : SoundManager : : Play_TypeSfx ,
MWBase : : SoundManager : : Play_Normal ) ;
2013-08-03 01:21:42 +00:00
}
2014-02-04 03:05:01 +00:00
else if ( ammunition )
2013-07-23 14:30:54 +00:00
{
if ( mWeaponType = = WeapType_Crossbow | | mWeaponType = = WeapType_BowAndArrow | |
2014-02-04 02:55:40 +00:00
mWeaponType = = WeapType_Thrown )
2013-07-23 14:30:54 +00:00
mAttackType = " shoot " ;
else
{
2016-09-19 17:13:10 +00:00
if ( mPtr = = getPlayer ( ) )
2014-12-31 15:59:21 +00:00
{
2016-09-19 17:13:10 +00:00
if ( isWeapon )
2016-03-19 17:03:59 +00:00
{
2016-09-01 13:43:33 +00:00
if ( Settings : : Manager : : getBool ( " best attack " , " Game " ) )
{
2017-02-27 21:50:10 +00:00
MWWorld : : ConstContainerStoreIterator weapon = mPtr . getClass ( ) . getInventoryStore ( mPtr ) . getSlot ( MWWorld : : InventoryStore : : Slot_CarriedRight ) ;
2016-09-01 13:43:33 +00:00
mAttackType = getBestAttack ( weapon - > get < ESM : : Weapon > ( ) - > mBase ) ;
}
else
setAttackTypeBasedOnMovement ( ) ;
2016-03-19 17:03:59 +00:00
}
2016-09-19 17:13:10 +00:00
else
setAttackTypeRandomly ( mAttackType ) ;
2014-12-31 15:59:21 +00:00
}
2016-09-19 17:13:10 +00:00
// else if (mPtr != getPlayer()) use mAttackType set by AiCombat
2014-01-02 19:54:41 +00:00
}
2013-07-23 10:26:24 +00:00
2015-07-15 12:40:36 +00:00
mAnimation - > play ( mCurrentWeapon , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , false ,
2013-07-23 14:30:54 +00:00
weapSpeed , mAttackType + " start " , mAttackType + " min attack " ,
0.0f , 0 ) ;
2014-02-23 19:11:05 +00:00
mUpperBodyState = UpperCharState_StartToMinAttack ;
2013-07-23 14:30:54 +00:00
}
2013-07-23 10:26:24 +00:00
}
2014-01-06 20:00:01 +00:00
2013-07-23 18:12:19 +00:00
animPlaying = mAnimation - > getInfo ( mCurrentWeapon , & complete ) ;
2015-07-03 03:58:12 +00:00
if ( mUpperBodyState = = UpperCharState_MinAttackToMaxAttack & & mHitState ! = CharState_KnockDown )
mAttackStrength = complete ;
2013-07-23 10:26:24 +00:00
}
2013-07-23 18:12:19 +00:00
else
2013-07-23 10:26:24 +00:00
{
2013-07-23 18:12:19 +00:00
animPlaying = mAnimation - > getInfo ( mCurrentWeapon , & complete ) ;
2014-01-08 14:05:14 +00:00
if ( mUpperBodyState = = UpperCharState_MinAttackToMaxAttack & & mHitState ! = CharState_KnockDown )
2013-07-23 18:12:19 +00:00
{
2014-09-07 02:46:47 +00:00
float attackStrength = complete ;
if ( ! mPtr . getClass ( ) . isNpc ( ) )
{
// most creatures don't actually have an attack wind-up animation, so use a uniform random value
// (even some creatures that can use weapons don't have a wind-up animation either, e.g. Rieklings)
// Note: vanilla MW uses a random value for *all* non-player actors, but we probably don't need to go that far.
2015-04-24 23:20:07 +00:00
attackStrength = std : : min ( 1.f , 0.1f + Misc : : Rng : : rollClosedProbability ( ) ) ;
2014-09-07 02:46:47 +00:00
}
2015-01-31 14:37:16 +00:00
if ( mWeaponType ! = WeapType_Crossbow & & mWeaponType ! = WeapType_BowAndArrow )
2013-07-24 17:34:53 +00:00
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
2013-08-11 07:35:19 +00:00
if ( isWerewolf )
{
const MWWorld : : ESMStore & store = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
const ESM : : Sound * sound = store . get < ESM : : Sound > ( ) . searchRandom ( " WolfSwing " ) ;
if ( sound )
sndMgr - > playSound3D ( mPtr , sound - > mId , 1.0f , 1.0f ) ;
}
2013-08-02 07:20:12 +00:00
else
2013-08-11 07:35:19 +00:00
{
2016-12-11 18:35:53 +00:00
playSwishSound ( attackStrength ) ;
2013-08-11 07:35:19 +00:00
}
2013-07-24 17:34:53 +00:00
}
2015-06-26 03:15:07 +00:00
mAttackStrength = attackStrength ;
2014-01-08 14:05:14 +00:00
2013-07-23 18:12:19 +00:00
mAnimation - > disable ( mCurrentWeapon ) ;
2015-07-15 12:40:36 +00:00
mAnimation - > play ( mCurrentWeapon , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , false ,
2013-07-23 18:12:19 +00:00
weapSpeed , mAttackType + " max attack " , mAttackType + " min hit " ,
1.0f - complete , 0 ) ;
2014-06-06 19:15:23 +00:00
complete = 0.f ;
2013-07-23 18:12:19 +00:00
mUpperBodyState = UpperCharState_MaxAttackToMinHit ;
}
2014-01-08 14:05:14 +00:00
else if ( mHitState = = CharState_KnockDown )
{
2014-12-05 21:02:18 +00:00
if ( mUpperBodyState > UpperCharState_WeapEquiped )
mUpperBodyState = UpperCharState_WeapEquiped ;
2014-01-08 14:05:14 +00:00
mAnimation - > disable ( mCurrentWeapon ) ;
}
2013-07-23 10:26:24 +00:00
}
2014-02-04 03:00:52 +00:00
mAnimation - > setPitchFactor ( 0.f ) ;
if ( mWeaponType = = WeapType_BowAndArrow | | mWeaponType = = WeapType_Thrown )
{
switch ( mUpperBodyState )
{
case UpperCharState_StartToMinAttack :
mAnimation - > setPitchFactor ( complete ) ;
break ;
case UpperCharState_MinAttackToMaxAttack :
case UpperCharState_MaxAttackToMinHit :
case UpperCharState_MinHitToHit :
mAnimation - > setPitchFactor ( 1.f ) ;
break ;
case UpperCharState_FollowStartToFollowStop :
if ( animPlaying )
mAnimation - > setPitchFactor ( 1.f - complete ) ;
break ;
default :
break ;
}
}
else if ( mWeaponType = = WeapType_Crossbow )
{
switch ( mUpperBodyState )
{
case UpperCharState_EquipingWeap :
mAnimation - > setPitchFactor ( complete ) ;
break ;
case UpperCharState_UnEquipingWeap :
mAnimation - > setPitchFactor ( 1.f - complete ) ;
break ;
case UpperCharState_WeapEquiped :
case UpperCharState_StartToMinAttack :
case UpperCharState_MinAttackToMaxAttack :
case UpperCharState_MaxAttackToMinHit :
case UpperCharState_MinHitToHit :
case UpperCharState_FollowStartToFollowStop :
mAnimation - > setPitchFactor ( 1.f ) ;
break ;
default :
break ;
}
}
2013-07-23 10:26:24 +00:00
if ( ! animPlaying )
{
if ( mUpperBodyState = = UpperCharState_EquipingWeap | |
2013-07-23 14:30:54 +00:00
mUpperBodyState = = UpperCharState_FollowStartToFollowStop | |
mUpperBodyState = = UpperCharState_CastingSpell )
2013-12-31 15:55:04 +00:00
{
2014-02-04 03:11:46 +00:00
if ( ammunition & & mWeaponType = = WeapType_Crossbow )
mAnimation - > attachArrow ( ) ;
2013-07-23 10:26:24 +00:00
mUpperBodyState = UpperCharState_WeapEquiped ;
2013-12-31 15:55:04 +00:00
}
2013-07-23 10:26:24 +00:00
else if ( mUpperBodyState = = UpperCharState_UnEquipingWeap )
2014-01-01 19:40:31 +00:00
mUpperBodyState = UpperCharState_Nothing ;
2013-07-23 10:26:24 +00:00
}
else if ( complete > = 1.0f )
{
2014-01-01 19:40:31 +00:00
std : : string start , stop ;
switch ( mUpperBodyState )
2013-07-23 10:26:24 +00:00
{
2014-01-01 19:40:31 +00:00
case UpperCharState_StartToMinAttack :
start = mAttackType + " min attack " ;
stop = mAttackType + " max attack " ;
mUpperBodyState = UpperCharState_MinAttackToMaxAttack ;
break ;
2014-02-02 14:29:51 +00:00
case UpperCharState_MinAttackToMaxAttack :
//hack to avoid body pos desync when jumping/sneaking in 'max attack' state
if ( ! mAnimation - > isPlaying ( mCurrentWeapon ) )
2015-07-15 12:40:36 +00:00
mAnimation - > play ( mCurrentWeapon , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , false ,
2014-02-02 14:29:51 +00:00
0 , mAttackType + " min attack " , mAttackType + " max attack " , 0.999f , 0 ) ;
break ;
2014-01-01 19:40:31 +00:00
case UpperCharState_MaxAttackToMinHit :
if ( mAttackType = = " shoot " )
{
start = mAttackType + " min hit " ;
stop = mAttackType + " release " ;
}
else
{
start = mAttackType + " min hit " ;
stop = mAttackType + " hit " ;
}
mUpperBodyState = UpperCharState_MinHitToHit ;
break ;
case UpperCharState_MinHitToHit :
if ( mAttackType = = " shoot " )
{
start = mAttackType + " follow start " ;
stop = mAttackType + " follow stop " ;
}
else
{
2015-06-26 03:15:07 +00:00
float str = mAttackStrength ;
2014-01-01 19:40:31 +00:00
start = mAttackType + ( ( str < 0.5f ) ? " small follow start "
: ( str < 1.0f ) ? " medium follow start "
: " large follow start " ) ;
stop = mAttackType + ( ( str < 0.5f ) ? " small follow stop "
: ( str < 1.0f ) ? " medium follow stop "
: " large follow stop " ) ;
}
mUpperBodyState = UpperCharState_FollowStartToFollowStop ;
break ;
default :
break ;
2013-07-23 10:26:24 +00:00
}
2014-01-01 19:40:31 +00:00
if ( ! start . empty ( ) )
2013-07-23 10:26:24 +00:00
{
mAnimation - > disable ( mCurrentWeapon ) ;
2014-01-01 19:40:31 +00:00
if ( mUpperBodyState = = UpperCharState_FollowStartToFollowStop )
2015-07-15 12:40:36 +00:00
mAnimation - > play ( mCurrentWeapon , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , true ,
2014-01-01 19:40:31 +00:00
weapSpeed , start , stop , 0.0f , 0 ) ;
2013-07-23 13:13:08 +00:00
else
2015-07-15 12:40:36 +00:00
mAnimation - > play ( mCurrentWeapon , priorityWeapon ,
MWRender : : Animation : : BlendMask_All , false ,
2013-08-02 07:20:12 +00:00
weapSpeed , start , stop , 0.0f , 0 ) ;
2013-07-23 10:26:24 +00:00
}
}
2014-02-23 19:11:05 +00:00
2014-12-31 15:59:21 +00:00
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) )
2013-07-23 10:26:24 +00:00
{
2017-02-27 21:50:10 +00:00
const MWWorld : : InventoryStore & inv = mPtr . getClass ( ) . getInventoryStore ( mPtr ) ;
MWWorld : : ConstContainerStoreIterator torch = inv . getSlot ( MWWorld : : InventoryStore : : Slot_CarriedLeft ) ;
2014-12-31 15:59:21 +00:00
if ( torch ! = inv . end ( ) & & torch - > getTypeName ( ) = = typeid ( ESM : : Light ) . name ( )
& & updateCarriedLeftVisible ( mWeaponType ) )
{
2015-07-09 16:47:11 +00:00
mAnimation - > play ( " torch " , Priority_Torch , MWRender : : Animation : : BlendMask_LeftArm ,
2014-12-31 15:59:21 +00:00
false , 1.0f , " start " , " stop " , 0.0f , ( ~ ( size_t ) 0 ) , true ) ;
}
else if ( mAnimation - > isPlaying ( " torch " ) )
{
mAnimation - > disable ( " torch " ) ;
}
2013-07-23 10:26:24 +00:00
}
2015-11-10 00:01:41 +00:00
mAnimation - > setAccurateAiming ( mUpperBodyState > UpperCharState_WeapEquiped ) ;
2013-07-23 10:26:24 +00:00
return forcestateupdate ;
}
2016-08-22 21:02:57 +00:00
void CharacterController : : updateAnimQueue ( )
{
if ( mAnimQueue . size ( ) > 1 )
{
if ( mAnimation - > isPlaying ( mAnimQueue . front ( ) . mGroup ) = = false )
{
mAnimation - > disable ( mAnimQueue . front ( ) . mGroup ) ;
mAnimQueue . pop_front ( ) ;
bool loopfallback = ( mAnimQueue . front ( ) . mGroup . compare ( 0 , 4 , " idle " ) = = 0 ) ;
mAnimation - > play ( mAnimQueue . front ( ) . mGroup , Priority_Default ,
MWRender : : Animation : : BlendMask_All , false ,
1.0f , " start " , " stop " , 0.0f , mAnimQueue . front ( ) . mLoopCount , loopfallback ) ;
}
}
2016-08-22 21:03:26 +00:00
if ( ! mAnimQueue . empty ( ) )
mAnimation - > setLoopingEnabled ( mAnimQueue . front ( ) . mGroup , mAnimQueue . size ( ) < = 1 ) ;
2016-08-22 21:02:57 +00:00
}
2013-08-18 05:34:38 +00:00
void CharacterController : : update ( float duration )
2013-01-19 05:40:47 +00:00
{
2013-08-18 05:34:38 +00:00
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
2014-05-22 18:37:22 +00:00
const MWWorld : : Class & cls = mPtr . getClass ( ) ;
2015-05-12 01:02:15 +00:00
osg : : Vec3f movement ( 0.f , 0.f , 0.f ) ;
2015-07-25 16:22:48 +00:00
float speed = 0.f ;
2013-05-14 14:29:07 +00:00
2016-08-06 18:08:46 +00:00
updateMagicEffects ( ) ;
2013-05-14 14:29:07 +00:00
if ( ! cls . isActor ( ) )
2016-08-22 21:02:57 +00:00
updateAnimQueue ( ) ;
2013-05-14 14:29:07 +00:00
else if ( ! cls . getCreatureStats ( mPtr ) . isDead ( ) )
2013-02-18 14:29:16 +00:00
{
2013-02-21 04:08:04 +00:00
bool onground = world - > isOnGround ( mPtr ) ;
2013-02-18 14:29:16 +00:00
bool inwater = world - > isSwimming ( mPtr ) ;
2014-01-15 06:47:21 +00:00
bool sneak = cls . getCreatureStats ( mPtr ) . getStance ( MWMechanics : : CreatureStats : : Stance_Sneak ) ;
2013-08-18 12:59:06 +00:00
bool flying = world - > isFlying ( mPtr ) ;
2014-10-14 23:06:16 +00:00
// Can't run while flying (see speed formula in Npc/Creature::getSpeed)
bool isrunning = cls . getCreatureStats ( mPtr ) . getStance ( MWMechanics : : CreatureStats : : Stance_Run ) & & ! flying ;
2014-08-03 05:42:40 +00:00
CreatureStats & stats = cls . getCreatureStats ( mPtr ) ;
//Force Jump Logic
2015-01-27 00:40:36 +00:00
bool isMoving = ( std : : abs ( cls . getMovementSettings ( mPtr ) . mPosition [ 0 ] ) > .5 | | std : : abs ( cls . getMovementSettings ( mPtr ) . mPosition [ 1 ] ) > .5 ) ;
2014-08-03 05:42:40 +00:00
if ( ! inwater & & ! flying )
{
//Force Jump
if ( stats . getMovementFlag ( MWMechanics : : CreatureStats : : Flag_ForceJump ) )
{
if ( onground )
{
cls . getMovementSettings ( mPtr ) . mPosition [ 2 ] = 1 ;
}
else
cls . getMovementSettings ( mPtr ) . mPosition [ 2 ] = 0 ;
}
//Force Move Jump, only jump if they're otherwise moving
if ( stats . getMovementFlag ( MWMechanics : : CreatureStats : : Flag_ForceMoveJump ) & & isMoving )
{
if ( onground )
{
cls . getMovementSettings ( mPtr ) . mPosition [ 2 ] = 1 ;
}
else
cls . getMovementSettings ( mPtr ) . mPosition [ 2 ] = 0 ;
}
}
2015-05-12 01:02:15 +00:00
osg : : Vec3f vec ( cls . getMovementSettings ( mPtr ) . asVec3 ( ) ) ;
vec . normalize ( ) ;
2014-05-04 14:15:07 +00:00
2013-12-31 11:24:20 +00:00
if ( mHitState ! = CharState_None & & mJumpState = = JumpState_None )
2015-05-12 01:02:15 +00:00
vec = osg : : Vec3f ( 0.f , 0.f , 0.f ) ;
2015-06-03 17:41:19 +00:00
osg : : Vec3f rot = cls . getRotationVector ( mPtr ) ;
2014-04-25 20:20:55 +00:00
2015-07-25 16:22:48 +00:00
speed = cls . getSpeed ( mPtr ) ;
2013-02-06 03:05:07 +00:00
2015-07-25 16:22:48 +00:00
vec . x ( ) * = speed ;
vec . y ( ) * = speed ;
2013-04-28 05:53:04 +00:00
2013-08-18 12:59:06 +00:00
CharacterState movestate = CharState_None ;
CharacterState idlestate = CharState_SpecialIdle ;
2015-07-16 18:03:16 +00:00
JumpingState jumpstate = JumpState_None ;
2013-08-18 12:59:06 +00:00
bool forcestateupdate = false ;
2013-02-21 04:08:04 +00:00
2015-05-12 01:02:15 +00:00
mHasMovedInXY = std : : abs ( vec . x ( ) ) + std : : abs ( vec . y ( ) ) > 0.0f ;
2014-09-06 03:52:47 +00:00
isrunning = isrunning & & mHasMovedInXY ;
2013-07-16 05:56:23 +00:00
2013-12-31 11:24:20 +00:00
2013-08-09 13:53:07 +00:00
// advance athletics
2015-08-21 09:12:39 +00:00
if ( mHasMovedInXY & & mPtr = = getPlayer ( ) )
2013-08-09 13:53:07 +00:00
{
if ( inwater )
{
mSecondsOfSwimming + = duration ;
while ( mSecondsOfSwimming > 1 )
{
cls . skillUsageSucceeded ( mPtr , ESM : : Skill : : Athletics , 1 ) ;
mSecondsOfSwimming - = 1 ;
}
}
else if ( isrunning )
{
mSecondsOfRunning + = duration ;
while ( mSecondsOfRunning > 1 )
{
cls . skillUsageSucceeded ( mPtr , ESM : : Skill : : Athletics , 0 ) ;
mSecondsOfRunning - = 1 ;
}
}
}
2013-07-16 05:56:23 +00:00
2014-01-11 21:26:26 +00:00
// reduce fatigue
const MWWorld : : Store < ESM : : GameSetting > & gmst = world - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
float fatigueLoss = 0 ;
static const float fFatigueRunBase = gmst . find ( " fFatigueRunBase " ) - > getFloat ( ) ;
static const float fFatigueRunMult = gmst . find ( " fFatigueRunMult " ) - > getFloat ( ) ;
static const float fFatigueSwimWalkBase = gmst . find ( " fFatigueSwimWalkBase " ) - > getFloat ( ) ;
static const float fFatigueSwimRunBase = gmst . find ( " fFatigueSwimRunBase " ) - > getFloat ( ) ;
static const float fFatigueSwimWalkMult = gmst . find ( " fFatigueSwimWalkMult " ) - > getFloat ( ) ;
static const float fFatigueSwimRunMult = gmst . find ( " fFatigueSwimRunMult " ) - > getFloat ( ) ;
static const float fFatigueSneakBase = gmst . find ( " fFatigueSneakBase " ) - > getFloat ( ) ;
static const float fFatigueSneakMult = gmst . find ( " fFatigueSneakMult " ) - > getFloat ( ) ;
const float encumbrance = cls . getEncumbrance ( mPtr ) / cls . getCapacity ( mPtr ) ;
if ( encumbrance < 1 )
{
if ( sneak )
fatigueLoss = fFatigueSneakBase + encumbrance * fFatigueSneakMult ;
else
{
if ( inwater )
{
if ( ! isrunning )
fatigueLoss = fFatigueSwimWalkBase + encumbrance * fFatigueSwimWalkMult ;
else
fatigueLoss = fFatigueSwimRunBase + encumbrance * fFatigueSwimRunMult ;
}
if ( isrunning )
fatigueLoss = fFatigueRunBase + encumbrance * fFatigueRunMult ;
}
}
fatigueLoss * = duration ;
DynamicStat < float > fatigue = cls . getCreatureStats ( mPtr ) . getFatigue ( ) ;
fatigue . setCurrent ( fatigue . getCurrent ( ) - fatigueLoss , fatigue . getCurrent ( ) < 0 ) ;
cls . getCreatureStats ( mPtr ) . setFatigue ( fatigue ) ;
2014-04-25 20:20:55 +00:00
if ( sneak | | inwater | | flying )
2015-05-12 01:02:15 +00:00
vec . z ( ) = 0.0f ;
2013-08-18 12:59:06 +00:00
2013-12-28 23:36:36 +00:00
if ( inwater | | flying )
cls . getCreatureStats ( mPtr ) . land ( ) ;
2014-07-03 15:40:44 +00:00
bool inJump = true ;
2013-08-18 12:59:06 +00:00
if ( ! onground & & ! flying & & ! inwater )
{
2013-11-16 01:11:11 +00:00
// In the air (either getting up —ascending part of jump— or falling).
2013-10-01 21:35:34 +00:00
2013-10-02 20:54:36 +00:00
if ( world - > isSlowFalling ( mPtr ) )
{
// SlowFalling spell effect is active, do not keep previous fall height
2013-12-27 20:21:18 +00:00
cls . getCreatureStats ( mPtr ) . land ( ) ;
2013-10-02 20:54:36 +00:00
}
2013-10-01 21:35:34 +00:00
2014-09-17 00:20:46 +00:00
forcestateupdate = ( mJumpState ! = JumpState_InAir ) ;
2015-07-16 18:03:16 +00:00
jumpstate = JumpState_InAir ;
2013-08-19 06:42:56 +00:00
2014-09-16 01:15:04 +00:00
static const float fJumpMoveBase = gmst . find ( " fJumpMoveBase " ) - > getFloat ( ) ;
2015-01-13 02:11:29 +00:00
static const float fJumpMoveMult = gmst . find ( " fJumpMoveMult " ) - > getFloat ( ) ;
float factor = fJumpMoveBase + fJumpMoveMult * mPtr . getClass ( ) . getSkill ( mPtr , ESM : : Skill : : Acrobatics ) / 100.f ;
factor = std : : min ( 1.f , factor ) ;
2015-05-12 01:02:15 +00:00
vec . x ( ) * = factor ;
vec . y ( ) * = factor ;
vec . z ( ) = 0.0f ;
2013-08-18 12:59:06 +00:00
}
2015-05-12 01:02:15 +00:00
else if ( vec . z ( ) > 0.0f & & mJumpState = = JumpState_None )
2013-08-18 12:59:06 +00:00
{
2013-11-16 01:11:11 +00:00
// Started a jump.
2014-09-16 01:15:04 +00:00
float z = cls . getJump ( mPtr ) ;
2014-12-02 17:42:13 +00:00
if ( z > 0 )
2014-09-16 01:15:04 +00:00
{
2015-05-12 01:02:15 +00:00
if ( vec . x ( ) = = 0 & & vec . y ( ) = = 0 )
vec = osg : : Vec3f ( 0.0f , 0.0f , z ) ;
2014-12-02 17:42:13 +00:00
else
{
2015-05-12 01:02:15 +00:00
osg : : Vec3f lat ( vec . x ( ) , vec . y ( ) , 0.0f ) ;
lat . normalize ( ) ;
vec = osg : : Vec3f ( lat . x ( ) , lat . y ( ) , 1.0f ) * z * 0.707f ;
2014-12-02 17:42:13 +00:00
}
2013-08-18 12:59:06 +00:00
2014-12-02 17:42:13 +00:00
// advance acrobatics
2015-08-21 09:12:39 +00:00
if ( mPtr = = getPlayer ( ) )
2014-12-02 17:42:13 +00:00
cls . skillUsageSucceeded ( mPtr , ESM : : Skill : : Acrobatics , 0 ) ;
// decrease fatigue
const float fatigueJumpBase = gmst . find ( " fFatigueJumpBase " ) - > getFloat ( ) ;
const float fatigueJumpMult = gmst . find ( " fFatigueJumpMult " ) - > getFloat ( ) ;
float normalizedEncumbrance = mPtr . getClass ( ) . getNormalizedEncumbrance ( mPtr ) ;
if ( normalizedEncumbrance > 1 )
normalizedEncumbrance = 1 ;
2015-03-08 04:42:07 +00:00
const float fatigueDecrease = fatigueJumpBase + ( 1 - normalizedEncumbrance ) * fatigueJumpMult ;
2014-12-02 17:42:13 +00:00
fatigue . setCurrent ( fatigue . getCurrent ( ) - fatigueDecrease ) ;
cls . getCreatureStats ( mPtr ) . setFatigue ( fatigue ) ;
}
2013-08-18 12:59:06 +00:00
}
2014-09-17 00:20:46 +00:00
else if ( mJumpState = = JumpState_InAir )
2013-02-18 14:29:16 +00:00
{
2013-08-19 15:10:18 +00:00
forcestateupdate = true ;
2015-07-16 18:03:16 +00:00
jumpstate = JumpState_Landing ;
2015-05-12 01:02:15 +00:00
vec . z ( ) = 0.0f ;
2013-10-01 21:35:34 +00:00
2013-12-27 20:21:18 +00:00
float height = cls . getCreatureStats ( mPtr ) . land ( ) ;
2014-12-10 16:21:34 +00:00
float healthLost = getFallDamage ( mPtr , height ) ;
2013-10-01 21:35:34 +00:00
if ( healthLost > 0.0f )
{
2013-10-02 11:02:14 +00:00
const float fatigueTerm = cls . getCreatureStats ( mPtr ) . getFatigueTerm ( ) ;
2013-10-01 21:35:34 +00:00
// inflict fall damages
DynamicStat < float > health = cls . getCreatureStats ( mPtr ) . getHealth ( ) ;
2015-03-08 04:42:07 +00:00
float realHealthLost = static_cast < float > ( healthLost * ( 1.0f - 0.25f * fatigueTerm ) ) ;
2013-10-02 10:58:34 +00:00
health . setCurrent ( health . getCurrent ( ) - realHealthLost ) ;
2013-10-01 21:35:34 +00:00
cls . getCreatureStats ( mPtr ) . setHealth ( health ) ;
2016-09-12 11:40:40 +00:00
cls . onHit ( mPtr , realHealthLost , true , MWWorld : : Ptr ( ) , MWWorld : : Ptr ( ) , osg : : Vec3f ( ) , true ) ;
2013-10-01 21:35:34 +00:00
2015-03-08 04:42:07 +00:00
const int acrobaticsSkill = cls . getSkill ( mPtr , ESM : : Skill : : Acrobatics ) ;
2013-10-02 11:02:14 +00:00
if ( healthLost > ( acrobaticsSkill * fatigueTerm ) )
2013-10-01 21:35:34 +00:00
{
2014-01-15 11:08:12 +00:00
cls . getCreatureStats ( mPtr ) . setKnockedDown ( true ) ;
}
else
{
// report acrobatics progression
2015-08-21 09:12:39 +00:00
if ( mPtr = = getPlayer ( ) )
2014-01-15 11:08:12 +00:00
cls . skillUsageSucceeded ( mPtr , ESM : : Skill : : Acrobatics , 1 ) ;
2013-10-01 21:35:34 +00:00
}
}
2013-02-18 14:29:16 +00:00
}
2013-08-19 06:42:56 +00:00
else
2013-03-31 10:50:20 +00:00
{
2015-07-16 18:03:16 +00:00
jumpstate = JumpState_None ;
2015-05-12 01:02:15 +00:00
vec . z ( ) = 0.0f ;
2013-08-19 06:42:56 +00:00
2014-07-03 15:40:44 +00:00
inJump = false ;
2015-05-12 01:02:15 +00:00
if ( std : : abs ( vec . x ( ) / 2.0f ) > std : : abs ( vec . y ( ) ) )
2013-08-19 06:42:56 +00:00
{
2015-05-12 01:02:15 +00:00
if ( vec . x ( ) > 0.0f )
2013-08-19 06:42:56 +00:00
movestate = ( inwater ? ( isrunning ? CharState_SwimRunRight : CharState_SwimWalkRight )
: ( sneak ? CharState_SneakRight
: ( isrunning ? CharState_RunRight : CharState_WalkRight ) ) ) ;
2015-05-12 01:02:15 +00:00
else if ( vec . x ( ) < 0.0f )
2013-08-19 06:42:56 +00:00
movestate = ( inwater ? ( isrunning ? CharState_SwimRunLeft : CharState_SwimWalkLeft )
: ( sneak ? CharState_SneakLeft
: ( isrunning ? CharState_RunLeft : CharState_WalkLeft ) ) ) ;
}
2015-05-12 01:02:15 +00:00
else if ( vec . y ( ) ! = 0.0f )
2013-08-19 06:42:56 +00:00
{
2015-05-12 01:02:15 +00:00
if ( vec . y ( ) > 0.0f )
2013-08-19 06:42:56 +00:00
movestate = ( inwater ? ( isrunning ? CharState_SwimRunForward : CharState_SwimWalkForward )
: ( sneak ? CharState_SneakForward
: ( isrunning ? CharState_RunForward : CharState_WalkForward ) ) ) ;
2015-05-12 01:02:15 +00:00
else if ( vec . y ( ) < 0.0f )
2013-08-19 06:42:56 +00:00
movestate = ( inwater ? ( isrunning ? CharState_SwimRunBack : CharState_SwimWalkBack )
: ( sneak ? CharState_SneakBack
: ( isrunning ? CharState_RunBack : CharState_WalkBack ) ) ) ;
}
2016-07-09 23:48:54 +00:00
else if ( rot . z ( ) ! = 0.0f & & ! inwater & & ! sneak & & ! ( mPtr = = getPlayer ( ) & & MWBase : : Environment : : get ( ) . getWorld ( ) - > isFirstPerson ( ) ) )
2013-08-19 06:42:56 +00:00
{
2015-06-03 17:41:19 +00:00
if ( rot . z ( ) > 0.0f )
2013-08-19 06:42:56 +00:00
movestate = CharState_TurnRight ;
2015-06-03 17:41:19 +00:00
else if ( rot . z ( ) < 0.0f )
2013-08-19 06:42:56 +00:00
movestate = CharState_TurnLeft ;
}
2013-03-31 10:50:20 +00:00
}
2013-07-16 05:56:23 +00:00
2014-09-17 03:20:10 +00:00
mTurnAnimationThreshold - = duration ;
if ( movestate = = CharState_TurnRight | | movestate = = CharState_TurnLeft )
2015-03-08 04:42:07 +00:00
mTurnAnimationThreshold = 0.05f ;
2014-09-17 03:20:10 +00:00
else if ( movestate = = CharState_None & & ( mMovementState = = CharState_TurnRight | | mMovementState = = CharState_TurnLeft )
& & mTurnAnimationThreshold > 0 )
{
movestate = mMovementState ;
}
2013-12-28 23:36:36 +00:00
if ( onground )
2013-12-27 20:21:18 +00:00
cls . getCreatureStats ( mPtr ) . land ( ) ;
2017-03-23 18:45:59 +00:00
if ( movestate ! = CharState_None & & movestate ! = CharState_TurnLeft & & movestate ! = CharState_TurnRight )
2013-07-16 05:56:23 +00:00
clearAnimQueue ( ) ;
2015-06-17 18:49:01 +00:00
if ( mAnimQueue . empty ( ) | | inwater | | sneak )
2014-12-28 14:34:47 +00:00
{
2015-09-16 13:37:36 +00:00
idlestate = ( inwater ? CharState_IdleSwim : ( sneak & & ! inJump ? CharState_IdleSneak : CharState_Idle ) ) ;
2014-12-28 14:34:47 +00:00
}
2016-08-22 21:02:57 +00:00
else
updateAnimQueue ( ) ;
2013-03-31 08:29:24 +00:00
2014-12-31 16:25:06 +00:00
if ( ! mSkipAnim )
2015-07-17 01:28:17 +00:00
{
// bipedal means hand-to-hand could be used (which is handled in updateWeaponState). an existing InventoryStore means an actual weapon could be used.
if ( cls . isBipedal ( mPtr ) | | cls . hasInventoryStore ( mPtr ) )
forcestateupdate = updateWeaponState ( ) | | forcestateupdate ;
else
forcestateupdate = updateCreatureState ( ) | | forcestateupdate ;
2015-07-16 18:03:16 +00:00
refreshCurrentAnims ( idlestate , movestate , jumpstate , forcestateupdate ) ;
2015-11-03 16:48:35 +00:00
updateIdleStormState ( inwater ) ;
2015-07-17 01:28:17 +00:00
}
2014-07-03 15:40:44 +00:00
if ( inJump )
mMovementAnimationControlled = false ;
2013-08-18 05:34:38 +00:00
2014-09-17 03:20:10 +00:00
if ( mMovementState = = CharState_TurnLeft | | mMovementState = = CharState_TurnRight )
{
if ( duration > 0 )
2015-06-03 19:37:21 +00:00
mAnimation - > adjustSpeedMult ( mCurrentMovement , std : : min ( 1.5f , std : : abs ( rot . z ( ) ) / duration / static_cast < float > ( osg : : PI ) ) ) ;
2014-09-17 03:20:10 +00:00
}
2015-07-25 16:22:48 +00:00
else if ( mMovementState ! = CharState_None & & mAdjustMovementAnimSpeed )
{
float speedmult = speed / mMovementAnimSpeed ;
mAnimation - > adjustSpeedMult ( mCurrentMovement , speedmult ) ;
}
2014-09-17 03:20:10 +00:00
2014-01-03 16:06:05 +00:00
if ( ! mSkipAnim )
2014-01-02 19:54:41 +00:00
{
2014-02-02 02:24:40 +00:00
if ( mHitState ! = CharState_KnockDown & & mHitState ! = CharState_KnockOut )
2014-01-04 15:55:09 +00:00
{
2017-02-04 16:07:35 +00:00
if ( rot ! = osg : : Vec3f ( ) )
world - > rotateObject ( mPtr , rot . x ( ) , rot . y ( ) , rot . z ( ) , true ) ;
2014-01-04 15:55:09 +00:00
}
2014-01-08 14:05:14 +00:00
else //avoid z-rotating for knockdown
2017-02-04 16:07:35 +00:00
{
if ( rot . x ( ) ! = 0 & & rot . y ( ) ! = 0 )
world - > rotateObject ( mPtr , rot . x ( ) , rot . y ( ) , 0.0f , true ) ;
}
2014-01-04 15:55:09 +00:00
2014-07-03 15:40:44 +00:00
if ( ! mMovementAnimationControlled )
2014-02-02 13:01:49 +00:00
world - > queueMovement ( mPtr , vec ) ;
2014-01-02 19:54:41 +00:00
}
2014-06-26 15:26:33 +00:00
else
// We must always queue movement, even if there is none, to apply gravity.
2015-05-12 01:02:15 +00:00
world - > queueMovement ( mPtr , osg : : Vec3f ( 0.f , 0.f , 0.f ) ) ;
2013-08-18 05:34:38 +00:00
movement = vec ;
2014-09-17 00:20:46 +00:00
cls . getMovementSettings ( mPtr ) . mPosition [ 0 ] = cls . getMovementSettings ( mPtr ) . mPosition [ 1 ] = 0 ;
// Can't reset jump state (mPosition[2]) here; we don't know for sure whether the PhysicSystem will actually handle it in this frame
// due to the fixed minimum timestep used for the physics update. It will be reset in PhysicSystem::move once the jump is handled.
2014-12-16 19:47:45 +00:00
2014-12-31 16:25:06 +00:00
if ( ! mSkipAnim )
updateHeadTracking ( duration ) ;
2013-02-03 15:15:34 +00:00
}
2013-07-16 05:56:23 +00:00
else if ( cls . getCreatureStats ( mPtr ) . isDead ( ) )
2013-06-27 21:11:20 +00:00
{
2015-09-17 02:30:55 +00:00
// initial start of death animation for actors that started the game as dead
// not done in constructor since we need to give scripts a chance to set the mSkipAnim flag
if ( ! mSkipAnim & & mDeathState ! = CharState_None & & mCurrentDeath . empty ( ) )
{
playDeath ( 1.f , mDeathState ) ;
}
2016-02-01 22:13:43 +00:00
// We must always queue movement, even if there is none, to apply gravity.
2015-05-12 01:02:15 +00:00
world - > queueMovement ( mPtr , osg : : Vec3f ( 0.f , 0.f , 0.f ) ) ;
2013-06-27 21:11:20 +00:00
}
2013-02-03 07:39:43 +00:00
2015-04-24 23:20:07 +00:00
osg : : Vec3f moved = mAnimation - > runAnimation ( mSkipAnim ? 0.f : duration ) ;
2014-12-31 16:25:06 +00:00
if ( duration > 0.0f )
moved / = duration ;
else
2015-04-24 23:20:07 +00:00
moved = osg : : Vec3f ( 0.f , 0.f , 0.f ) ;
2014-01-12 09:04:06 +00:00
2014-12-31 16:25:06 +00:00
// Ensure we're moving in generally the right direction...
2015-07-25 16:22:48 +00:00
if ( speed > 0.f )
2014-12-31 16:25:06 +00:00
{
float l = moved . length ( ) ;
2015-05-12 01:02:15 +00:00
if ( ( movement . x ( ) < 0.0f & & movement . x ( ) < moved . x ( ) * 2.0f ) | |
( movement . x ( ) > 0.0f & & movement . x ( ) > moved . x ( ) * 2.0f ) )
moved . x ( ) = movement . x ( ) ;
if ( ( movement . y ( ) < 0.0f & & movement . y ( ) < moved . y ( ) * 2.0f ) | |
( movement . y ( ) > 0.0f & & movement . y ( ) > moved . y ( ) * 2.0f ) )
moved . y ( ) = movement . y ( ) ;
if ( ( movement . z ( ) < 0.0f & & movement . z ( ) < moved . z ( ) * 2.0f ) | |
( movement . z ( ) > 0.0f & & movement . z ( ) > moved . z ( ) * 2.0f ) )
moved . z ( ) = movement . z ( ) ;
2014-12-31 16:25:06 +00:00
// but keep the original speed
float newLength = moved . length ( ) ;
if ( newLength > 0 )
moved * = ( l / newLength ) ;
2013-02-04 15:10:14 +00:00
}
2014-12-31 16:25:06 +00:00
if ( mSkipAnim )
2014-06-28 15:54:14 +00:00
mAnimation - > updateEffects ( duration ) ;
2014-12-31 16:25:06 +00:00
2016-06-15 17:38:04 +00:00
if ( mFloatToSurface & & cls . isActor ( ) & & cls . getCreatureStats ( mPtr ) . isDead ( ) & & cls . canSwim ( mPtr ) )
2016-02-01 22:13:43 +00:00
moved . z ( ) = 1.0 ;
2014-12-31 16:25:06 +00:00
// Update movement
if ( mMovementAnimationControlled & & mPtr . getClass ( ) . isActor ( ) )
2015-05-12 01:02:15 +00:00
world - > queueMovement ( mPtr , moved ) ;
2014-12-31 16:25:06 +00:00
2013-01-17 05:25:50 +00:00
mSkipAnim = false ;
2014-08-11 03:00:13 +00:00
mAnimation - > enableHeadAnimation ( cls . isActor ( ) & & ! cls . getCreatureStats ( mPtr ) . isDead ( ) ) ;
2013-01-17 00:31:09 +00:00
}
2016-07-30 17:24:03 +00:00
void CharacterController : : persistAnimationState ( )
{
ESM : : AnimationState & state = mPtr . getRefData ( ) . getAnimationState ( ) ;
2013-01-17 00:31:09 +00:00
2016-07-30 17:24:03 +00:00
state . mScriptedAnims . clear ( ) ;
for ( AnimationQueue : : const_iterator iter = mAnimQueue . begin ( ) ; iter ! = mAnimQueue . end ( ) ; + + iter )
{
if ( ! iter - > mPersist )
continue ;
ESM : : AnimationState : : ScriptedAnimation anim ;
anim . mGroup = iter - > mGroup ;
if ( iter = = mAnimQueue . begin ( ) )
{
anim . mLoopCount = mAnimation - > getCurrentLoopCount ( anim . mGroup ) ;
float complete ;
mAnimation - > getInfo ( anim . mGroup , & complete , NULL ) ;
anim . mTime = complete ;
}
else
{
anim . mLoopCount = iter - > mLoopCount ;
anim . mTime = 0.f ;
}
state . mScriptedAnims . push_back ( anim ) ;
}
}
void CharacterController : : unpersistAnimationState ( )
{
const ESM : : AnimationState & state = mPtr . getRefData ( ) . getAnimationState ( ) ;
if ( ! state . mScriptedAnims . empty ( ) )
{
clearAnimQueue ( ) ;
for ( ESM : : AnimationState : : ScriptedAnimations : : const_iterator iter = state . mScriptedAnims . begin ( ) ; iter ! = state . mScriptedAnims . end ( ) ; + + iter )
{
AnimationQueueEntry entry ;
entry . mGroup = iter - > mGroup ;
entry . mLoopCount = iter - > mLoopCount ;
entry . mPersist = true ;
mAnimQueue . push_back ( entry ) ;
}
const ESM : : AnimationState : : ScriptedAnimation & anim = state . mScriptedAnims . front ( ) ;
float complete = anim . mTime ;
if ( anim . mAbsolute )
{
float start = mAnimation - > getTextKeyTime ( anim . mGroup + " : start " ) ;
float stop = mAnimation - > getTextKeyTime ( anim . mGroup + " : stop " ) ;
float time = std : : max ( start , std : : min ( stop , anim . mTime ) ) ;
complete = ( time - start ) / ( stop - start ) ;
}
mAnimation - > disable ( mCurrentIdle ) ;
mCurrentIdle . clear ( ) ;
mIdleState = CharState_SpecialIdle ;
2016-08-23 10:50:56 +00:00
bool loopfallback = ( mAnimQueue . front ( ) . mGroup . compare ( 0 , 4 , " idle " ) = = 0 ) ;
2016-07-30 17:24:03 +00:00
mAnimation - > play ( anim . mGroup ,
Priority_Default , MWRender : : Animation : : BlendMask_All , false , 1.0f ,
2016-08-23 10:50:56 +00:00
" start " , " stop " , complete , anim . mLoopCount , loopfallback ) ;
2016-07-30 17:24:03 +00:00
}
}
bool CharacterController : : playGroup ( const std : : string & groupname , int mode , int count , bool persist )
2013-01-17 01:53:18 +00:00
{
2013-01-21 11:24:52 +00:00
if ( ! mAnimation | | ! mAnimation - > hasAnimation ( groupname ) )
2015-07-29 18:15:06 +00:00
{
2015-05-28 13:44:58 +00:00
std : : cerr < < " Animation " < < groupname < < " not found for " < < mPtr . getCellRef ( ) . getRefId ( ) < < std : : endl ;
2015-07-29 18:15:06 +00:00
return false ;
}
2013-01-21 11:24:52 +00:00
else
2013-01-17 21:18:40 +00:00
{
2016-08-17 16:08:54 +00:00
// If the given animation is a looped animation, is already playing
// and has not yet reached its Loop Stop key, make it the only animation
// in the queue, and retain the loop count from the animation that was
// already playing. This emulates observed behavior from the original
// engine and allows banners to animate correctly.
if ( ! mAnimQueue . empty ( ) & & mAnimQueue . front ( ) . mGroup = = groupname & &
mAnimation - > getTextKeyTime ( mAnimQueue . front ( ) . mGroup + " : loop start " ) > = 0 )
{
float endOfLoop = mAnimation - > getTextKeyTime ( mAnimQueue . front ( ) . mGroup + " : loop stop " ) ;
if ( endOfLoop < 0 ) // if no Loop Stop key was found, use the Stop key
endOfLoop = mAnimation - > getTextKeyTime ( mAnimQueue . front ( ) . mGroup + " : stop " ) ;
if ( endOfLoop > 0 & & ( mAnimation - > getCurrentTime ( mAnimQueue . front ( ) . mGroup ) < endOfLoop ) )
{
mAnimation - > setLoopingEnabled ( mAnimQueue . front ( ) . mGroup , true ) ;
mAnimQueue . resize ( 1 ) ;
return true ;
}
}
2016-08-17 13:48:55 +00:00
2013-01-17 21:18:40 +00:00
count = std : : max ( count , 1 ) ;
2016-07-30 17:24:03 +00:00
AnimationQueueEntry entry ;
entry . mGroup = groupname ;
entry . mLoopCount = count - 1 ;
entry . mPersist = persist ;
if ( mode ! = 0 | | mAnimQueue . empty ( ) | | ! isAnimPlaying ( mAnimQueue . front ( ) . mGroup ) )
2013-01-17 21:18:40 +00:00
{
2013-05-16 13:59:41 +00:00
clearAnimQueue ( ) ;
2016-07-30 17:24:03 +00:00
mAnimQueue . push_back ( entry ) ;
2013-05-12 12:08:01 +00:00
2013-07-16 05:56:23 +00:00
mAnimation - > disable ( mCurrentIdle ) ;
mCurrentIdle . clear ( ) ;
mIdleState = CharState_SpecialIdle ;
2016-08-15 16:15:26 +00:00
bool loopfallback = ( entry . mGroup . compare ( 0 , 4 , " idle " ) = = 0 ) ;
2013-05-13 07:54:44 +00:00
mAnimation - > play ( groupname , Priority_Default ,
2015-07-09 16:47:11 +00:00
MWRender : : Animation : : BlendMask_All , false , 1.0f ,
2016-08-15 16:15:26 +00:00
( ( mode = = 2 ) ? " loop start " : " start " ) , " stop " , 0.0f , count - 1 , loopfallback ) ;
2013-01-17 21:18:40 +00:00
}
else if ( mode = = 0 )
{
2013-05-12 12:08:01 +00:00
mAnimQueue . resize ( 1 ) ;
2016-07-30 17:24:03 +00:00
mAnimQueue . push_back ( entry ) ;
2013-01-17 21:18:40 +00:00
}
}
2015-07-29 18:15:06 +00:00
return true ;
2013-01-17 01:53:18 +00:00
}
void CharacterController : : skipAnim ( )
{
2013-01-17 05:25:50 +00:00
mSkipAnim = true ;
2013-01-17 01:53:18 +00:00
}
2013-05-25 03:10:07 +00:00
bool CharacterController : : isAnimPlaying ( const std : : string & groupName )
{
if ( mAnimation = = NULL )
return false ;
2013-07-16 05:56:23 +00:00
return mAnimation - > isPlaying ( groupName ) ;
2013-05-25 03:10:07 +00:00
}
2013-01-17 01:53:18 +00:00
2013-05-16 13:59:41 +00:00
void CharacterController : : clearAnimQueue ( )
{
2013-07-31 16:46:32 +00:00
if ( ! mAnimQueue . empty ( ) )
2016-07-30 17:24:03 +00:00
mAnimation - > disable ( mAnimQueue . front ( ) . mGroup ) ;
2013-05-16 13:59:41 +00:00
mAnimQueue . clear ( ) ;
}
2013-04-25 14:08:11 +00:00
void CharacterController : : forceStateUpdate ( )
{
2013-01-20 05:55:04 +00:00
if ( ! mAnimation )
2013-01-16 23:00:06 +00:00
return ;
2013-05-16 13:59:41 +00:00
clearAnimQueue ( ) ;
2013-01-19 22:56:24 +00:00
2015-07-16 18:03:16 +00:00
refreshCurrentAnims ( mIdleState , mMovementState , mJumpState , true ) ;
2013-07-16 06:43:33 +00:00
if ( mDeathState ! = CharState_None )
2013-07-16 05:56:23 +00:00
{
2014-01-02 19:54:41 +00:00
playRandomDeath ( ) ;
2013-07-16 05:56:23 +00:00
}
2014-12-31 16:25:06 +00:00
mAnimation - > runAnimation ( 0.f ) ;
2013-01-16 18:45:18 +00:00
}
2016-06-11 22:04:50 +00:00
CharacterController : : KillResult CharacterController : : kill ( )
2013-07-16 06:43:33 +00:00
{
2016-06-11 22:04:50 +00:00
if ( mDeathState = = CharState_None )
2013-12-15 16:50:25 +00:00
{
2016-06-11 22:04:50 +00:00
playRandomDeath ( ) ;
2013-07-17 09:19:22 +00:00
2016-06-11 22:04:50 +00:00
mAnimation - > disable ( mCurrentIdle ) ;
2013-12-15 16:50:25 +00:00
2016-06-11 22:04:50 +00:00
mIdleState = CharState_None ;
mCurrentIdle . clear ( ) ;
return Result_DeathAnimStarted ;
}
2014-09-14 20:29:06 +00:00
2016-06-11 22:04:50 +00:00
MWMechanics : : CreatureStats & cStats = mPtr . getClass ( ) . getCreatureStats ( mPtr ) ;
if ( isAnimPlaying ( mCurrentDeath ) )
return Result_DeathAnimPlaying ;
if ( ! cStats . isDeathAnimationFinished ( ) )
{
cStats . setDeathAnimationFinished ( true ) ;
return Result_DeathAnimJustFinished ;
}
return Result_DeathAnimFinished ;
2013-07-16 06:43:33 +00:00
}
void CharacterController : : resurrect ( )
{
if ( mDeathState = = CharState_None )
return ;
if ( mAnimation )
mAnimation - > disable ( mCurrentDeath ) ;
2013-12-06 06:36:16 +00:00
mCurrentDeath . clear ( ) ;
2013-07-16 06:43:33 +00:00
mDeathState = CharState_None ;
}
2013-11-13 14:44:43 +00:00
void CharacterController : : updateContinuousVfx ( )
{
// Keeping track of when to stop a continuous VFX seems to be very difficult to do inside the spells code,
// as it's extremely spread out (ActiveSpells, Spells, InventoryStore effects, etc...) so we do it here.
// Stop any effects that are no longer active
std : : vector < int > effects ;
mAnimation - > getLoopingEffects ( effects ) ;
for ( std : : vector < int > : : iterator it = effects . begin ( ) ; it ! = effects . end ( ) ; + + it )
{
2013-11-14 13:52:30 +00:00
if ( mPtr . getClass ( ) . getCreatureStats ( mPtr ) . isDead ( )
2014-08-16 20:38:22 +00:00
| | mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( MWMechanics : : EffectKey ( * it ) ) . getMagnitude ( ) < = 0 )
2013-11-13 14:44:43 +00:00
mAnimation - > removeEffect ( * it ) ;
}
}
2014-12-12 01:39:59 +00:00
void CharacterController : : updateMagicEffects ( )
2013-12-08 22:05:21 +00:00
{
if ( ! mPtr . getClass ( ) . isActor ( ) )
return ;
float alpha = 1.f ;
2016-09-18 22:05:50 +00:00
if ( mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Invisibility ) . getModifier ( ) ) // Ignore base magnitude (see bug #3555).
2013-12-08 22:05:21 +00:00
{
2015-08-21 09:12:39 +00:00
if ( mPtr = = getPlayer ( ) )
2013-12-08 22:05:21 +00:00
alpha = 0.4f ;
else
alpha = 0.f ;
}
2014-08-16 20:38:22 +00:00
float chameleon = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Chameleon ) . getMagnitude ( ) ;
2013-12-08 22:05:21 +00:00
if ( chameleon )
{
alpha * = std : : max ( 0.2f , ( 100.f - chameleon ) / 100.f ) ;
}
mAnimation - > setAlpha ( alpha ) ;
2014-08-23 23:50:29 +00:00
2014-12-12 01:39:59 +00:00
bool vampire = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Vampirism ) . getMagnitude ( ) > 0.0f ;
mAnimation - > setVampire ( vampire ) ;
2014-08-23 23:50:29 +00:00
float light = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Light ) . getMagnitude ( ) ;
mAnimation - > setLightEffect ( light ) ;
2013-12-08 22:05:21 +00:00
}
2016-03-19 17:03:59 +00:00
void CharacterController : : setAttackTypeBasedOnMovement ( )
2014-01-23 21:14:20 +00:00
{
2014-05-04 14:15:07 +00:00
float * move = mPtr . getClass ( ) . getMovementSettings ( mPtr ) . mPosition ;
2014-02-23 19:11:05 +00:00
2014-12-31 15:59:21 +00:00
if ( move [ 1 ] & & ! move [ 0 ] ) // forward-backward
mAttackType = " thrust " ;
else if ( move [ 0 ] & & ! move [ 1 ] ) //sideway
mAttackType = " slash " ;
else
mAttackType = " chop " ;
2013-01-10 16:35:24 +00:00
}
2014-01-23 21:14:20 +00:00
2014-12-12 15:49:22 +00:00
bool CharacterController : : isReadyToBlock ( ) const
{
return updateCarriedLeftVisible ( mWeaponType ) ;
}
2014-12-12 16:39:00 +00:00
bool CharacterController : : isKnockedOut ( ) const
{
return mHitState = = CharState_KnockOut ;
}
2015-09-16 13:37:36 +00:00
bool CharacterController : : isSneaking ( ) const
{
return mIdleState = = CharState_IdleSneak | |
mMovementState = = CharState_SneakForward | |
mMovementState = = CharState_SneakBack | |
mMovementState = = CharState_SneakLeft | |
mMovementState = = CharState_SneakRight ;
}
2015-07-02 17:14:28 +00:00
void CharacterController : : setAttackingOrSpell ( bool attackingOrSpell )
{
mAttackingOrSpell = attackingOrSpell ;
}
2016-09-01 13:43:33 +00:00
void CharacterController : : setAIAttackType ( std : : string attackType )
{
mAttackType = attackType ;
}
2016-09-19 17:13:10 +00:00
void CharacterController : : setAttackTypeRandomly ( std : : string & attackType )
{
float random = Misc : : Rng : : rollProbability ( ) ;
if ( random > = 2 / 3.f )
attackType = " thrust " ;
else if ( random > = 1 / 3.f )
attackType = " slash " ;
else
attackType = " chop " ;
}
2015-07-02 20:25:19 +00:00
bool CharacterController : : readyToPrepareAttack ( ) const
{
2015-07-15 12:54:37 +00:00
return ( mHitState = = CharState_None | | mHitState = = CharState_Block )
& & mUpperBodyState < = UpperCharState_WeapEquiped ;
2015-07-02 20:25:19 +00:00
}
2015-07-03 03:58:12 +00:00
bool CharacterController : : readyToStartAttack ( ) const
{
2015-07-15 12:54:37 +00:00
if ( mHitState ! = CharState_None & & mHitState ! = CharState_Block )
2015-07-03 03:58:12 +00:00
return false ;
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) | | mPtr . getClass ( ) . isBipedal ( mPtr ) )
return mUpperBodyState = = UpperCharState_WeapEquiped ;
else
return mUpperBodyState = = UpperCharState_Nothing ;
}
float CharacterController : : getAttackStrength ( ) const
{
return mAttackStrength ;
}
2015-04-29 21:48:08 +00:00
void CharacterController : : setActive ( bool active )
{
mAnimation - > setActive ( active ) ;
}
2015-12-19 15:02:47 +00:00
void CharacterController : : setHeadTrackTarget ( const MWWorld : : ConstPtr & target )
2014-12-16 19:47:45 +00:00
{
mHeadTrackTarget = target ;
}
2016-12-11 18:35:53 +00:00
void CharacterController : : playSwishSound ( float attackStrength )
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
std : : string sound = " Weapon Swish " ;
if ( attackStrength < 0.5f )
sndMgr - > playSound3D ( mPtr , sound , 1.0f , 0.8f ) ; //Weak attack
else if ( attackStrength < 1.0f )
sndMgr - > playSound3D ( mPtr , sound , 1.0f , 1.0f ) ; //Medium attack
else
sndMgr - > playSound3D ( mPtr , sound , 1.0f , 1.2f ) ; //Strong attack
}
2014-12-16 19:47:45 +00:00
void CharacterController : : updateHeadTracking ( float duration )
{
2015-05-31 16:04:14 +00:00
const osg : : Node * head = mAnimation - > getNode ( " Bip01 Head " ) ;
2014-12-16 19:47:45 +00:00
if ( ! head )
return ;
2015-05-31 16:04:14 +00:00
float zAngleRadians = 0.f ;
float xAngleRadians = 0.f ;
2014-12-16 19:47:45 +00:00
if ( ! mHeadTrackTarget . isEmpty ( ) )
{
2016-02-22 17:58:19 +00:00
osg : : NodePathList nodepaths = head - > getParentalNodePaths ( ) ;
if ( nodepaths . empty ( ) )
2015-05-31 16:04:14 +00:00
return ;
2016-02-22 17:58:19 +00:00
osg : : Matrixf mat = osg : : computeLocalToWorld ( nodepaths [ 0 ] ) ;
2015-05-31 16:04:14 +00:00
osg : : Vec3f headPos = mat . getTrans ( ) ;
2015-11-03 17:27:15 +00:00
osg : : Vec3f direction ;
2015-12-19 15:02:47 +00:00
if ( const MWRender : : Animation * anim = MWBase : : Environment : : get ( ) . getWorld ( ) - > getAnimation ( mHeadTrackTarget ) )
2014-12-16 19:47:45 +00:00
{
2015-05-31 16:04:14 +00:00
const osg : : Node * node = anim - > getNode ( " Head " ) ;
if ( node = = NULL )
node = anim - > getNode ( " Bip01 Head " ) ;
if ( node ! = NULL )
{
2016-10-02 08:48:54 +00:00
nodepaths = node - > getParentalNodePaths ( ) ;
2016-02-22 18:06:12 +00:00
if ( ! nodepaths . empty ( ) )
2016-02-22 17:58:19 +00:00
direction = osg : : computeLocalToWorld ( nodepaths [ 0 ] ) . getTrans ( ) - headPos ;
2015-05-31 16:04:14 +00:00
}
2015-11-03 17:27:15 +00:00
else
// no head node to look at, fall back to look at center of collision box
direction = MWBase : : Environment : : get ( ) . getWorld ( ) - > aimToTarget ( mPtr , mHeadTrackTarget ) ;
2014-12-16 19:47:45 +00:00
}
2015-05-31 16:04:14 +00:00
direction . normalize ( ) ;
if ( ! mPtr . getRefData ( ) . getBaseNode ( ) )
return ;
const osg : : Vec3f actorDirection = mPtr . getRefData ( ) . getBaseNode ( ) - > getAttitude ( ) * osg : : Vec3f ( 0 , 1 , 0 ) ;
2014-12-16 19:47:45 +00:00
2015-05-31 16:04:14 +00:00
zAngleRadians = std : : atan2 ( direction . x ( ) , direction . y ( ) ) - std : : atan2 ( actorDirection . x ( ) , actorDirection . y ( ) ) ;
xAngleRadians = - std : : asin ( direction . z ( ) ) ;
2014-12-16 19:47:45 +00:00
2015-05-31 16:04:14 +00:00
wrap ( zAngleRadians ) ;
wrap ( xAngleRadians ) ;
2014-12-16 19:47:45 +00:00
2015-05-31 16:04:14 +00:00
xAngleRadians = std : : min ( xAngleRadians , osg : : DegreesToRadians ( 40.f ) ) ;
xAngleRadians = std : : max ( xAngleRadians , osg : : DegreesToRadians ( - 40.f ) ) ;
zAngleRadians = std : : min ( zAngleRadians , osg : : DegreesToRadians ( 30.f ) ) ;
zAngleRadians = std : : max ( zAngleRadians , osg : : DegreesToRadians ( - 30.f ) ) ;
2014-12-16 19:47:45 +00:00
}
2015-05-31 16:04:14 +00:00
2014-12-16 19:47:45 +00:00
float factor = duration * 5 ;
factor = std : : min ( factor , 1.f ) ;
2015-05-31 16:04:14 +00:00
xAngleRadians = ( 1.f - factor ) * mAnimation - > getHeadPitch ( ) + factor * ( - xAngleRadians ) ;
zAngleRadians = ( 1.f - factor ) * mAnimation - > getHeadYaw ( ) + factor * ( - zAngleRadians ) ;
2014-12-16 19:47:45 +00:00
2015-05-31 16:04:14 +00:00
mAnimation - > setHeadPitch ( xAngleRadians ) ;
mAnimation - > setHeadYaw ( zAngleRadians ) ;
2014-12-16 19:47:45 +00:00
}
2013-01-10 16:35:24 +00:00
}