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
2018-03-08 20:23:24 +00:00
* https : //www.gnu.org/licenses/ .
2013-01-10 16:35:24 +00:00
*/
# include "character.hpp"
2015-06-14 21:13:26 +00:00
# include <iostream>
2020-08-27 11:48:59 +00:00
# include <components/misc/mathutil.hpp>
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"
2018-09-21 12:34:23 +00:00
# include "../mwbase/mechanicsmanager.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"
2020-03-28 15:30:56 +00:00
# include "../mwrender/npcanimation.hpp"
2013-01-16 21:09:21 +00:00
2018-06-28 12:58:51 +00:00
# include "aicombataction.hpp"
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
2020-05-17 14:25:51 +00:00
# ifdef USE_OPENXR
# include "../mwvr/vrenvironment.hpp"
# include "../mwvr/vranimation.hpp"
# endif
2013-07-16 21:32:41 +00:00
namespace
{
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 > ( ) ;
2018-08-29 15:38:12 +00:00
const float fallDistanceMin = store . find ( " fFallDamageDistanceMin " ) - > mValue . getFloat ( ) ;
2014-12-10 16:21:34 +00:00
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 ( ) ;
2018-08-29 15:38:12 +00:00
const float fallAcroBase = store . find ( " fFallAcroBase " ) - > mValue . getFloat ( ) ;
const float fallAcroMult = store . find ( " fFallAcroMult " ) - > mValue . getFloat ( ) ;
const float fallDistanceBase = store . find ( " fFallDistanceBase " ) - > mValue . getFloat ( ) ;
const float fallDistanceMult = store . find ( " fFallDistanceMult " ) - > mValue . getFloat ( ) ;
2014-12-10 16:21:34 +00:00
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 " } ,
2017-09-22 12:07:00 +00:00
{ CharState_SwimTurnLeft , " swimturnleft " } ,
{ CharState_SwimTurnRight , " swimturnright " } ,
2013-07-16 05:56:23 +00:00
} ;
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 ; }
} ;
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 ;
2019-01-02 21:31:59 +00:00
while ( mAnimation - > hasAnimation ( prefix + std : : to_string ( 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 ;
2019-01-02 21:31:59 +00:00
return prefix + std : : to_string ( roll ) ;
2014-01-19 20:11:48 +00:00
}
2013-05-01 17:19:16 +00:00
2018-08-20 18:04:02 +00:00
void CharacterController : : refreshHitRecoilAnims ( CharacterState & idle )
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 ( ) ;
2017-09-22 11:26:35 +00:00
bool isSwimming = MWBase : : Environment : : get ( ) . getWorld ( ) - > isSwimming ( mPtr ) ;
2016-06-15 01:14:44 +00:00
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
{
2017-11-04 19:37:20 +00:00
mTimeUntilWake = Misc : : Rng : : rollClosedProbability ( ) * 2 + 1 ; // Wake up after 1 to 3 seconds
2017-09-22 11:26:35 +00:00
if ( isSwimming & & mAnimation - > hasAnimation ( " swimknockout " ) )
{
mHitState = CharState_SwimKnockOut ;
mCurrentHit = " swimknockout " ;
}
else
{
mHitState = CharState_KnockOut ;
mCurrentHit = " knockout " ;
}
2016-06-15 01:14:44 +00:00
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 " ) )
{
2017-09-22 11:26:35 +00:00
if ( isSwimming & & mAnimation - > hasAnimation ( " swimknockdown " ) )
{
mHitState = CharState_SwimKnockDown ;
mCurrentHit = " swimknockdown " ;
}
else
{
mHitState = CharState_KnockDown ;
mCurrentHit = " knockdown " ;
}
2016-06-15 01:14:44 +00:00
mAnimation - > play ( mCurrentHit , Priority_Knockdown , MWRender : : Animation : : BlendMask_All , true , 1 , " start " , " stop " , 0.0f , 0 ) ;
}
else if ( recovery )
{
2018-04-21 13:42:28 +00:00
std : : string anim = chooseRandomGroup ( " swimhit " ) ;
2017-09-22 11:49:42 +00:00
if ( isSwimming & & mAnimation - > hasAnimation ( anim ) )
2014-01-21 00:01:21 +00:00
{
2017-09-22 11:49:42 +00:00
mHitState = CharState_SwimHit ;
2016-06-15 01:14:44 +00:00
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
}
2017-09-22 11:49:42 +00:00
else
{
anim = chooseRandomGroup ( " hit " ) ;
if ( mAnimation - > hasAnimation ( anim ) )
{
mHitState = CharState_Hit ;
mCurrentHit = anim ;
mAnimation - > play ( mCurrentHit , Priority_Hit , MWRender : : Animation : : BlendMask_All , true , 1 , " start " , " stop " , 0.0f , 0 ) ;
}
}
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 ;
2020-10-21 16:25:45 +00:00
priorityBlock [ MWRender : : Animation : : BoneGroup_LowerBody ] = Priority_WeaponLowerBody ;
2016-06-15 01:14:44 +00:00
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
2017-09-22 11:26:35 +00:00
if ( isKnockedOut ( ) | | isKnockedDown ( ) )
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 ;
2018-12-26 09:45:28 +00:00
if ( mWeaponType > ESM : : Weapon : : None )
2019-08-09 05:05:59 +00:00
mAnimation - > showWeapons ( true ) ;
2016-06-15 01:14:44 +00:00
}
else if ( mUpperBodyState > UpperCharState_Nothing & & mUpperBodyState < UpperCharState_WeapEquiped )
{
mAnimation - > disable ( mCurrentWeapon ) ;
mUpperBodyState = UpperCharState_Nothing ;
}
2014-02-02 02:24:40 +00:00
}
2020-10-21 16:25:45 +00:00
if ( mHitState ! = CharState_None )
idle = CharState_None ;
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 ;
}
2017-11-01 23:44:50 +00:00
else if ( isKnockedOut ( ) & & mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getFatigue ( ) . getCurrent ( ) > 0
2017-11-04 19:37:20 +00:00
& & mTimeUntilWake < = 0 )
2016-06-15 01:14:44 +00:00
{
2017-09-22 11:26:35 +00:00
mHitState = isSwimming ? CharState_SwimKnockDown : CharState_KnockDown ;
2016-06-15 01:14:44 +00:00
mAnimation - > disable ( mCurrentHit ) ;
mAnimation - > play ( mCurrentHit , Priority_Knockdown , MWRender : : Animation : : BlendMask_All , true , 1 , " loop stop " , " stop " , 0.0f , 0 ) ;
}
}
2013-12-31 11:24:20 +00:00
2018-12-26 09:45:28 +00:00
void CharacterController : : refreshJumpAnims ( const std : : string & weapShortGroup , JumpingState jump , CharacterState & idle , bool force )
2016-06-15 01:14:44 +00:00
{
2019-01-09 15:05:51 +00:00
if ( ! force & & jump = = mJumpState & & idle = = CharState_None )
2018-09-22 21:08:16 +00:00
return ;
2018-08-20 18:04:02 +00:00
2018-09-22 21:08:16 +00:00
std : : string jumpAnimName ;
MWRender : : Animation : : BlendMask jumpmask = MWRender : : Animation : : BlendMask_All ;
if ( jump ! = JumpState_None )
{
jumpAnimName = " jump " ;
2018-12-26 09:45:28 +00:00
if ( ! weapShortGroup . empty ( ) )
2013-08-19 15:10:18 +00:00
{
2018-12-26 09:45:28 +00:00
jumpAnimName + = weapShortGroup ;
2018-09-22 21:08:16 +00:00
if ( ! mAnimation - > hasAnimation ( jumpAnimName ) )
2013-08-19 15:10:18 +00:00
{
2019-08-09 08:10:28 +00:00
jumpAnimName = fallbackShortWeaponGroup ( " jump " , & jumpmask ) ;
2018-06-22 14:40:49 +00:00
2019-08-09 08:10:28 +00:00
// If we apply jump only for lower body, do not reset idle animations.
2018-09-22 21:08:16 +00:00
// For upper body there will be idle animation.
2019-08-09 08:10:28 +00:00
if ( jumpmask = = MWRender : : Animation : : BlendMask_LowerBody & & idle = = CharState_None )
2018-09-22 21:08:16 +00:00
idle = CharState_Idle ;
2013-08-19 15:10:18 +00:00
}
}
2018-09-22 21:08:16 +00:00
}
2013-08-19 15:10:18 +00:00
2018-09-22 21:08:16 +00:00
if ( ! force & & jump = = mJumpState )
return ;
2019-01-09 15:05:51 +00:00
bool startAtLoop = ( jump = = mJumpState ) ;
2018-09-22 21:08:16 +00:00
mJumpState = jump ;
2017-11-05 19:19:47 +00:00
2018-09-22 21:08:16 +00:00
if ( ! mCurrentJump . empty ( ) )
{
mAnimation - > disable ( mCurrentJump ) ;
mCurrentJump . clear ( ) ;
}
if ( mJumpState = = JumpState_InAir )
{
if ( mAnimation - > hasAnimation ( jumpAnimName ) )
2017-11-05 19:19:47 +00:00
{
2018-09-22 21:08:16 +00:00
mAnimation - > play ( jumpAnimName , Priority_Jump , jumpmask , false ,
2019-01-09 15:05:51 +00:00
1.0f , startAtLoop ? " loop start " : " start " , " stop " , 0.f , ~ 0ul ) ;
2018-09-22 21:08:16 +00:00
mCurrentJump = jumpAnimName ;
2013-08-19 15:10:18 +00:00
}
2018-09-22 21:08:16 +00:00
}
else if ( mJumpState = = JumpState_Landing )
{
if ( mAnimation - > hasAnimation ( jumpAnimName ) )
2013-08-19 15:10:18 +00:00
{
2018-09-22 21:08:16 +00:00
mAnimation - > play ( jumpAnimName , Priority_Jump , jumpmask , true ,
1.0f , " loop stop " , " stop " , 0.0f , 0 ) ;
mCurrentJump = jumpAnimName ;
2017-10-31 13:22:24 +00:00
}
2013-08-19 15:10:18 +00:00
}
2016-06-15 01:14:44 +00:00
}
2013-08-19 15:10:18 +00:00
2018-09-22 08:57:50 +00:00
bool CharacterController : : onOpen ( )
{
if ( mPtr . getTypeName ( ) = = typeid ( ESM : : Container ) . name ( ) )
{
if ( ! mAnimation - > hasAnimation ( " containeropen " ) )
return true ;
if ( mAnimation - > isPlaying ( " containeropen " ) )
return false ;
if ( mAnimation - > isPlaying ( " containerclose " ) )
return false ;
mAnimation - > play ( " containeropen " , Priority_Persistent , MWRender : : Animation : : BlendMask_All , false , 1.0f , " start " , " stop " , 0.f , 0 ) ;
if ( mAnimation - > isPlaying ( " containeropen " ) )
return false ;
}
return true ;
}
void CharacterController : : onClose ( )
{
if ( mPtr . getTypeName ( ) = = typeid ( ESM : : Container ) . name ( ) )
{
if ( ! mAnimation - > hasAnimation ( " containerclose " ) )
return ;
float complete , startPoint = 0.f ;
bool animPlaying = mAnimation - > getInfo ( " containeropen " , & complete ) ;
if ( animPlaying )
startPoint = 1.f - complete ;
mAnimation - > play ( " containerclose " , Priority_Persistent , MWRender : : Animation : : BlendMask_All , false , 1.0f , " start " , " stop " , startPoint , 0 ) ;
}
}
2019-08-09 08:58:20 +00:00
std : : string CharacterController : : getWeaponAnimation ( int weaponType ) const
{
std : : string weaponGroup = getWeaponType ( weaponType ) - > mLongGroup ;
bool isRealWeapon = weaponType ! = ESM : : Weapon : : HandToHand & & weaponType ! = ESM : : Weapon : : Spell & & weaponType ! = ESM : : Weapon : : None ;
if ( isRealWeapon & & ! mAnimation - > hasAnimation ( weaponGroup ) )
{
static const std : : string oneHandFallback = getWeaponType ( ESM : : Weapon : : LongBladeOneHand ) - > mLongGroup ;
static const std : : string twoHandFallback = getWeaponType ( ESM : : Weapon : : LongBladeTwoHand ) - > mLongGroup ;
const ESM : : WeaponType * weapInfo = getWeaponType ( weaponType ) ;
// For real two-handed melee weapons use 2h swords animations as fallback, otherwise use the 1h ones
if ( weapInfo - > mFlags & ESM : : WeaponType : : TwoHanded & & weapInfo - > mWeaponClass = = ESM : : WeaponType : : Melee )
weaponGroup = twoHandFallback ;
else if ( isRealWeapon )
weaponGroup = oneHandFallback ;
}
return weaponGroup ;
}
2019-08-09 08:10:28 +00:00
std : : string CharacterController : : fallbackShortWeaponGroup ( const std : : string & baseGroupName , MWRender : : Animation : : BlendMask * blendMask )
{
bool isRealWeapon = mWeaponType ! = ESM : : Weapon : : HandToHand & & mWeaponType ! = ESM : : Weapon : : Spell & & mWeaponType ! = ESM : : Weapon : : None ;
if ( ! isRealWeapon )
{
if ( blendMask ! = nullptr )
* blendMask = MWRender : : Animation : : BlendMask_LowerBody ;
return baseGroupName ;
}
static const std : : string oneHandFallback = getWeaponType ( ESM : : Weapon : : LongBladeOneHand ) - > mShortGroup ;
static const std : : string twoHandFallback = getWeaponType ( ESM : : Weapon : : LongBladeTwoHand ) - > mShortGroup ;
std : : string groupName = baseGroupName ;
const ESM : : WeaponType * weapInfo = getWeaponType ( mWeaponType ) ;
// For real two-handed melee weapons use 2h swords animations as fallback, otherwise use the 1h ones
if ( isRealWeapon & & weapInfo - > mFlags & ESM : : WeaponType : : TwoHanded & & weapInfo - > mWeaponClass = = ESM : : WeaponType : : Melee )
groupName + = twoHandFallback ;
else if ( isRealWeapon )
groupName + = oneHandFallback ;
// Special case for crossbows - we shouls apply 1h animations a fallback only for lower body
if ( mWeaponType = = ESM : : Weapon : : MarksmanCrossbow & & blendMask ! = nullptr )
* blendMask = MWRender : : Animation : : BlendMask_LowerBody ;
if ( ! mAnimation - > hasAnimation ( groupName ) )
{
groupName = baseGroupName ;
if ( blendMask ! = nullptr )
* blendMask = MWRender : : Animation : : BlendMask_LowerBody ;
}
return groupName ;
}
2018-12-26 09:45:28 +00:00
void CharacterController : : refreshMovementAnims ( const std : : string & weapShortGroup , CharacterState movement , CharacterState & idle , bool force )
2016-06-15 01:14:44 +00:00
{
2018-09-17 09:35:22 +00:00
if ( movement = = mMovementState & & idle = = mIdleState & & ! force )
return ;
2019-05-14 13:34:41 +00:00
// Reset idle if we actually play movement animations excepts of these cases:
// 1. When we play turning animations
// 2. When we use a fallback animation for lower body since movement animation for given weapon is missing (e.g. for crossbows and spellcasting)
bool resetIdle = ( movement ! = CharState_None & & ! isTurning ( ) ) ;
2018-08-20 18:04:02 +00:00
std : : string movementAnimName ;
MWRender : : Animation : : BlendMask movemask ;
const StateInfo * movestate ;
2018-09-17 09:35:22 +00:00
movemask = MWRender : : Animation : : BlendMask_All ;
movestate = std : : find_if ( sMovementList , sMovementListEnd , FindCharState ( movement ) ) ;
if ( movestate ! = sMovementListEnd )
2013-07-16 05:56:23 +00:00
{
2018-09-17 09:35:22 +00:00
movementAnimName = movestate - > groupname ;
2018-12-26 09:45:28 +00:00
if ( ! weapShortGroup . empty ( ) )
2013-07-16 05:56:23 +00:00
{
2018-10-05 13:29:57 +00:00
std : : string : : size_type swimpos = movementAnimName . find ( " swim " ) ;
if ( swimpos = = std : : string : : npos )
{
2018-12-26 09:45:28 +00:00
if ( mWeaponType = = ESM : : Weapon : : Spell & & ( movement = = CharState_TurnLeft | | movement = = CharState_TurnRight ) ) // Spellcasting stance turning is a special case
movementAnimName = weapShortGroup + movementAnimName ;
2018-10-05 13:29:57 +00:00
else
2018-12-26 09:45:28 +00:00
movementAnimName + = weapShortGroup ;
2018-10-05 13:29:57 +00:00
}
2018-09-17 09:35:22 +00:00
if ( ! mAnimation - > hasAnimation ( movementAnimName ) )
2013-07-16 05:56:23 +00:00
{
2018-09-17 09:35:22 +00:00
movementAnimName = movestate - > groupname ;
2018-10-05 13:29:57 +00:00
if ( swimpos = = std : : string : : npos )
{
2019-08-09 08:10:28 +00:00
movementAnimName = fallbackShortWeaponGroup ( movementAnimName , & movemask ) ;
// If we apply movement only for lower body, do not reset idle animations.
2018-10-05 13:29:57 +00:00
// For upper body there will be idle animation.
2019-08-09 08:10:28 +00:00
if ( movemask = = MWRender : : Animation : : BlendMask_LowerBody & & idle = = CharState_None )
2018-10-05 13:29:57 +00:00
idle = CharState_Idle ;
2019-05-14 13:34:41 +00:00
if ( movemask = = MWRender : : Animation : : BlendMask_LowerBody )
resetIdle = false ;
2018-10-05 13:29:57 +00:00
}
2013-07-16 05:56:23 +00:00
}
2018-08-20 18:04:02 +00:00
}
}
2013-07-16 05:56:23 +00:00
2018-08-20 18:04:02 +00:00
if ( force | | movement ! = mMovementState )
{
mMovementState = movement ;
if ( movestate ! = sMovementListEnd )
{
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 " ) ;
2019-08-09 21:55:18 +00:00
if ( swimpos ! = std : : string : : npos )
2013-07-16 05:56:23 +00:00
{
2015-07-15 13:46:31 +00:00
movementAnimName . erase ( swimpos , 4 ) ;
2018-12-26 09:45:28 +00:00
if ( ! weapShortGroup . empty ( ) )
2015-12-11 00:24:33 +00:00
{
2018-12-26 09:45:28 +00:00
std : : string weapMovementAnimName = movementAnimName + weapShortGroup ;
2015-12-11 00:24:33 +00:00
if ( mAnimation - > hasAnimation ( weapMovementAnimName ) )
movementAnimName = weapMovementAnimName ;
else
2019-05-14 13:34:41 +00:00
{
2019-08-09 08:10:28 +00:00
movementAnimName = fallbackShortWeaponGroup ( movementAnimName , & movemask ) ;
2019-05-14 13:34:41 +00:00
if ( movemask = = MWRender : : Animation : : BlendMask_LowerBody )
resetIdle = false ;
}
2015-12-11 00:24:33 +00:00
}
2019-08-09 21:55:18 +00:00
}
2015-12-11 00:24:33 +00:00
2019-08-09 21:55:18 +00:00
if ( swimpos = = std : : string : : npos | | ! mAnimation - > hasAnimation ( movementAnimName ) )
{
std : : string : : size_type runpos = movementAnimName . find ( " run " ) ;
if ( runpos ! = std : : string : : npos )
{
movementAnimName . replace ( runpos , runpos + 3 , " walk " ) ;
if ( ! mAnimation - > hasAnimation ( movementAnimName ) )
movementAnimName . clear ( ) ;
}
else
2015-07-15 13:46:31 +00:00
movementAnimName . clear ( ) ;
2013-07-16 05:56:23 +00:00
}
}
}
2018-07-04 22:37:11 +00:00
// If we're playing the same animation, start it from the point it ended
2018-09-22 21:08:16 +00:00
float startpoint = 0.f ;
if ( ! mCurrentMovement . empty ( ) & & movementAnimName = = mCurrentMovement )
mAnimation - > getInfo ( mCurrentMovement , & startpoint ) ;
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 ) ;
2019-05-14 13:34:41 +00:00
if ( ! mAnimation - > hasAnimation ( movementAnimName ) )
movementAnimName . clear ( ) ;
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
{
2019-12-03 17:50:57 +00:00
if ( resetIdle )
{
mAnimation - > disable ( mCurrentIdle ) ;
mIdleState = CharState_None ;
idle = CharState_None ;
}
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.
2020-04-21 06:32:34 +00:00
bool sneaking = mMovementState = = CharState_SneakForward | | mMovementState = = CharState_SneakBack
| | mMovementState = = CharState_SneakLeft | | mMovementState = = CharState_SneakRight ;
mMovementAnimSpeed = ( sneaking ? 33.5452f : ( 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 ,
2018-09-22 21:08:16 +00:00
1.f , " start " , " stop " , startpoint , ~ 0ul , true ) ;
2013-07-16 08:30:03 +00:00
}
2019-05-14 13:34:41 +00:00
else
mMovementState = CharState_None ;
2013-07-16 05:56:23 +00:00
}
2016-06-15 01:14:44 +00:00
}
2014-12-31 22:01:31 +00:00
2018-12-26 09:45:28 +00:00
void CharacterController : : refreshIdleAnims ( const std : : string & weapShortGroup , CharacterState idle , bool force )
2016-06-15 01:14:44 +00:00
{
2018-08-20 18:04:02 +00:00
// 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 & & mUpperBodyState ! = UpperCharState_WeapEquiped )
| | ( mMovementState ! = CharState_None & & ! isTurning ( ) )
| | mHitState ! = CharState_None )
& & ! mPtr . getClass ( ) . isBipedal ( mPtr ) )
idle = CharState_None ;
if ( force | | 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 " ;
2018-12-26 09:45:28 +00:00
if ( ! weapShortGroup . empty ( ) )
2014-12-31 22:01:31 +00:00
{
2018-12-26 09:45:28 +00:00
idleGroup + = weapShortGroup ;
2016-10-05 16:12:06 +00:00
if ( ! mAnimation - > hasAnimation ( idleGroup ) )
2019-08-09 08:10:28 +00:00
{
idleGroup = fallbackShortWeaponGroup ( " 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 ) ;
2018-08-16 07:21:48 +00:00
}
2014-12-31 22:01:31 +00:00
}
2018-08-16 10:37:18 +00:00
// There is no need to restart anim if the new and old anims are the same.
// Just update a number of loops.
float startPoint = 0 ;
if ( ! mCurrentIdle . empty ( ) & & mCurrentIdle = = idleGroup )
{
mAnimation - > getInfo ( mCurrentIdle , & startPoint ) ;
}
2018-08-16 07:21:48 +00:00
if ( ! mCurrentIdle . empty ( ) )
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 ,
2018-08-16 10:37:18 +00:00
1.0f , " start " , " stop " , startPoint , 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 )
{
2018-06-11 13:18:51 +00:00
// If the current animation is persistent, do not touch it
2018-06-12 07:51:54 +00:00
if ( isPersistentAnimPlaying ( ) )
return ;
2018-06-11 13:18:51 +00:00
2016-06-15 01:14:44 +00:00
if ( mPtr . getClass ( ) . isActor ( ) )
2018-08-20 18:04:02 +00:00
refreshHitRecoilAnims ( idle ) ;
2016-06-15 01:14:44 +00:00
2018-12-26 09:45:28 +00:00
std : : string weap ;
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) )
weap = getWeaponType ( mWeaponType ) - > mShortGroup ;
2016-06-15 01:14:44 +00:00
2019-01-09 15:05:51 +00:00
refreshJumpAnims ( weap , jump , idle , force ) ;
2018-08-20 18:04:02 +00:00
refreshMovementAnims ( weap , movement , idle , force ) ;
2016-06-15 01:14:44 +00:00
// idle handled last as it can depend on the other states
refreshIdleAnims ( weap , idle , force ) ;
}
2014-05-26 17:56:32 +00:00
void CharacterController : : playDeath ( float startpoint , CharacterState death )
{
2018-07-05 19:49:40 +00:00
// Make sure the character was swimming upon death for forward-compatibility
const bool wasSwimming = MWBase : : Environment : : get ( ) . getWorld ( ) - > isSwimming ( mPtr ) ;
2014-05-26 17:56:32 +00:00
switch ( death )
{
case CharState_SwimDeath :
mCurrentDeath = " swimdeath " ;
break ;
2017-09-22 10:51:06 +00:00
case CharState_SwimDeathKnockDown :
2018-07-05 19:49:40 +00:00
mCurrentDeath = ( wasSwimming ? " swimdeathknockdown " : " deathknockdown " ) ;
2017-09-22 10:51:06 +00:00
break ;
case CharState_SwimDeathKnockOut :
2018-07-05 19:49:40 +00:00
mCurrentDeath = ( wasSwimming ? " swimdeathknockout " : " deathknockout " ) ;
2017-09-22 10:51:06 +00:00
break ;
2014-05-26 17:56:32 +00:00
case CharState_DeathKnockDown :
mCurrentDeath = " deathknockdown " ;
break ;
case CharState_DeathKnockOut :
mCurrentDeath = " deathknockout " ;
break ;
default :
2019-01-02 21:31:59 +00:00
mCurrentDeath = " death " + std : : to_string ( 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 ( ) ;
}
2017-09-22 11:26:35 +00:00
if ( mHitState = = CharState_SwimKnockDown & & mAnimation - > hasAnimation ( " swimdeathknockdown " ) )
2017-09-22 10:51:06 +00:00
{
mDeathState = CharState_SwimDeathKnockDown ;
}
2017-09-22 11:26:35 +00:00
else if ( mHitState = = CharState_SwimKnockOut & & mAnimation - > hasAnimation ( " swimdeathknockout " ) )
2017-09-22 10:51:06 +00:00
{
mDeathState = CharState_SwimDeathKnockOut ;
}
2017-09-22 11:26:35 +00:00
else 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
}
2018-06-12 05:55:43 +00:00
// Do not interrupt scripted animation by death
2018-06-12 07:51:54 +00:00
if ( isPersistentAnimPlaying ( ) )
return ;
2018-06-12 05:55:43 +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
2018-07-19 12:38:32 +00:00
std : : string CharacterController : : chooseRandomAttackAnimation ( ) const
{
std : : string result ;
bool isSwimming = MWBase : : Environment : : get ( ) . getWorld ( ) - > isSwimming ( mPtr ) ;
if ( isSwimming )
result = chooseRandomGroup ( " swimattack " ) ;
if ( ! isSwimming | | ! mAnimation - > hasAnimation ( result ) )
result = chooseRandomGroup ( " attack " ) ;
return result ;
}
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 )
2018-05-02 10:13:49 +00:00
, mWeapon ( MWWorld : : Ptr ( ) )
2013-04-29 20:07:49 +00:00
, 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 )
2018-12-26 09:45:28 +00:00
, mWeaponType ( ESM : : Weapon : : 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 )
2018-06-28 12:58:51 +00:00
, mCastingManualSpell ( false )
2017-11-05 18:30:34 +00:00
, mTimeUntilWake ( 0.f )
2020-06-26 20:04:02 +00:00
, mIsMovingBackward ( 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
{
2018-12-26 09:45:28 +00:00
getActiveWeapon ( mPtr , & mWeaponType ) ;
if ( mWeaponType ! = ESM : : Weapon : : None )
2013-07-25 04:08:16 +00:00
{
mUpperBodyState = UpperCharState_WeapEquiped ;
2019-08-09 08:58:20 +00:00
mCurrentWeapon = getWeaponAnimation ( mWeaponType ) ;
2014-07-03 18:28:52 +00:00
}
2018-12-26 09:45:28 +00:00
if ( mWeaponType ! = ESM : : Weapon : : None & & mWeaponType ! = ESM : : Weapon : : Spell & & mWeaponType ! = ESM : : Weapon : : HandToHand )
2014-07-03 18:28:52 +00:00
{
2014-01-15 20:56:55 +00:00
mAnimation - > showWeapons ( true ) ;
2019-03-05 09:33:58 +00:00
// Note: controllers for ranged weapon should use time for beginning of animation to play shooting properly,
// for other weapons they should use absolute time. Some mods rely on this behaviour (to rotate throwing projectiles, for example)
2018-12-26 09:45:28 +00:00
ESM : : WeaponType : : Class weaponClass = getWeaponType ( mWeaponType ) - > mWeaponClass ;
bool useRelativeDuration = weaponClass = = ESM : : WeaponType : : Ranged ;
2019-03-05 09:33:58 +00:00
mAnimation - > setWeaponGroup ( mCurrentWeapon , useRelativeDuration ) ;
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
2018-06-11 19:17:54 +00:00
// Do not update animation status for dead actors
2018-06-12 07:27:18 +00:00
if ( mDeathState = = CharState_None & & ( ! cls . isActor ( ) | | ! cls . getCreatureStats ( mPtr ) . isDead ( ) ) )
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 ( ) ;
2018-10-09 06:21:12 +00:00
mAnimation - > setTextKeyListener ( nullptr ) ;
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 ) ;
}
}
2020-11-18 20:48:47 +00:00
void CharacterController : : handleTextKey ( const std : : string & groupname , SceneUtil : : TextKeyMap : : ConstIterator key , const SceneUtil : : TextKeyMap & map )
2015-05-21 22:55:43 +00:00
{
const std : : string & evt = key - > second ;
if ( evt . compare ( 0 , 7 , " sound: " ) = = 0 )
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
sndMgr - > playSound3D ( mPtr , evt . substr ( 7 ) , 1.0f , 1.0f ) ;
return ;
}
2018-08-22 12:26:21 +00:00
if ( evt . compare ( 0 , 10 , " soundgen: " ) = = 0 )
2015-05-21 22:55:43 +00:00
{
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 ) ;
2018-08-22 11:43:12 +00:00
if ( ! sound . empty ( ) )
2015-05-21 22:55:43 +00:00
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
2018-10-14 17:11:21 +00:00
// NB: landing sound is not played for NPCs here
if ( soundgen = = " left " | | soundgen = = " right " | | soundgen = = " land " )
2015-11-27 09:02:53 +00:00
{
2017-09-15 08:03:41 +00:00
sndMgr - > playSound3D ( mPtr , sound , volume , pitch , MWSound : : Type : : Foot ,
MWSound : : PlayMode : : NoPlayerLocal ) ;
2015-11-27 09:02:53 +00:00
}
else
2016-03-24 17:31:37 +00:00
{
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 ;
2019-08-11 11:01:48 +00:00
if ( groupname = = " shield " & & evt . compare ( off , len , " equip attach " ) = = 0 )
mAnimation - > showCarriedLeft ( true ) ;
else if ( groupname = = " shield " & & evt . compare ( off , len , " unequip detach " ) = = 0 )
mAnimation - > showCarriedLeft ( false ) ;
else if ( evt . compare ( off , len , " equip attach " ) = = 0 )
2015-05-21 22:55:43 +00:00
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 )
{
2017-09-22 10:51:06 +00:00
if ( groupname = = " attack1 " | | groupname = = " swimattack1 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Chop ) ;
2017-09-22 10:51:06 +00:00
else if ( groupname = = " attack2 " | | groupname = = " swimattack2 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Slash ) ;
2017-09-23 04:25:58 +00:00
else if ( groupname = = " attack3 " | | groupname = = " swimattack3 " )
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
2020-03-15 14:31:38 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , - 1 ) ;
2015-05-21 22:55:43 +00:00
}
2017-09-22 10:51:06 +00:00
else if ( ! groupname . empty ( )
& & ( groupname . compare ( 0 , groupname . size ( ) - 1 , " attack " ) = = 0 | | groupname . compare ( 0 , groupname . size ( ) - 1 , " swimattack " ) = = 0 )
2015-05-21 22:55:43 +00:00
& & 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 )
{
2017-09-22 10:51:06 +00:00
if ( groupname = = " attack1 " | | groupname = = " swimattack1 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Chop ) ;
2017-09-22 10:51:06 +00:00
else if ( groupname = = " attack2 " | | groupname = = " swimattack2 " )
2015-06-26 03:15:07 +00:00
mPtr . getClass ( ) . hit ( mPtr , mAttackStrength , ESM : : Weapon : : AT_Slash ) ;
2017-09-22 10:51:06 +00:00
else if ( groupname = = " attack3 " | | groupname = = " swimattack3 " )
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
{
2018-06-28 12:58:51 +00:00
MWBase : : Environment : : get ( ) . getWorld ( ) - > castSpell ( mPtr , mCastingManualSpell ) ;
mCastingManualSpell = false ;
2015-05-21 22:55:43 +00:00
}
else if ( groupname = = " shield " & & evt . compare ( off , len , " block hit " ) = = 0 )
mPtr . getClass ( ) . block ( mPtr ) ;
2018-09-22 08:57:50 +00:00
else if ( groupname = = " containeropen " & & evt . compare ( off , len , " loot " ) = = 0 )
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > pushGuiMode ( MWGui : : GM_Container , mPtr ) ;
2015-05-21 22:55:43 +00:00
}
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
{
2019-05-24 15:40:06 +00:00
if ( ! mAnimation - > hasAnimation ( " idlestorm " ) | | mUpperBodyState ! = UpperCharState_Nothing | | inwater )
{
mAnimation - > disable ( " idlestorm " ) ;
return ;
}
2014-06-24 16:37:38 +00:00
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 ) ;
2019-05-24 15:40:06 +00:00
stormDirection . normalize ( ) ;
characterDirection . normalize ( ) ;
if ( stormDirection * characterDirection < - 0.5f )
2014-06-24 16:37:38 +00:00
{
2019-05-24 15:40:06 +00:00
if ( ! mAnimation - > isPlaying ( " idlestorm " ) )
2014-06-24 16:37:38 +00:00
{
2019-06-22 20:05:20 +00:00
int mask = MWRender : : Animation : : BlendMask_Torso | MWRender : : Animation : : BlendMask_RightArm ;
mAnimation - > play ( " idlestorm " , Priority_Storm , mask , true , 1.0f , " start " , " stop " , 0.0f , ~ 0ul ) ;
2014-06-24 16:37:38 +00:00
}
2019-05-24 15:40:06 +00:00
else
{
mAnimation - > setLoopingEnabled ( " idlestorm " , true ) ;
}
return ;
2014-06-24 16:37:38 +00:00
}
2019-05-24 15:40:06 +00:00
}
if ( mAnimation - > isPlaying ( " idlestorm " ) )
{
mAnimation - > setLoopingEnabled ( " idlestorm " , false ) ;
2014-06-24 16:37:38 +00:00
}
}
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 ) ;
2018-12-26 09:45:28 +00:00
int weapType = ESM : : Weapon : : None ;
2014-09-14 22:29:21 +00:00
if ( stats . getDrawState ( ) = = DrawState_Weapon )
2018-12-26 09:45:28 +00:00
weapType = ESM : : Weapon : : HandToHand ;
2014-09-14 22:29:21 +00:00
else if ( stats . getDrawState ( ) = = DrawState_Spell )
2018-12-26 09:45:28 +00:00
weapType = ESM : : Weapon : : Spell ;
2014-09-14 22:29:21 +00:00
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 " ;
2018-12-26 09:45:28 +00:00
if ( weapType = = ESM : : Weapon : : Spell )
2014-09-14 22:29:21 +00:00
{
const std : : string spellid = stats . getSpells ( ) . getSelectedSpell ( ) ;
2018-06-28 12:58:51 +00:00
bool canCast = mCastingManualSpell | | MWBase : : Environment : : get ( ) . getWorld ( ) - > startSpellCast ( mPtr ) ;
if ( ! spellid . empty ( ) & & canCast )
2014-09-14 22:29:21 +00:00
{
2018-10-09 06:21:12 +00:00
MWMechanics : : CastSpell cast ( mPtr , nullptr , false , mCastingManualSpell ) ;
2019-04-08 08:07:44 +00:00
cast . playSpellCastingEffects ( spellid , false ) ;
2014-09-18 01:24:47 +00:00
if ( ! mAnimation - > hasAnimation ( " spellcast " ) )
2018-06-28 12:58:51 +00:00
{
MWBase : : Environment : : get ( ) . getWorld ( ) - > castSpell ( mPtr , mCastingManualSpell ) ; // No "release" text key to use, so cast immediately
mCastingManualSpell = false ;
}
2014-09-18 01:24:47 +00:00
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 = " " ;
}
2018-07-19 12:38:32 +00:00
2018-12-26 09:45:28 +00:00
if ( weapType ! = ESM : : Weapon : : Spell | | ! mAnimation - > hasAnimation ( " spellcast " ) ) // Not all creatures have a dedicated spellcast animation
2014-09-18 01:24:47 +00:00
{
2018-07-19 12:38:32 +00:00
mCurrentWeapon = chooseRandomAttackAnimation ( ) ;
2014-09-18 01:24:47 +00:00
}
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
2018-12-26 09:45:28 +00:00
if ( weapType = = ESM : : Weapon : : HandToHand )
2016-12-11 18:35:53 +00:00
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 ;
}
2018-12-26 09:45:28 +00:00
bool CharacterController : : updateCarriedLeftVisible ( const int weaptype ) const
2014-12-12 15:49:22 +00:00
{
// 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.
2019-08-11 11:01:48 +00:00
return mAnimation - > updateCarriedLeftVisible ( weaptype ) ;
2014-12-12 15:49:22 +00:00
}
2018-08-20 18:04:02 +00:00
bool CharacterController : : updateWeaponState ( CharacterState & idle )
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 ) ;
2018-12-26 09:45:28 +00:00
int weaptype = ESM : : Weapon : : None ;
2014-12-31 15:59:21 +00:00
if ( stats . getDrawState ( ) = = DrawState_Weapon )
2018-12-26 09:45:28 +00:00
weaptype = ESM : : Weapon : : HandToHand ;
2014-12-31 15:59:21 +00:00
else if ( stats . getDrawState ( ) = = DrawState_Spell )
2018-12-26 09:45:28 +00:00
weaptype = ESM : : Weapon : : Spell ;
2014-12-31 15:59:21 +00:00
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
2018-05-02 10:13:49 +00:00
std : : string upSoundId ;
std : : string downSoundId ;
2018-11-23 16:07:52 +00:00
bool weaponChanged = false ;
2014-12-31 15:59:21 +00:00
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) )
{
MWWorld : : InventoryStore & inv = cls . getInventoryStore ( mPtr ) ;
2018-12-26 09:45:28 +00:00
MWWorld : : ContainerStoreIterator weapon = getActiveWeapon ( mPtr , & weaptype ) ;
2018-05-02 10:13:49 +00:00
if ( stats . getDrawState ( ) = = DrawState_Spell )
weapon = inv . getSlot ( MWWorld : : InventoryStore : : Slot_CarriedRight ) ;
2018-12-26 09:45:28 +00:00
if ( weapon ! = inv . end ( ) & & mWeaponType ! = ESM : : Weapon : : HandToHand & & weaptype ! = ESM : : Weapon : : HandToHand & & weaptype ! = ESM : : Weapon : : Spell & & weaptype ! = ESM : : Weapon : : None )
2018-05-02 10:13:49 +00:00
upSoundId = weapon - > getClass ( ) . getUpSoundId ( * weapon ) ;
2018-12-26 09:45:28 +00:00
if ( weapon ! = inv . end ( ) & & mWeaponType ! = ESM : : Weapon : : HandToHand & & mWeaponType ! = ESM : : Weapon : : Spell & & mWeaponType ! = ESM : : Weapon : : None )
2018-05-02 10:13:49 +00:00
downSoundId = weapon - > getClass ( ) . getDownSoundId ( * weapon ) ;
// weapon->HtH switch: weapon is empty already, so we need to take sound from previous weapon
2018-12-26 09:45:28 +00:00
if ( weapon = = inv . end ( ) & & ! mWeapon . isEmpty ( ) & & weaptype = = ESM : : Weapon : : HandToHand & & mWeaponType ! = ESM : : Weapon : : Spell )
2018-05-02 10:13:49 +00:00
downSoundId = mWeapon . getClass ( ) . getDownSoundId ( mWeapon ) ;
2018-11-23 16:07:52 +00:00
MWWorld : : Ptr newWeapon = weapon ! = inv . end ( ) ? * weapon : MWWorld : : Ptr ( ) ;
if ( mWeapon ! = newWeapon )
{
mWeapon = newWeapon ;
weaponChanged = true ;
}
2014-12-31 15:59:21 +00:00
}
2020-05-26 14:01:45 +00:00
// For biped actors, blend weapon animations with lower body animations with higher priority
2015-07-15 12:40:36 +00:00
MWRender : : Animation : : AnimPriority priorityWeapon ( Priority_Weapon ) ;
2020-05-26 14:01:45 +00:00
if ( mPtr . getClass ( ) . isBipedal ( mPtr ) )
2018-07-17 17:00:13 +00:00
priorityWeapon [ MWRender : : Animation : : BoneGroup_LowerBody ] = Priority_WeaponLowerBody ;
2018-07-17 16:59:05 +00:00
2013-07-23 10:26:24 +00:00
bool forcestateupdate = false ;
2017-09-21 06:32:34 +00:00
// We should not play equipping animation and sound during weapon->weapon transition
2018-12-26 09:45:28 +00:00
bool isStillWeapon = weaptype ! = ESM : : Weapon : : HandToHand & & weaptype ! = ESM : : Weapon : : Spell & & weaptype ! = ESM : : Weapon : : None & &
mWeaponType ! = ESM : : Weapon : : HandToHand & & mWeaponType ! = ESM : : Weapon : : Spell & & mWeaponType ! = ESM : : Weapon : : None ;
2017-09-21 06:32:34 +00:00
2018-10-18 07:42:03 +00:00
// If the current weapon type was changed in the middle of attack (e.g. by Equip console command or when bound spell expires),
// we should force actor to the "weapon equipped" state, interrupt attack and update animations.
if ( isStillWeapon & & mWeaponType ! = weaptype & & mUpperBodyState > UpperCharState_WeapEquiped )
{
forcestateupdate = true ;
mUpperBodyState = UpperCharState_WeapEquiped ;
mAttackingOrSpell = false ;
mAnimation - > disable ( mCurrentWeapon ) ;
2019-08-09 05:05:59 +00:00
mAnimation - > showWeapons ( true ) ;
2018-10-18 07:42:03 +00:00
if ( mPtr = = getPlayer ( ) )
MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . setAttackingOrSpell ( false ) ;
}
2018-07-30 18:24:25 +00:00
if ( ! isKnockedOut ( ) & & ! isKnockedDown ( ) & & ! isRecovery ( ) )
2013-07-23 10:26:24 +00:00
{
std : : string weapgroup ;
2018-12-26 09:45:28 +00:00
if ( ( ! isWerewolf | | mWeaponType ! = ESM : : Weapon : : Spell )
2018-07-30 18:24:25 +00:00
& & weaptype ! = mWeaponType
2018-05-02 10:13:49 +00:00
& & mUpperBodyState ! = UpperCharState_UnEquipingWeap
& & ! isStillWeapon )
2013-07-23 10:26:24 +00:00
{
2018-11-23 16:07:52 +00:00
// We can not play un-equip animation if weapon changed since last update
if ( ! weaponChanged )
2018-07-17 10:06:41 +00:00
{
// Note: we do not disable unequipping animation automatically to avoid body desync
2019-08-09 08:58:20 +00:00
weapgroup = getWeaponAnimation ( mWeaponType ) ;
2019-08-11 11:01:48 +00:00
int unequipMask = MWRender : : Animation : : BlendMask_All ;
bool useShieldAnims = mAnimation - > useShieldAnimations ( ) ;
if ( useShieldAnims & & mWeaponType ! = ESM : : Weapon : : HandToHand & & mWeaponType ! = ESM : : Weapon : : Spell & & ! ( mWeaponType = = ESM : : Weapon : : None & & weaptype = = ESM : : Weapon : : Spell ) )
{
unequipMask = unequipMask | ~ MWRender : : Animation : : BlendMask_LeftArm ;
mAnimation - > play ( " shield " , Priority_Block ,
MWRender : : Animation : : BlendMask_LeftArm , true ,
1.0f , " unequip start " , " unequip stop " , 0.0f , 0 ) ;
}
else if ( mWeaponType = = ESM : : Weapon : : HandToHand )
mAnimation - > showCarriedLeft ( false ) ;
mAnimation - > play ( weapgroup , priorityWeapon , unequipMask , false ,
2018-07-17 10:06:41 +00:00
1.0f , " unequip start " , " unequip stop " , 0.0f , 0 ) ;
mUpperBodyState = UpperCharState_UnEquipingWeap ;
2018-07-19 12:38:32 +00:00
2020-08-28 11:28:26 +00:00
mAnimation - > detachArrow ( ) ;
2018-07-19 12:38:32 +00:00
// If we do not have the "unequip detach" key, hide weapon manually.
if ( mAnimation - > getTextKeyTime ( weapgroup + " : unequip detach " ) < 0 )
mAnimation - > showWeapons ( false ) ;
2018-07-17 10:06:41 +00:00
}
2018-05-02 10:13:49 +00:00
if ( ! downSoundId . empty ( ) )
2015-12-26 17:45:09 +00:00
{
2018-05-02 10:13:49 +00:00
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
sndMgr - > playSound3D ( mPtr , downSoundId , 1.0f , 1.0f ) ;
2015-12-26 17:45:09 +00:00
}
2013-07-23 10:26:24 +00:00
}
2018-05-02 10:13:49 +00:00
float complete ;
bool animPlaying = mAnimation - > getInfo ( mCurrentWeapon , & complete ) ;
if ( ! animPlaying | | complete > = 1.0f )
2013-07-23 10:26:24 +00:00
{
2018-07-30 18:24:25 +00:00
// Weapon is changed, no current animation (e.g. unequipping or attack).
// Start equipping animation now.
if ( weaptype ! = mWeaponType )
{
forcestateupdate = true ;
2019-08-11 11:01:48 +00:00
bool useShieldAnims = mAnimation - > useShieldAnimations ( ) ;
if ( ! useShieldAnims )
mAnimation - > showCarriedLeft ( updateCarriedLeftVisible ( weaptype ) ) ;
2019-08-09 08:58:20 +00:00
weapgroup = getWeaponAnimation ( weaptype ) ;
2018-05-02 10:13:49 +00:00
2019-03-05 09:33:58 +00:00
// Note: controllers for ranged weapon should use time for beginning of animation to play shooting properly,
// for other weapons they should use absolute time. Some mods rely on this behaviour (to rotate throwing projectiles, for example)
2018-12-26 09:45:28 +00:00
ESM : : WeaponType : : Class weaponClass = getWeaponType ( weaptype ) - > mWeaponClass ;
bool useRelativeDuration = weaponClass = = ESM : : WeaponType : : Ranged ;
2019-03-05 09:33:58 +00:00
mAnimation - > setWeaponGroup ( weapgroup , useRelativeDuration ) ;
2014-02-23 19:11:05 +00:00
2018-07-30 18:24:25 +00:00
if ( ! isStillWeapon )
2018-05-02 10:13:49 +00:00
{
mAnimation - > disable ( mCurrentWeapon ) ;
2018-12-26 09:45:28 +00:00
if ( weaptype ! = ESM : : Weapon : : None )
2018-07-30 18:24:25 +00:00
{
mAnimation - > showWeapons ( false ) ;
2019-08-11 11:01:48 +00:00
int equipMask = MWRender : : Animation : : BlendMask_All ;
if ( useShieldAnims & & weaptype ! = ESM : : Weapon : : Spell )
{
equipMask = equipMask | ~ MWRender : : Animation : : BlendMask_LeftArm ;
mAnimation - > play ( " shield " , Priority_Block ,
MWRender : : Animation : : BlendMask_LeftArm , true ,
1.0f , " equip start " , " equip stop " , 0.0f , 0 ) ;
}
mAnimation - > play ( weapgroup , priorityWeapon , equipMask , true ,
2018-07-30 18:24:25 +00:00
1.0f , " equip start " , " equip stop " , 0.0f , 0 ) ;
mUpperBodyState = UpperCharState_EquipingWeap ;
2018-07-19 12:38:32 +00:00
// If we do not have the "equip attach" key, show weapon manually.
2018-12-26 09:45:28 +00:00
if ( weaptype ! = ESM : : Weapon : : Spell )
2018-07-19 12:38:32 +00:00
{
if ( mAnimation - > getTextKeyTime ( weapgroup + " : equip attach " ) < 0 )
mAnimation - > showWeapons ( true ) ;
}
2018-07-30 18:24:25 +00:00
}
2018-05-02 10:13:49 +00:00
}
2018-07-30 18:24:25 +00:00
if ( isWerewolf )
2018-05-02 10:13:49 +00:00
{
2018-07-30 18:24:25 +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 ) ;
}
2018-05-02 10:13:49 +00:00
}
2013-11-23 19:24:52 +00:00
2018-07-30 18:24:25 +00:00
mWeaponType = weaptype ;
2019-08-09 08:58:20 +00:00
mCurrentWeapon = getWeaponAnimation ( mWeaponType ) ;
2018-07-30 18:24:25 +00:00
if ( ! upSoundId . empty ( ) & & ! isStillWeapon )
2013-08-11 07:35:19 +00:00
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
2018-07-30 18:24:25 +00:00
sndMgr - > playSound3D ( mPtr , upSoundId , 1.0f , 1.0f ) ;
2013-08-11 07:35:19 +00:00
}
2013-08-09 13:40:16 +00:00
}
2013-07-23 10:26:24 +00:00
2018-07-30 18:24:25 +00:00
// Make sure that we disabled unequipping animation
if ( mUpperBodyState = = UpperCharState_UnEquipingWeap )
2018-05-02 10:13:49 +00:00
{
2018-07-30 18:24:25 +00:00
mUpperBodyState = UpperCharState_Nothing ;
mAnimation - > disable ( mCurrentWeapon ) ;
2018-12-26 09:45:28 +00:00
mWeaponType = ESM : : Weapon : : None ;
2019-08-09 08:58:20 +00:00
mCurrentWeapon = getWeaponAnimation ( mWeaponType ) ;
2018-05-02 10:13:49 +00:00
}
}
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 )
2018-12-26 09:45:28 +00:00
& & mWeaponType = = ESM : : Weapon : : None )
2013-08-09 13:40:16 +00:00
{
if ( ! sndMgr - > getSoundPlaying ( mPtr , " WolfRun " ) )
2017-09-15 08:03:41 +00:00
sndMgr - > playSound3D ( mPtr , " WolfRun " , 1.0f , 1.0f , MWSound : : Type : : Sfx ,
MWSound : : PlayMode : : Loop ) ;
2013-08-09 13:40:16 +00:00
}
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 ) ;
2018-12-26 09:45:28 +00:00
MWWorld : : ConstContainerStoreIterator weapon = getActiveWeapon ( mPtr , & weaptype ) ;
2014-12-31 15:59:21 +00:00
isWeapon = ( weapon ! = inv . end ( ) & & weapon - > getTypeName ( ) = = typeid ( ESM : : Weapon ) . name ( ) ) ;
2018-12-26 09:45:28 +00:00
if ( isWeapon )
{
2014-12-31 15:59:21 +00:00
weapSpeed = weapon - > get < ESM : : Weapon > ( ) - > mBase - > mData . mSpeed ;
2018-12-26 09:45:28 +00:00
MWWorld : : ConstContainerStoreIterator ammo = inv . getSlot ( MWWorld : : InventoryStore : : Slot_Ammunition ) ;
int ammotype = getWeaponType ( weapon - > get < ESM : : Weapon > ( ) - > mBase - > mData . mType ) - > mAmmoType ;
if ( ammotype ! = ESM : : Weapon : : None & & ( ammo = = inv . end ( ) | | ammo - > get < ESM : : Weapon > ( ) - > mBase - > mData . mType ! = ammotype ) )
ammunition = false ;
}
2014-12-31 15:59:21 +00:00
if ( ! ammunition & & mUpperBodyState > UpperCharState_WeapEquiped )
{
mAnimation - > disable ( mCurrentWeapon ) ;
mUpperBodyState = UpperCharState_WeapEquiped ;
}
2014-02-04 03:05:01 +00:00
}
2018-06-12 08:55:28 +00:00
// Combat for actors with persistent animations obviously will be buggy
if ( isPersistentAnimPlaying ( ) )
return forcestateupdate ;
2013-07-23 10:26:24 +00:00
float complete ;
2013-07-23 18:12:19 +00:00
bool animPlaying ;
2018-12-26 09:45:28 +00:00
ESM : : WeaponType : : Class weapclass = getWeaponType ( mWeaponType ) - > mWeaponClass ;
2015-07-02 17:14:28 +00:00
if ( mAttackingOrSpell )
2013-07-23 10:26:24 +00:00
{
2017-11-24 14:25:57 +00:00
MWWorld : : Ptr player = getPlayer ( ) ;
2018-07-25 14:50:45 +00:00
bool resetIdle = ammunition ;
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 ) ;
2017-09-16 23:04:54 +00:00
mAttackStrength = 0 ;
2018-07-19 12:38:32 +00:00
2018-08-20 17:52:05 +00:00
// Randomize attacks for non-bipedal creatures with Weapon flag
if ( mPtr . getClass ( ) . getTypeName ( ) = = typeid ( ESM : : Creature ) . name ( ) & &
! mPtr . getClass ( ) . isBipedal ( mPtr ) & &
( ! mAnimation - > hasAnimation ( mCurrentWeapon ) | | isRandomAttackAnimation ( mCurrentWeapon ) ) )
{
mCurrentWeapon = chooseRandomAttackAnimation ( ) ;
}
2018-12-26 09:45:28 +00:00
if ( mWeaponType = = ESM : : Weapon : : 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 ;
2017-11-24 14:25:57 +00:00
if ( mPtr = = player )
2015-07-16 17:56:09 +00:00
{
MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . setAttackingOrSpell ( false ) ;
2013-07-23 18:12:19 +00:00
2018-08-15 10:16:19 +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)
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 ( ) ;
2019-04-08 08:07:44 +00:00
bool isMagicItem = false ;
2018-06-28 12:58:51 +00:00
bool canCast = mCastingManualSpell | | MWBase : : Environment : : get ( ) . getWorld ( ) - > startSpellCast ( mPtr ) ;
2013-12-26 21:32:39 +00:00
2019-04-08 08:07:44 +00:00
if ( spellid . empty ( ) )
{
if ( mPtr . getClass ( ) . hasInventoryStore ( mPtr ) )
{
MWWorld : : InventoryStore & inv = mPtr . getClass ( ) . getInventoryStore ( mPtr ) ;
if ( inv . getSelectedEnchantItem ( ) ! = inv . end ( ) )
{
const MWWorld : : Ptr & enchantItem = * inv . getSelectedEnchantItem ( ) ;
spellid = enchantItem . getClass ( ) . getEnchantment ( enchantItem ) ;
isMagicItem = true ;
}
}
}
static const bool useCastingAnimations = Settings : : Manager : : getBool ( " use magic item animations " , " Game " ) ;
if ( isMagicItem & & ! useCastingAnimations )
{
// Enchanted items by default do not use casting animations
MWBase : : Environment : : get ( ) . getWorld ( ) - > castSpell ( mPtr ) ;
resetIdle = false ;
}
else if ( ! spellid . empty ( ) & & canCast )
2013-07-23 14:30:54 +00:00
{
2018-10-09 06:21:12 +00:00
MWMechanics : : CastSpell cast ( mPtr , nullptr , false , mCastingManualSpell ) ;
2019-04-08 08:07:44 +00:00
cast . playSpellCastingEffects ( spellid , isMagicItem ) ;
2013-07-23 18:12:19 +00:00
2019-04-08 08:07:44 +00:00
std : : vector < ESM : : ENAMstruct > effects ;
2018-08-15 10:16:19 +00:00
const MWWorld : : ESMStore & store = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
2019-04-08 08:07:44 +00:00
if ( isMagicItem )
{
const ESM : : Enchantment * enchantment = store . get < ESM : : Enchantment > ( ) . find ( spellid ) ;
effects = enchantment - > mEffects . mList ;
}
else
{
const ESM : : Spell * spell = store . get < ESM : : Spell > ( ) . find ( spellid ) ;
effects = spell - > mEffects . mList ;
}
2016-09-02 11:10:13 +00:00
2019-04-08 08:07:44 +00:00
const ESM : : MagicEffect * effect = store . get < ESM : : MagicEffect > ( ) . find ( effects . back ( ) . 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
2019-04-08 08:07:44 +00:00
for ( size_t iter = 0 ; iter < effects . 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 ) ;
}
2019-04-08 08:07:44 +00:00
const ESM : : ENAMstruct & firstEffect = effects . at ( 0 ) ; // first effect used for casting animation
2013-11-17 22:15:57 +00:00
2018-07-19 12:38:32 +00:00
std : : string startKey ;
std : : string stopKey ;
if ( isRandomAttackAnimation ( mCurrentWeapon ) )
{
startKey = " start " ;
stopKey = " stop " ;
MWBase : : Environment : : get ( ) . getWorld ( ) - > castSpell ( mPtr , mCastingManualSpell ) ; // No "release" text key to use, so cast immediately
mCastingManualSpell = false ;
}
else
2014-01-26 21:32:57 +00:00
{
2018-07-19 12:38:32 +00:00
switch ( firstEffect . mRange )
{
case 0 : mAttackType = " self " ; break ;
case 1 : mAttackType = " touch " ; break ;
case 2 : mAttackType = " target " ; break ;
}
startKey = mAttackType + " start " ;
stopKey = mAttackType + " stop " ;
2014-01-26 21:32:57 +00:00
}
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 ,
2018-07-19 12:38:32 +00:00
1 , startKey , stopKey ,
2013-07-23 14:30:54 +00:00
0.0f , 0 ) ;
mUpperBodyState = UpperCharState_CastingSpell ;
}
2018-07-25 14:50:45 +00:00
else
{
resetIdle = false ;
}
2013-07-23 13:13:08 +00:00
}
2018-12-26 09:45:28 +00:00
else if ( mWeaponType = = ESM : : Weapon : : PickProbe )
2013-08-03 01:21:42 +00:00
{
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 ;
2020-05-17 14:25:51 +00:00
std : : string resultMessage , resultSound ;
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.
2020-05-17 14:25:51 +00:00
# ifdef USE_OPENXR
auto * anim = MWVR : : Environment : : get ( ) . getPlayerAnimation ( ) ;
auto target = anim - > getTarget ( " weapon bone " ) ;
# else
2013-08-03 01:21:42 +00:00
MWWorld : : Ptr target = MWBase : : Environment : : get ( ) . getWorld ( ) - > getFacedObject ( ) ;
2020-05-17 14:25:51 +00:00
# endif
2013-08-03 01:21:42 +00:00
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 ( ) )
2017-09-15 08:03:41 +00:00
MWBase : : Environment : : get ( ) . getSoundManager ( ) - > playSound3D ( target , resultSound ,
1.0f , 1.0f ) ;
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
{
2018-07-19 12:38:32 +00:00
std : : string startKey ;
std : : string stopKey ;
2018-12-26 09:45:28 +00:00
if ( weapclass = = ESM : : WeaponType : : Ranged | | weapclass = = ESM : : WeaponType : : Thrown )
2018-07-19 12:38:32 +00:00
{
2013-07-23 14:30:54 +00:00
mAttackType = " shoot " ;
2018-07-19 12:38:32 +00:00
startKey = mAttackType + " start " ;
stopKey = mAttackType + " min attack " ;
}
2018-08-15 10:16:19 +00:00
else if ( isRandomAttackAnimation ( mCurrentWeapon ) )
2018-07-19 12:38:32 +00:00
{
startKey = " start " ;
stopKey = " stop " ;
}
2018-08-15 10:16:19 +00:00
else
2013-07-23 14:30:54 +00:00
{
2016-09-19 17:13:10 +00:00
if ( mPtr = = getPlayer ( ) )
2014-12-31 15:59:21 +00:00
{
2019-03-31 19:07:24 +00:00
if ( Settings : : Manager : : getBool ( " best attack " , " Game " ) )
2016-03-19 17:03:59 +00:00
{
2019-03-31 19:07:24 +00:00
if ( isWeapon )
2016-09-01 13:43:33 +00:00
{
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
2019-03-31 19:07:24 +00:00
{
// There is no "best attack" for Hand-to-Hand
setAttackTypeRandomly ( mAttackType ) ;
}
2016-03-19 17:03:59 +00:00
}
2016-09-19 17:13:10 +00:00
else
2018-07-19 12:38:32 +00:00
{
2019-03-31 19:07:24 +00:00
setAttackTypeBasedOnMovement ( ) ;
2018-07-19 12:38:32 +00:00
}
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
2018-07-19 12:38:32 +00:00
startKey = mAttackType + " start " ;
stopKey = mAttackType + " min attack " ;
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 ,
2018-07-19 12:38:32 +00:00
weapSpeed , startKey , stopKey ,
2013-07-23 14:30:54 +00:00
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
2018-12-09 21:04:12 +00:00
// We should not break swim and sneak animations
2018-11-23 16:07:52 +00:00
if ( resetIdle & &
idle ! = CharState_IdleSneak & & idle ! = CharState_IdleSwim & &
mIdleState ! = CharState_IdleSneak & & mIdleState ! = CharState_IdleSwim )
{
2019-04-28 17:50:31 +00:00
mAnimation - > disable ( mCurrentIdle ) ;
2019-12-03 17:50:57 +00:00
mIdleState = CharState_None ;
2018-11-23 16:07:52 +00:00
}
2018-07-25 14:50:45 +00:00
2013-07-23 18:12:19 +00:00
animPlaying = mAnimation - > getInfo ( mCurrentWeapon , & complete ) ;
2017-09-22 11:26:35 +00:00
if ( mUpperBodyState = = UpperCharState_MinAttackToMaxAttack & & ! isKnockedDown ( ) )
2015-07-03 03:58:12 +00:00
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 ) ;
2017-09-22 11:26:35 +00:00
if ( mUpperBodyState = = UpperCharState_MinAttackToMaxAttack & & ! isKnockedDown ( ) )
2013-07-23 18:12:19 +00:00
{
2014-09-07 02:46:47 +00:00
float attackStrength = complete ;
2019-09-24 19:42:04 +00:00
float minAttackTime = mAnimation - > getTextKeyTime ( mCurrentWeapon + " : " + mAttackType + " " + " min attack " ) ;
float maxAttackTime = mAnimation - > getTextKeyTime ( mCurrentWeapon + " : " + mAttackType + " " + " max attack " ) ;
if ( minAttackTime = = maxAttackTime )
2014-09-07 02:46:47 +00:00
{
// 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
}
2018-12-26 09:45:28 +00:00
if ( weapclass ! = ESM : : WeaponType : : Ranged & & weapclass ! = ESM : : WeaponType : : Thrown )
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 ;
}
2017-09-22 11:26:35 +00:00
else if ( isKnockedDown ( ) )
2014-01-08 14:05:14 +00:00
{
2014-12-05 21:02:18 +00:00
if ( mUpperBodyState > UpperCharState_WeapEquiped )
2019-08-09 05:05:59 +00:00
{
2014-12-05 21:02:18 +00:00
mUpperBodyState = UpperCharState_WeapEquiped ;
2018-12-26 09:45:28 +00:00
if ( mWeaponType > ESM : : Weapon : : None )
2019-08-09 05:05:59 +00:00
mAnimation - > showWeapons ( true ) ;
}
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 ) ;
2018-12-26 09:45:28 +00:00
if ( weapclass = = ESM : : WeaponType : : Ranged | | weapclass = = ESM : : WeaponType : : Thrown )
2014-02-04 03:00:52 +00:00
{
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 )
2018-10-09 07:29:57 +00:00
{
// technically we do not need a pitch for crossbow reload animation,
// but we should avoid abrupt repositioning
2018-12-26 09:45:28 +00:00
if ( mWeaponType = = ESM : : Weapon : : MarksmanCrossbow )
2018-10-09 07:29:57 +00:00
mAnimation - > setPitchFactor ( std : : max ( 0.f , 1.f - complete * 10.f ) ) ;
else
mAnimation - > setPitchFactor ( 1.f - complete ) ;
}
2014-02-04 03:00:52 +00:00
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
{
2018-12-26 09:45:28 +00:00
if ( ammunition & & mWeaponType = = ESM : : Weapon : : MarksmanCrossbow )
2014-02-04 03:11:46 +00:00
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
}
2018-07-19 12:38:32 +00:00
else if ( complete > = 1.0f & & ! isRandomAttackAnimation ( mCurrentWeapon ) )
2013-07-23 10:26:24 +00:00
{
2014-01-01 19:40:31 +00:00
std : : string start , stop ;
switch ( mUpperBodyState )
2013-07-23 10:26:24 +00:00
{
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 ;
2018-08-15 13:33:27 +00:00
case UpperCharState_StartToMinAttack :
2019-11-19 11:34:21 +00:00
case UpperCharState_MaxAttackToMinHit :
2018-08-15 13:33:27 +00:00
{
2019-11-19 11:34:21 +00:00
if ( mUpperBodyState = = UpperCharState_StartToMinAttack )
2018-08-15 13:33:27 +00:00
{
2019-11-19 11:34:21 +00:00
// If actor is already stopped preparing attack, do not play the "min attack -> max attack" part.
// Happens if the player did not hold the attack button.
// Note: if the "min attack"->"max attack" is a stub, "play" it anyway. Attack strength will be random.
float minAttackTime = mAnimation - > getTextKeyTime ( mCurrentWeapon + " : " + mAttackType + " " + " min attack " ) ;
float maxAttackTime = mAnimation - > getTextKeyTime ( mCurrentWeapon + " : " + mAttackType + " " + " max attack " ) ;
if ( mAttackingOrSpell | | minAttackTime = = maxAttackTime )
{
start = mAttackType + " min attack " ;
stop = mAttackType + " max attack " ;
mUpperBodyState = UpperCharState_MinAttackToMaxAttack ;
break ;
}
2020-09-03 05:11:22 +00:00
if ( weapclass ! = ESM : : WeaponType : : Ranged & & weapclass ! = ESM : : WeaponType : : Thrown )
playSwishSound ( 0.0f ) ;
2018-08-15 13:33:27 +00:00
}
2019-11-19 11:34:21 +00:00
2014-01-01 19:40:31 +00:00
if ( mAttackType = = " shoot " )
{
start = mAttackType + " min hit " ;
stop = mAttackType + " release " ;
}
else
{
start = mAttackType + " min hit " ;
stop = mAttackType + " hit " ;
}
mUpperBodyState = UpperCharState_MinHitToHit ;
break ;
2019-11-19 11:34:21 +00:00
}
2014-01-01 19:40:31 +00:00
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
2018-07-17 16:59:05 +00:00
// Note: apply crossbow reload animation only for upper body
// since blending with movement animations can give weird result.
2014-01-01 19:40:31 +00:00
if ( ! start . empty ( ) )
2013-07-23 10:26:24 +00:00
{
2018-07-17 16:59:05 +00:00
int mask = MWRender : : Animation : : BlendMask_All ;
2018-12-26 09:45:28 +00:00
if ( mWeaponType = = ESM : : Weapon : : MarksmanCrossbow )
2018-07-17 16:59:05 +00:00
mask = MWRender : : Animation : : BlendMask_UpperBody ;
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 ,
2018-07-17 16:59:05 +00:00
mask , 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 ,
2018-07-17 16:59:05 +00:00
mask , false ,
2013-08-02 07:20:12 +00:00
weapSpeed , start , stop , 0.0f , 0 ) ;
2013-07-23 10:26:24 +00:00
}
}
2018-07-19 12:38:32 +00:00
else if ( complete > = 1.0f & & isRandomAttackAnimation ( mCurrentWeapon ) )
{
mAnimation - > disable ( mCurrentWeapon ) ;
mUpperBodyState = UpperCharState_WeapEquiped ;
}
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
}
2018-09-21 12:34:23 +00:00
void CharacterController : : update ( float duration , bool animationOnly )
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 ( ) ;
2017-11-24 14:25:57 +00:00
2017-11-04 19:37:20 +00:00
if ( isKnockedOut ( ) )
2017-11-05 18:30:34 +00:00
mTimeUntilWake - = duration ;
2016-08-06 18:08:46 +00:00
2018-09-15 15:38:21 +00:00
bool isPlayer = mPtr = = MWMechanics : : getPlayer ( ) ;
2020-07-09 06:47:37 +00:00
bool isFirstPersonPlayer = isPlayer & & MWBase : : Environment : : get ( ) . getWorld ( ) - > isFirstPerson ( ) ;
2018-09-15 15:38:21 +00:00
bool godmode = isPlayer & & MWBase : : Environment : : get ( ) . getWorld ( ) - > getGodModeState ( ) ;
2017-03-25 18:40:11 +00:00
2019-11-12 14:10:28 +00:00
float scale = mPtr . getCellRef ( ) . getScale ( ) ;
static const bool normalizeSpeed = Settings : : Manager : : getBool ( " normalise race speed " , " Game " ) ;
if ( ! normalizeSpeed & & mPtr . getClass ( ) . isNpc ( ) )
{
const ESM : : NPC * npc = mPtr . get < ESM : : NPC > ( ) - > mBase ;
const ESM : : Race * race = world - > getStore ( ) . get < ESM : : Race > ( ) . find ( npc - > mRace ) ;
float weight = npc - > isMale ( ) ? race - > mData . mWeight . mMale : race - > mData . mWeight . mFemale ;
scale * = weight ;
}
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 ) ;
2020-10-09 17:34:36 +00:00
bool incapacitated = ( ( ! godmode & & cls . getCreatureStats ( mPtr ) . isParalyzed ( ) ) | | cls . getCreatureStats ( mPtr ) . getKnockedDown ( ) ) ;
2013-02-18 14:29:16 +00:00
bool inwater = world - > isSwimming ( mPtr ) ;
2013-08-18 12:59:06 +00:00
bool flying = world - > isFlying ( mPtr ) ;
2018-12-05 20:28:26 +00:00
bool solid = world - > isActorCollisionEnabled ( mPtr ) ;
2018-08-28 11:28:27 +00:00
// Can't run and sneak while flying (see speed formula in Npc/Creature::getSpeed)
bool sneak = cls . getCreatureStats ( mPtr ) . getStance ( MWMechanics : : CreatureStats : : Stance_Sneak ) & & ! flying ;
2014-10-14 23:06:16 +00:00
bool isrunning = cls . getCreatureStats ( mPtr ) . getStance ( MWMechanics : : CreatureStats : : Stance_Run ) & & ! flying ;
2014-08-03 05:42:40 +00:00
CreatureStats & stats = cls . getCreatureStats ( mPtr ) ;
2020-06-22 00:03:38 +00:00
Movement & movementSettings = cls . getMovementSettings ( mPtr ) ;
2014-08-03 05:42:40 +00:00
//Force Jump Logic
2020-06-22 00:03:38 +00:00
bool isMoving = ( std : : abs ( movementSettings . mPosition [ 0 ] ) > .5 | | std : : abs ( movementSettings . mPosition [ 1 ] ) > .5 ) ;
2018-12-05 20:28:26 +00:00
if ( ! inwater & & ! flying & & solid )
2014-08-03 05:42:40 +00:00
{
//Force Jump
if ( stats . getMovementFlag ( MWMechanics : : CreatureStats : : Flag_ForceJump ) )
2020-06-22 00:03:38 +00:00
movementSettings . mPosition [ 2 ] = onground ? 1 : 0 ;
2014-08-03 05:42:40 +00:00
//Force Move Jump, only jump if they're otherwise moving
if ( stats . getMovementFlag ( MWMechanics : : CreatureStats : : Flag_ForceMoveJump ) & & isMoving )
2020-06-22 00:03:38 +00:00
movementSettings . mPosition [ 2 ] = onground ? 1 : 0 ;
2014-08-03 05:42:40 +00:00
}
2015-06-03 17:41:19 +00:00
osg : : Vec3f rot = cls . getRotationVector ( mPtr ) ;
2020-06-22 00:03:38 +00:00
osg : : Vec3f vec ( movementSettings . asVec3 ( ) ) ;
2020-08-31 21:16:10 +00:00
movementSettings . mSpeedFactor = std : : min ( vec . length ( ) , 1.f ) ;
2020-06-22 00:03:38 +00:00
vec . normalize ( ) ;
2014-04-25 20:20:55 +00:00
2020-08-31 21:16:10 +00:00
// TODO: Move this check to mwinput.
// Joystick analogue movement.
// Due to the half way split between walking/running, we multiply speed by 2 while walking, unless a keyboard was used.
if ( isPlayer & & ! isrunning & & ! sneak & & ! flying & & movementSettings . mSpeedFactor < = 0.5f )
movementSettings . mSpeedFactor * = 2.f ;
2020-09-04 13:03:33 +00:00
static const bool smoothMovement = Settings : : Manager : : getBool ( " smooth movement " , " Game " ) ;
if ( smoothMovement & & ! isFirstPersonPlayer )
{
2020-10-17 14:11:22 +00:00
static const float playerTurningCoef = 1.0 / std : : max ( 0.01f , Settings : : Manager : : getFloat ( " smooth movement player turning delay " , " Game " ) ) ;
2020-09-04 13:03:33 +00:00
float angle = mPtr . getRefData ( ) . getPosition ( ) . rot [ 2 ] ;
osg : : Vec2f targetSpeed = Misc : : rotateVec2f ( osg : : Vec2f ( vec . x ( ) , vec . y ( ) ) , - angle ) * movementSettings . mSpeedFactor ;
osg : : Vec2f delta = targetSpeed - mSmoothedSpeed ;
float speedDelta = movementSettings . mSpeedFactor - mSmoothedSpeed . length ( ) ;
float deltaLen = delta . length ( ) ;
float maxDelta ;
if ( std : : abs ( speedDelta ) < deltaLen / 2 )
// Turning is smooth for player and less smooth for NPCs (otherwise NPC can miss a path point).
2020-10-17 14:11:22 +00:00
maxDelta = duration * ( isPlayer ? playerTurningCoef : 6.f ) ;
2020-09-04 13:03:33 +00:00
else if ( isPlayer & & speedDelta < - deltaLen / 2 )
// As soon as controls are released, mwinput switches player from running to walking.
// So stopping should be instant for player, otherwise it causes a small twitch.
maxDelta = 1 ;
else // In all other cases speeding up and stopping are smooth.
maxDelta = duration * 3.f ;
if ( deltaLen > maxDelta )
delta * = maxDelta / deltaLen ;
mSmoothedSpeed + = delta ;
osg : : Vec2f newSpeed = Misc : : rotateVec2f ( mSmoothedSpeed , angle ) ;
movementSettings . mSpeedFactor = newSpeed . normalize ( ) ;
vec . x ( ) = newSpeed . x ( ) ;
vec . y ( ) = newSpeed . y ( ) ;
const float eps = 0.001f ;
if ( movementSettings . mSpeedFactor < eps )
{
movementSettings . mSpeedFactor = 0 ;
vec . x ( ) = 0 ;
vec . y ( ) = 1 ;
}
else if ( ( vec . y ( ) < 0 ) ! = mIsMovingBackward )
{
if ( targetSpeed . length ( ) < eps | | ( movementSettings . mPosition [ 1 ] < 0 ) = = mIsMovingBackward )
vec . y ( ) = mIsMovingBackward ? - eps : eps ;
}
vec . normalize ( ) ;
2019-03-02 23:46:48 +00:00
}
2013-02-06 03:05:07 +00:00
2020-06-22 00:03:38 +00:00
float effectiveRotation = rot . z ( ) ;
2020-08-31 22:37:37 +00:00
bool canMove = cls . getMaxSpeed ( mPtr ) > 0 ;
2020-06-22 00:03:38 +00:00
static const bool turnToMovementDirection = Settings : : Manager : : getBool ( " turn to movement direction " , " Game " ) ;
2020-08-31 22:37:37 +00:00
if ( ! turnToMovementDirection | | isFirstPersonPlayer )
movementSettings . mIsStrafing = std : : abs ( vec . x ( ) ) > std : : abs ( vec . y ( ) ) * 2 ;
else if ( canMove )
2020-06-22 00:03:38 +00:00
{
float targetMovementAngle = vec . y ( ) > = 0 ? std : : atan2 ( - vec . x ( ) , vec . y ( ) ) : std : : atan2 ( vec . x ( ) , - vec . y ( ) ) ;
movementSettings . mIsStrafing = ( stats . getDrawState ( ) ! = MWMechanics : : DrawState_Nothing | | inwater )
& & std : : abs ( targetMovementAngle ) > osg : : DegreesToRadians ( 60.0f ) ;
if ( movementSettings . mIsStrafing )
targetMovementAngle = 0 ;
float delta = targetMovementAngle - stats . getSideMovementAngle ( ) ;
float cosDelta = cosf ( delta ) ;
2020-06-26 20:04:02 +00:00
if ( ( vec . y ( ) < 0 ) = = mIsMovingBackward )
movementSettings . mSpeedFactor * = std : : min ( std : : max ( cosDelta , 0.f ) + 0.3f , 1.f ) ; // slow down when turn
if ( std : : abs ( delta ) < osg : : DegreesToRadians ( 20.0f ) )
mIsMovingBackward = vec . y ( ) < 0 ;
float maxDelta = osg : : PI * duration * ( 2.5f - cosDelta ) ;
delta = osg : : clampBetween ( delta , - maxDelta , maxDelta ) ;
2020-06-22 00:03:38 +00:00
stats . setSideMovementAngle ( stats . getSideMovementAngle ( ) + delta ) ;
effectiveRotation + = delta ;
2019-03-02 23:46:48 +00:00
}
2020-06-22 00:03:38 +00:00
mAnimation - > setLegsYawRadians ( stats . getSideMovementAngle ( ) ) ;
if ( stats . getDrawState ( ) = = MWMechanics : : DrawState_Nothing | | inwater )
mAnimation - > setUpperBodyYawRadians ( stats . getSideMovementAngle ( ) / 2 ) ;
else
mAnimation - > setUpperBodyYawRadians ( stats . getSideMovementAngle ( ) / 4 ) ;
2020-09-15 21:25:32 +00:00
if ( smoothMovement & & ! isPlayer & & ! inwater )
mAnimation - > setUpperBodyYawRadians ( mAnimation - > getUpperBodyYawRadians ( ) + mAnimation - > getHeadYaw ( ) / 2 ) ;
2013-02-06 03:05:07 +00:00
2020-08-27 11:48:59 +00:00
speed = cls . getCurrentSpeed ( mPtr ) ;
2015-07-25 16:22:48 +00:00
vec . x ( ) * = speed ;
vec . y ( ) * = speed ;
2013-04-28 05:53:04 +00:00
2020-06-22 00:03:38 +00:00
if ( mHitState ! = CharState_None & & mJumpState = = JumpState_None )
vec = osg : : Vec3f ( ) ;
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-08-09 13:53:07 +00:00
// advance athletics
2018-09-15 15:38:21 +00:00
if ( mHasMovedInXY & & isPlayer )
2013-08-09 13:53:07 +00:00
{
if ( inwater )
{
mSecondsOfSwimming + = duration ;
while ( mSecondsOfSwimming > 1 )
{
cls . skillUsageSucceeded ( mPtr , ESM : : Skill : : Athletics , 1 ) ;
mSecondsOfSwimming - = 1 ;
}
}
2017-08-16 16:30:47 +00:00
else if ( isrunning & & ! sneak )
2013-08-09 13:53:07 +00:00
{
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 ;
2018-08-29 15:38:12 +00:00
static const float fFatigueRunBase = gmst . find ( " fFatigueRunBase " ) - > mValue . getFloat ( ) ;
static const float fFatigueRunMult = gmst . find ( " fFatigueRunMult " ) - > mValue . getFloat ( ) ;
static const float fFatigueSwimWalkBase = gmst . find ( " fFatigueSwimWalkBase " ) - > mValue . getFloat ( ) ;
static const float fFatigueSwimRunBase = gmst . find ( " fFatigueSwimRunBase " ) - > mValue . getFloat ( ) ;
static const float fFatigueSwimWalkMult = gmst . find ( " fFatigueSwimWalkMult " ) - > mValue . getFloat ( ) ;
static const float fFatigueSwimRunMult = gmst . find ( " fFatigueSwimRunMult " ) - > mValue . getFloat ( ) ;
static const float fFatigueSneakBase = gmst . find ( " fFatigueSneakBase " ) - > mValue . getFloat ( ) ;
static const float fFatigueSneakMult = gmst . find ( " fFatigueSneakMult " ) - > mValue . getFloat ( ) ;
2014-01-11 21:26:26 +00:00
2017-07-25 06:28:33 +00:00
if ( cls . getEncumbrance ( mPtr ) < = cls . getCapacity ( mPtr ) )
2014-01-11 21:26:26 +00:00
{
2018-05-12 16:06:18 +00:00
const float encumbrance = cls . getNormalizedEncumbrance ( mPtr ) ;
2014-01-11 21:26:26 +00:00
if ( sneak )
fatigueLoss = fFatigueSneakBase + encumbrance * fFatigueSneakMult ;
else
{
if ( inwater )
{
if ( ! isrunning )
fatigueLoss = fFatigueSwimWalkBase + encumbrance * fFatigueSwimWalkMult ;
else
fatigueLoss = fFatigueSwimRunBase + encumbrance * fFatigueSwimRunMult ;
}
2017-08-16 16:30:47 +00:00
else if ( isrunning )
2014-01-11 21:26:26 +00:00
fatigueLoss = fFatigueRunBase + encumbrance * fFatigueRunMult ;
}
}
fatigueLoss * = duration ;
2020-08-27 11:48:59 +00:00
fatigueLoss * = movementSettings . mSpeedFactor ;
2014-01-11 21:26:26 +00:00
DynamicStat < float > fatigue = cls . getCreatureStats ( mPtr ) . getFatigue ( ) ;
2017-03-25 18:40:11 +00:00
if ( ! godmode )
{
fatigue . setCurrent ( fatigue . getCurrent ( ) - fatigueLoss , fatigue . getCurrent ( ) < 0 ) ;
cls . getCreatureStats ( mPtr ) . setFatigue ( fatigue ) ;
}
2014-01-11 21:26:26 +00:00
2019-07-30 19:07:55 +00:00
float z = cls . getJump ( mPtr ) ;
if ( sneak | | inwater | | flying | | incapacitated | | ! solid | | z < = 0 )
2015-05-12 01:02:15 +00:00
vec . z ( ) = 0.0f ;
2013-08-18 12:59:06 +00:00
2014-07-03 15:40:44 +00:00
bool inJump = true ;
2018-11-23 16:07:52 +00:00
bool playLandingSound = false ;
2018-12-05 20:28:26 +00:00
if ( ! onground & & ! flying & & ! inwater & & solid )
2013-08-18 12:59:06 +00:00
{
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
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
2018-08-29 15:38:12 +00:00
static const float fJumpMoveBase = gmst . find ( " fJumpMoveBase " ) - > mValue . getFloat ( ) ;
static const float fJumpMoveMult = gmst . find ( " fJumpMoveMult " ) - > mValue . getFloat ( ) ;
2015-01-13 02:11:29 +00:00
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
}
2017-12-31 23:48:51 +00:00
else if ( vec . z ( ) > 0.0f & & mJumpState ! = JumpState_InAir )
2013-08-18 12:59:06 +00:00
{
2013-11-16 01:11:11 +00:00
// Started a jump.
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
}
2018-12-05 20:28:26 +00:00
else if ( mJumpState = = JumpState_InAir & & ! inwater & & ! flying & & solid )
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
2019-04-28 17:50:31 +00:00
// We should reset idle animation during landing
mAnimation - > disable ( mCurrentIdle ) ;
2018-09-15 15:38:21 +00:00
float height = cls . getCreatureStats ( mPtr ) . land ( isPlayer ) ;
2014-12-10 16:21:34 +00:00
float healthLost = getFallDamage ( mPtr , height ) ;
2017-03-25 18:40:11 +00:00
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
2017-03-25 18:40:11 +00:00
if ( ! godmode )
{
float realHealthLost = static_cast < float > ( healthLost * ( 1.0f - 0.25f * fatigueTerm ) ) ;
cls . onHit ( mPtr , realHealthLost , true , MWWorld : : Ptr ( ) , MWWorld : : Ptr ( ) , osg : : Vec3f ( ) , true ) ;
}
2013-10-01 21:35:34 +00:00
2018-12-23 11:18:33 +00:00
const float 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
{
2018-08-28 13:42:15 +00:00
if ( ! godmode )
cls . getCreatureStats ( mPtr ) . setKnockedDown ( true ) ;
2014-01-15 11:08:12 +00:00
}
else
{
// report acrobatics progression
2018-09-15 15:38:21 +00:00
if ( isPlayer )
2014-01-15 11:08:12 +00:00
cls . skillUsageSucceeded ( mPtr , ESM : : Skill : : Acrobatics , 1 ) ;
2013-10-01 21:35:34 +00:00
}
}
2018-08-19 17:41:09 +00:00
2018-10-14 17:11:21 +00:00
if ( mPtr . getClass ( ) . isNpc ( ) )
2018-11-23 16:07:52 +00:00
playLandingSound = true ;
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
{
2018-12-05 20:28:26 +00:00
if ( mPtr . getClass ( ) . isNpc ( ) & & mJumpState = = JumpState_InAir & & ! flying & & solid )
2018-11-23 16:07:52 +00:00
playLandingSound = true ;
2017-10-31 13:22:24 +00:00
jumpstate = mAnimation - > isPlaying ( mCurrentJump ) ? JumpState_Landing : JumpState_None ;
2019-11-12 14:10:28 +00:00
vec . x ( ) * = scale ;
vec . y ( ) * = scale ;
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 ;
2020-06-22 00:03:38 +00:00
if ( movementSettings . mIsStrafing )
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 ) ) ) ;
}
2020-06-22 00:03:38 +00:00
else if ( vec . length2 ( ) > 0.0f )
2013-08-19 06:42:56 +00:00
{
2020-06-22 00:03:38 +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 ) ) ) ;
2020-06-22 00:03:38 +00:00
else
2013-08-19 06:42:56 +00:00
movestate = ( inwater ? ( isrunning ? CharState_SwimRunBack : CharState_SwimWalkBack )
: ( sneak ? CharState_SneakBack
: ( isrunning ? CharState_RunBack : CharState_WalkBack ) ) ) ;
}
2020-09-04 13:03:33 +00:00
else
2013-08-19 06:42:56 +00:00
{
2019-01-09 15:05:51 +00:00
// Do not play turning animation for player if rotation speed is very slow.
// Actual threshold should take framerate in account.
2020-09-04 13:03:33 +00:00
float rotationThreshold = ( isPlayer ? 0.015f : 0.001f ) * 60 * duration ;
2019-01-09 15:05:51 +00:00
2018-08-20 18:04:02 +00:00
// It seems only bipedal actors use turning animations.
// Also do not use turning animations in the first-person view and when sneaking.
2020-07-09 06:47:37 +00:00
if ( ! sneak & & jumpstate = = JumpState_None & & ! isFirstPersonPlayer & & mPtr . getClass ( ) . isBipedal ( mPtr ) )
2018-08-20 18:04:02 +00:00
{
2020-06-22 00:03:38 +00:00
if ( effectiveRotation > rotationThreshold )
2018-08-20 18:04:02 +00:00
movestate = inwater ? CharState_SwimTurnRight : CharState_TurnRight ;
2020-06-22 00:03:38 +00:00
else if ( effectiveRotation < - rotationThreshold )
2018-08-20 18:04:02 +00:00
movestate = inwater ? CharState_SwimTurnLeft : CharState_TurnLeft ;
}
2013-08-19 06:42:56 +00:00
}
2013-03-31 10:50:20 +00:00
}
2013-07-16 05:56:23 +00:00
2018-11-23 16:07:52 +00:00
if ( playLandingSound )
{
MWBase : : SoundManager * sndMgr = MWBase : : Environment : : get ( ) . getSoundManager ( ) ;
2019-01-09 15:05:51 +00:00
std : : string sound ;
2018-11-23 16:07:52 +00:00
osg : : Vec3f pos ( mPtr . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
if ( world - > isUnderwater ( mPtr . getCell ( ) , pos ) | | world - > isWalkingOnWater ( mPtr ) )
sound = " DefaultLandWater " ;
2019-01-09 15:05:51 +00:00
else if ( onground )
sound = " DefaultLand " ;
2018-11-23 16:07:52 +00:00
2019-01-09 15:05:51 +00:00
if ( ! sound . empty ( ) )
sndMgr - > playSound3D ( mPtr , sound , 1.f , 1.f , MWSound : : Type : : Foot , MWSound : : PlayMode : : NoPlayerLocal ) ;
2018-11-23 16:07:52 +00:00
}
2020-07-09 06:47:37 +00:00
if ( turnToMovementDirection )
{
float targetSwimmingPitch ;
if ( inwater & & vec . y ( ) ! = 0 & & ! isFirstPersonPlayer & & ! movementSettings . mIsStrafing )
targetSwimmingPitch = - mPtr . getRefData ( ) . getPosition ( ) . rot [ 0 ] ;
else
targetSwimmingPitch = 0 ;
float maxSwimPitchDelta = 3.0f * duration ;
float swimmingPitch = mAnimation - > getBodyPitchRadians ( ) ;
swimmingPitch + = osg : : clampBetween ( targetSwimmingPitch - swimmingPitch , - maxSwimPitchDelta , maxSwimPitchDelta ) ;
mAnimation - > setBodyPitchRadians ( swimmingPitch ) ;
}
2020-08-07 20:17:44 +00:00
static const bool swimUpwardCorrection = Settings : : Manager : : getBool ( " swim upward correction " , " Game " ) ;
if ( inwater & & isPlayer & & ! isFirstPersonPlayer & & swimUpwardCorrection )
2020-07-09 06:47:37 +00:00
{
static const float swimUpwardCoef = Settings : : Manager : : getFloat ( " swim upward coef " , " Game " ) ;
static const float swimForwardCoef = sqrtf ( 1.0f - swimUpwardCoef * swimUpwardCoef ) ;
vec . z ( ) = std : : abs ( vec . y ( ) ) * swimUpwardCoef ;
vec . y ( ) * = swimForwardCoef ;
}
2018-07-12 08:43:49 +00:00
// Player can not use smooth turning as NPCs, so we play turning animation a bit to avoid jittering
2018-09-15 15:38:21 +00:00
if ( isPlayer )
2018-07-12 08:43:49 +00:00
{
float threshold = mCurrentMovement . find ( " swim " ) = = std : : string : : npos ? 0.4f : 0.8f ;
float complete ;
bool animPlaying = mAnimation - > getInfo ( mCurrentMovement , & complete ) ;
2019-01-09 15:05:51 +00:00
if ( movestate = = CharState_None & & jumpstate = = JumpState_None & & isTurning ( ) )
2018-07-12 08:43:49 +00:00
{
2018-08-08 12:52:09 +00:00
if ( animPlaying & & complete < threshold )
2018-07-12 08:43:49 +00:00
movestate = mMovementState ;
}
}
else
2014-09-17 03:20:10 +00:00
{
2018-08-20 18:04:02 +00:00
if ( mPtr . getClass ( ) . isBipedal ( mPtr ) )
2018-07-12 08:43:49 +00:00
{
2018-08-20 18:04:02 +00:00
if ( mTurnAnimationThreshold > 0 )
mTurnAnimationThreshold - = duration ;
if ( movestate = = CharState_TurnRight | | movestate = = CharState_TurnLeft | |
movestate = = CharState_SwimTurnRight | | movestate = = CharState_SwimTurnLeft )
{
mTurnAnimationThreshold = 0.05f ;
}
else if ( movestate = = CharState_None & & isTurning ( )
& & mTurnAnimationThreshold > 0 )
{
movestate = mMovementState ;
}
2018-07-12 08:43:49 +00:00
}
2014-09-17 03:20:10 +00:00
}
2017-09-24 12:26:41 +00:00
if ( movestate ! = CharState_None & & ! isTurning ( ) )
2013-07-16 05:56:23 +00:00
clearAnimQueue ( ) ;
2020-01-05 12:09:02 +00:00
if ( mAnimQueue . empty ( ) | | inwater | | ( sneak & & mIdleState ! = CharState_SpecialIdle ) )
2014-12-28 14:34:47 +00:00
{
2019-05-14 13:34:41 +00:00
if ( inwater )
2018-08-16 07:21:48 +00:00
idlestate = CharState_IdleSwim ;
else if ( sneak & & ! inJump )
idlestate = CharState_IdleSneak ;
else
idlestate = 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 ) )
2018-08-20 18:04:02 +00:00
forcestateupdate = updateWeaponState ( idlestate ) | | forcestateupdate ;
2015-07-17 01:28:17 +00:00
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
2017-09-22 12:07:00 +00:00
if ( isTurning ( ) )
2014-09-17 03:20:10 +00:00
{
2018-07-12 08:43:49 +00:00
// Adjust animation speed from 1.0 to 1.5 multiplier
2014-09-17 03:20:10 +00:00
if ( duration > 0 )
2018-09-08 14:17:11 +00:00
{
float turnSpeed = std : : min ( 1.5f , std : : abs ( rot . z ( ) ) / duration / static_cast < float > ( osg : : PI ) ) ;
2018-07-12 08:43:49 +00:00
mAnimation - > adjustSpeedMult ( mCurrentMovement , std : : max ( turnSpeed , 1.0f ) ) ;
2018-09-08 14:17:11 +00:00
}
2014-09-17 03:20:10 +00:00
}
2015-07-25 16:22:48 +00:00
else if ( mMovementState ! = CharState_None & & mAdjustMovementAnimSpeed )
{
2020-05-11 12:11:32 +00:00
// Vanilla caps the played animation speed.
const float maxSpeedMult = 10.f ;
const float speedMult = speed / mMovementAnimSpeed ;
mAnimation - > adjustSpeedMult ( mCurrentMovement , std : : min ( maxSpeedMult , speedMult ) ) ;
// Make sure the actual speed is the "expected" speed even though the animation is slower
scale * = std : : max ( 1.f , speedMult / maxSpeedMult ) ;
2015-07-25 16:22:48 +00:00
}
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
{
2017-09-22 11:26:35 +00:00
if ( ! isKnockedDown ( ) & & ! isKnockedOut ( ) )
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
2018-09-21 12:34:23 +00:00
if ( ! animationOnly & & ! mMovementAnimationControlled )
2014-02-02 13:01:49 +00:00
world - > queueMovement ( mPtr , vec ) ;
2014-01-02 19:54:41 +00:00
}
2018-09-21 12:34:23 +00:00
else if ( ! animationOnly )
2014-06-26 15:26:33 +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-08-18 05:34:38 +00:00
movement = vec ;
2020-06-22 00:03:38 +00:00
movementSettings . mPosition [ 0 ] = movementSettings . mPosition [ 1 ] = 0 ;
2019-06-30 14:38:49 +00:00
if ( movement . z ( ) = = 0.f )
2020-06-22 00:03:38 +00:00
movementSettings . mPosition [ 2 ] = 0 ;
2019-06-30 14:38:49 +00:00
// Can't reset jump state (mPosition[2]) here in full; we don't know for sure whether the PhysicSystem will actually handle it in this frame
2014-09-17 00:20:46 +00:00
// 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
2018-06-20 08:37:58 +00:00
if ( ! mSkipAnim & & mDeathState ! = CharState_None & & mCurrentDeath . empty ( ) )
2015-09-17 02:30:55 +00:00
{
2018-06-20 08:37:58 +00:00
// Fast-forward death animation to end for persisting corpses or corpses after end of death animation
if ( cls . isPersistent ( mPtr ) | | cls . getCreatureStats ( mPtr ) . isDeathAnimationFinished ( ) )
playDeath ( 1.f , mDeathState ) ;
2015-09-17 02:30:55 +00:00
}
2016-02-01 22:13:43 +00:00
// We must always queue movement, even if there is none, to apply gravity.
2018-09-21 12:34:23 +00:00
if ( ! animationOnly )
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
2018-06-12 12:07:36 +00:00
bool isPersist = isPersistentAnimPlaying ( ) ;
osg : : Vec3f moved = mAnimation - > runAnimation ( mSkipAnim & & ! isPersist ? 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
2019-04-17 13:22:03 +00:00
moved . x ( ) * = scale ;
moved . y ( ) * = scale ;
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 ( ) ;
2020-06-27 18:43:08 +00:00
if ( std : : abs ( movement . x ( ) - moved . x ( ) ) > std : : abs ( moved . x ( ) ) / 2 | |
std : : abs ( movement . y ( ) - moved . y ( ) ) > std : : abs ( moved . y ( ) ) / 2 | |
std : : abs ( movement . z ( ) - moved . z ( ) ) > std : : abs ( moved . z ( ) ) / 2 )
{
moved = movement ;
// For some creatures getSpeed doesn't work, so we adjust speed to the animation.
// TODO: Fix Creature::getSpeed.
float newLength = moved . length ( ) ;
if ( newLength > 0 & & ! cls . isNpc ( ) )
moved * = ( l / newLength ) ;
}
2013-02-04 15:10:14 +00:00
}
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
2018-09-21 12:34:23 +00:00
if ( ! animationOnly & & 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 ( ) ) ;
2020-03-28 15:30:56 +00:00
# ifdef USE_OPENXR
if ( isPlayer )
{
auto disabled = MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . isDisabled ( ) ;
auto animation = static_cast < MWRender : : NpcAnimation * > ( mAnimation ) ;
if ( disabled )
animation - > setViewMode ( MWRender : : NpcAnimation : : VM_VRNormal ) ;
else
animation - > setViewMode ( MWRender : : NpcAnimation : : VM_VRFirstPerson ) ;
}
# endif
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 ;
2018-10-09 06:21:12 +00:00
mAnimation - > getInfo ( anim . mGroup , & complete , nullptr ) ;
2016-07-30 17:24:03 +00:00
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 ,
2018-06-27 04:22:45 +00:00
Priority_Persistent , 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
return false ;
2017-09-18 08:21:18 +00:00
2018-06-11 13:18:51 +00:00
// We should not interrupt persistent animations by non-persistent ones
2018-06-12 07:51:54 +00:00
if ( isPersistentAnimPlaying ( ) & & ! persist )
return false ;
2018-06-11 13:18:51 +00:00
2017-09-18 08:21:18 +00:00
// If this animation is a looped animation (has a "loop start" key) that is already playing
// and has not yet reached the end of the loop, allow it to continue animating with its existing loop count
// and remove any other animations that were queued.
// This emulates observed behavior from the original allows the script "OutsideBanner" to animate banners correctly.
if ( ! mAnimQueue . empty ( ) & & mAnimQueue . front ( ) . mGroup = = groupname & &
mAnimation - > getTextKeyTime ( mAnimQueue . front ( ) . mGroup + " : loop start " ) > = 0 & &
mAnimation - > isPlaying ( groupname ) )
2013-01-17 21:18:40 +00:00
{
2017-09-18 08:21:18 +00:00
float endOfLoop = mAnimation - > getTextKeyTime ( mAnimQueue . front ( ) . mGroup + " : loop stop " ) ;
2016-08-17 16:08:54 +00:00
2017-09-18 08:21:18 +00:00
if ( endOfLoop < 0 ) // if no Loop Stop key was found, use the Stop key
endOfLoop = mAnimation - > getTextKeyTime ( mAnimQueue . front ( ) . mGroup + " : stop " ) ;
2016-08-17 16:08:54 +00:00
2017-09-18 08:21:18 +00:00
if ( endOfLoop > 0 & & ( mAnimation - > getCurrentTime ( mAnimQueue . front ( ) . mGroup ) < endOfLoop ) )
{
mAnimQueue . resize ( 1 ) ;
return true ;
2016-08-17 16:08:54 +00:00
}
2017-09-18 08:21:18 +00:00
}
2016-08-17 13:48:55 +00:00
2017-09-18 08:21:18 +00:00
count = std : : max ( count , 1 ) ;
2016-07-30 17:24:03 +00:00
2017-09-18 08:21:18 +00:00
AnimationQueueEntry entry ;
entry . mGroup = groupname ;
entry . mLoopCount = count - 1 ;
entry . mPersist = persist ;
2016-07-30 17:24:03 +00:00
2017-09-18 08:21:18 +00:00
if ( mode ! = 0 | | mAnimQueue . empty ( ) | | ! isAnimPlaying ( mAnimQueue . front ( ) . mGroup ) )
{
2018-06-12 10:04:03 +00:00
clearAnimQueue ( persist ) ;
2013-05-12 12:08:01 +00:00
2017-09-18 08:21:18 +00:00
mAnimation - > disable ( mCurrentIdle ) ;
mCurrentIdle . clear ( ) ;
2013-07-16 05:56:23 +00:00
2017-09-18 08:21:18 +00:00
mIdleState = CharState_SpecialIdle ;
bool loopfallback = ( entry . mGroup . compare ( 0 , 4 , " idle " ) = = 0 ) ;
2018-06-12 10:04:03 +00:00
mAnimation - > play ( groupname , persist & & groupname ! = " idle " ? Priority_Persistent : Priority_Default ,
2017-09-18 08:21:18 +00:00
MWRender : : Animation : : BlendMask_All , false , 1.0f ,
( ( mode = = 2 ) ? " loop start " : " start " ) , " stop " , 0.0f , count - 1 , loopfallback ) ;
}
Some PVS-Studio and cppcheck fixes
cppcheck:
[apps/esmtool/record.cpp:697]: (performance) Prefer prefix ++/-- operators for non-primitive types.
[apps/esmtool/record.cpp:1126]: (performance) Prefer prefix ++/-- operators for non-primitive types.
[apps/esmtool/record.cpp:1138]: (performance) Prefer prefix ++/-- operators for non-primitive types.
[apps/niftest/niftest.cpp:36]: (performance) Function parameter 'filename' should be passed by reference.
[apps/niftest/niftest.cpp:41]: (performance) Function parameter 'filename' should be passed by reference.
[apps/opencs/model/prefs/boolsetting.cpp:25]: (warning) Possible leak in public function. The pointer 'mWidget' is not deallocated before it is allocated.
[apps/opencs/model/prefs/shortcuteventhandler.cpp:52]: (warning) Return value of std::remove() ignored. Elements remain in container.
[apps/openmw/mwstate/quicksavemanager.cpp:5]: (performance) Variable 'mSaveName' is assigned in constructor body. Consider performing initialization in initialization list.
PVS-Studio:
apps/opencs/model/filter/parser.cpp 582 warn V560 A part of conditional expression is always true: allowPredefined.
apps/opencs/view/world/referencecreator.cpp 67 warn V547 Expression '!errors.empty()' is always false.
apps/opencs/view/world/referencecreator.cpp 74 warn V547 Expression '!errors.empty()' is always false.
apps/opencs/view/doc/loader.cpp 170 warn V560 A part of conditional expression is always true: !completed.
apps/opencs/view/doc/loader.cpp 170 warn V560 A part of conditional expression is always true: !error.empty().
apps/opencs/model/tools/pathgridcheck.cpp 32 err V517 The use of 'if (A) {...} else if (A) {...}' pattern was detected. There is a probability of logical error presence. Check lines: 32, 34.
apps/opencs/model/world/refidadapterimp.cpp 1376 err V547 Expression 'subColIndex < 3' is always true.
apps/openmw/mwgui/widgets.hpp 318 warn V703 It is odd that the 'mEnableRepeat' field in derived class 'MWScrollBar' overwrites field in base class 'ScrollBar'. Check lines: widgets.hpp:318, MyGUI_ScrollBar.h:179.
apps/openmw/mwgui/widgets.hpp 319 warn V703 It is odd that the 'mRepeatTriggerTime' field in derived class 'MWScrollBar' overwrites field in base class 'ScrollBar'. Check lines: widgets.hpp:319, MyGUI_ScrollBar.h:180.
apps/openmw/mwgui/widgets.hpp 320 warn V703 It is odd that the 'mRepeatStepTime' field in derived class 'MWScrollBar' overwrites field in base class 'ScrollBar'. Check lines: widgets.hpp:320, MyGUI_ScrollBar.h:181
apps/openmw/mwmechanics/actors.cpp 1425 warn V547 Expression '!detected' is always true.
apps/openmw/mwmechanics/character.cpp 2155 err V547 Expression 'mode == 0' is always true.
apps/openmw/mwmechanics/character.cpp 1192 warn V592 The expression was enclosed by parentheses twice: ((expression)). One pair of parentheses is unnecessary or misprint is present.
apps/openmw/mwmechanics/character.cpp 521 warn V560 A part of conditional expression is always true: (idle == mIdleState).
apps/openmw/mwmechanics/pathfinding.cpp 317 err V547 Expression 'mPath.size() >= 2' is always true.
apps/openmw/mwscript/interpretercontext.cpp 409 warn V560 A part of conditional expression is always false: rank > 9.
apps/openmw/mwgui/windowbase.cpp 28 warn V560 A part of conditional expression is always true: !visible.
apps/openmw/mwgui/journalwindow.cpp 561 warn V547 Expression '!mAllQuests' is always false.
apps/openmw/mwgui/referenceinterface.cpp 18 warn V571 Recurring check. The '!mPtr.isEmpty()' condition was already verified in line 16.
apps/openmw/mwworld/scene.cpp 463 warn V547 Expression 'adjustPlayerPos' is always true.
apps/openmw/mwworld/worldimp.cpp 409 err V766 An item with the same key '"sCompanionShare"' has already been added.
apps/openmw/mwworld/cellstore.cpp 691 warn V519 The 'state.mWaterLevel' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 689, 691.
apps/openmw/mwworld/weather.cpp 1125 warn V519 The 'mResult.mParticleEffect' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 1123, 1125.
apps/openmw/mwworld/weather.cpp 1137 warn V519 The 'mResult.mParticleEffect' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 1135, 1137.
apps/wizard/unshield/unshieldworker.cpp 475 warn V728 An excessive check can be simplified. The '(A && B) || (!A && !B)' expression is equivalent to the 'bool(A) == bool(B)' expression.
apps/wizard/installationpage.cpp 163 warn V735 Possibly an incorrect HTML. The "</p" closing tag was encountered, while the "</span" tag was expected.
components/fontloader/fontloader.cpp 427 err V547 Expression 'i == 1' is always true.
components/nifosg/nifloader.cpp 282 warn V519 The 'created' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 278, 282.
components/esm/loadregn.cpp 119 err V586 The 'clear' function is called twice for deallocation of the same resource. Check lines: 112, 119.
components/esm/cellref.cpp 178 warn V581 The conditional expressions of the 'if' statements situated alongside each other are identical. Check lines: 175, 178.
components/esmterrain/storage.cpp 235 warn V560 A part of conditional expression is always true: colStart == 0.
components/esmterrain/storage.cpp 237 warn V560 A part of conditional expression is always true: rowStart == 0.
2018-04-09 15:55:16 +00:00
else
2017-09-18 08:21:18 +00:00
{
mAnimQueue . resize ( 1 ) ;
2013-01-17 21:18:40 +00:00
}
2018-06-12 10:04:03 +00:00
// "PlayGroup idle" is a special case, used to remove to stop scripted animations playing
if ( groupname = = " idle " )
entry . mPersist = false ;
mAnimQueue . push_back ( entry ) ;
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
}
2018-06-12 07:51:54 +00:00
bool CharacterController : : isPersistentAnimPlaying ( )
{
if ( ! mAnimQueue . empty ( ) )
{
AnimationQueueEntry & first = mAnimQueue . front ( ) ;
return first . mPersist & & isAnimPlaying ( first . mGroup ) ;
}
return false ;
}
2013-05-25 03:10:07 +00:00
bool CharacterController : : isAnimPlaying ( const std : : string & groupName )
{
2018-10-09 06:21:12 +00:00
if ( mAnimation = = nullptr )
2013-05-25 03:10:07 +00:00
return false ;
2013-07-16 05:56:23 +00:00
return mAnimation - > isPlaying ( groupName ) ;
2013-05-25 03:10:07 +00:00
}
2018-06-12 10:04:03 +00:00
void CharacterController : : clearAnimQueue ( bool clearPersistAnims )
2013-05-16 13:59:41 +00:00
{
2018-06-12 10:04:03 +00:00
// Do not interrupt scripted animations, if we want to keep them
if ( ( ! isPersistentAnimPlaying ( ) | | clearPersistAnims ) & & ! mAnimQueue . empty ( ) )
2018-06-12 07:51:54 +00:00
mAnimation - > disable ( mAnimQueue . front ( ) . mGroup ) ;
2018-06-11 13:18:51 +00:00
for ( AnimationQueue : : iterator it = mAnimQueue . begin ( ) ; it ! = mAnimQueue . end ( ) ; )
{
2018-06-12 10:04:03 +00:00
if ( clearPersistAnims | | ! it - > mPersist )
2018-06-11 13:18:51 +00:00
it = mAnimQueue . erase ( it ) ;
else
+ + it ;
}
2013-05-16 13:59:41 +00:00
}
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
2019-03-20 08:19:35 +00:00
// Make sure we canceled the current attack or spellcasting,
// because we disabled attack animations anyway.
mCastingManualSpell = false ;
mAttackingOrSpell = false ;
if ( mUpperBodyState ! = UpperCharState_Nothing )
mUpperBodyState = UpperCharState_WeapEquiped ;
2015-07-16 18:03:16 +00:00
refreshCurrentAnims ( mIdleState , mMovementState , mJumpState , true ) ;
2018-06-11 14:52:20 +00:00
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 ;
2018-12-26 09:45:28 +00:00
mWeaponType = ESM : : Weapon : : None ;
2013-07-16 06:43:33 +00:00
}
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 ) ;
2020-07-25 11:54:49 +00:00
for ( int effectId : effects )
2013-11-13 14:44:43 +00:00
{
2019-06-13 21:42:25 +00:00
if ( mPtr . getClass ( ) . getCreatureStats ( mPtr ) . isDeathAnimationFinished ( )
2020-07-25 11:54:49 +00:00
| | mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( MWMechanics : : EffectKey ( effectId ) ) . getMagnitude ( ) < = 0 )
mAnimation - > removeEffect ( effectId ) ;
2013-11-13 14:44:43 +00:00
}
}
2014-12-12 01:39:59 +00:00
void CharacterController : : updateMagicEffects ( )
2013-12-08 22:05:21 +00:00
{
if ( ! mPtr . getClass ( ) . isActor ( ) )
return ;
2014-08-23 23:50:29 +00:00
float light = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Light ) . getMagnitude ( ) ;
mAnimation - > setLightEffect ( light ) ;
2019-09-04 13:42:34 +00:00
// If you're dead you don't care about whether you've started/stopped being a vampire or not
if ( mPtr . getClass ( ) . getCreatureStats ( mPtr ) . isDead ( ) )
return ;
bool vampire = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Vampirism ) . getMagnitude ( ) > 0.0f ;
mAnimation - > setVampire ( vampire ) ;
2013-12-08 22:05:21 +00:00
}
2018-09-21 12:34:23 +00:00
void CharacterController : : setVisibility ( float visibility )
{
// We should take actor's invisibility in account
if ( mPtr . getClass ( ) . isActor ( ) )
{
float alpha = 1.f ;
if ( mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Invisibility ) . getModifier ( ) ) // Ignore base magnitude (see bug #3555).
{
if ( mPtr = = getPlayer ( ) )
2019-06-30 13:13:11 +00:00
alpha = 0.25f ;
2018-09-21 12:34:23 +00:00
else
2019-06-30 13:13:11 +00:00
alpha = 0.05f ;
2018-09-21 12:34:23 +00:00
}
float chameleon = mPtr . getClass ( ) . getCreatureStats ( mPtr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Chameleon ) . getMagnitude ( ) ;
if ( chameleon )
{
2019-06-30 13:13:11 +00:00
alpha * = std : : min ( 0.75f , std : : max ( 0.25f , ( 100.f - chameleon ) / 100.f ) ) ;
2018-09-21 12:34:23 +00:00
}
visibility = std : : min ( visibility , alpha ) ;
}
// TODO: implement a dithering shader rather than just change object transparency.
mAnimation - > setAlpha ( visibility ) ;
}
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 ;
2020-08-30 23:25:53 +00:00
if ( std : : abs ( move [ 1 ] ) > std : : abs ( move [ 0 ] ) + 0.2f ) // forward-backward
2014-12-31 15:59:21 +00:00
mAttackType = " thrust " ;
2020-08-30 23:25:53 +00:00
else if ( std : : abs ( move [ 0 ] ) > std : : abs ( move [ 1 ] ) + 0.2f ) // sideway
2014-12-31 15:59:21 +00:00
mAttackType = " slash " ;
else
mAttackType = " chop " ;
2013-01-10 16:35:24 +00:00
}
2014-01-23 21:14:20 +00:00
2018-07-19 12:38:32 +00:00
bool CharacterController : : isRandomAttackAnimation ( const std : : string & group ) const
{
return ( group = = " attack1 " | | group = = " swimattack1 " | |
group = = " attack2 " | | group = = " swimattack2 " | |
group = = " attack3 " | | group = = " swimattack3 " ) ;
}
2018-08-16 13:47:06 +00:00
bool CharacterController : : isAttackPreparing ( ) const
2017-09-01 05:34:15 +00:00
{
return mUpperBodyState = = UpperCharState_StartToMinAttack | |
mUpperBodyState = = UpperCharState_MinAttackToMaxAttack ;
}
2018-06-28 12:58:51 +00:00
bool CharacterController : : isCastingSpell ( ) const
{
return mCastingManualSpell | | mUpperBodyState = = UpperCharState_CastingSpell ;
}
2014-12-12 15:49:22 +00:00
bool CharacterController : : isReadyToBlock ( ) const
{
return updateCarriedLeftVisible ( mWeaponType ) ;
}
2017-09-22 11:26:35 +00:00
bool CharacterController : : isKnockedDown ( ) const
{
return mHitState = = CharState_KnockDown | |
mHitState = = CharState_SwimKnockDown ;
}
2014-12-12 16:39:00 +00:00
bool CharacterController : : isKnockedOut ( ) const
{
2017-09-22 11:26:35 +00:00
return mHitState = = CharState_KnockOut | |
mHitState = = CharState_SwimKnockOut ;
2014-12-12 16:39:00 +00:00
}
2017-09-22 12:07:00 +00:00
bool CharacterController : : isTurning ( ) const
{
return mMovementState = = CharState_TurnLeft | |
mMovementState = = CharState_TurnRight | |
mMovementState = = CharState_SwimTurnLeft | |
mMovementState = = CharState_SwimTurnRight ;
}
2017-09-22 11:49:42 +00:00
bool CharacterController : : isRecovery ( ) const
{
return mHitState = = CharState_Hit | |
mHitState = = CharState_SwimHit ;
}
2017-08-18 15:24:34 +00:00
bool CharacterController : : isAttackingOrSpell ( ) const
{
return mUpperBodyState ! = UpperCharState_Nothing & &
mUpperBodyState ! = UpperCharState_WeapEquiped ;
}
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 ;
}
2017-08-16 16:30:47 +00:00
bool CharacterController : : isRunning ( ) const
{
return mMovementState = = CharState_RunForward | |
mMovementState = = CharState_RunBack | |
mMovementState = = CharState_RunLeft | |
mMovementState = = CharState_RunRight | |
mMovementState = = CharState_SwimRunForward | |
mMovementState = = CharState_SwimRunBack | |
mMovementState = = CharState_SwimRunLeft | |
mMovementState = = CharState_SwimRunRight ;
}
2015-07-02 17:14:28 +00:00
void CharacterController : : setAttackingOrSpell ( bool attackingOrSpell )
{
mAttackingOrSpell = attackingOrSpell ;
}
2018-06-28 12:58:51 +00:00
void CharacterController : : castSpell ( const std : : string spellId , bool manualSpell )
{
mAttackingOrSpell = true ;
mCastingManualSpell = manualSpell ;
ActionSpell action = ActionSpell ( spellId ) ;
action . prepare ( mPtr ) ;
}
2017-04-20 11:36:14 +00:00
void CharacterController : : setAIAttackType ( const std : : string & attackType )
2016-09-01 13:43:33 +00:00
{
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 )
2018-07-19 12:38:32 +00:00
& & 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 ;
}
2018-01-11 01:49:35 +00:00
void CharacterController : : setActive ( int active )
2015-04-29 21:48:08 +00:00
{
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 " ) ;
2018-10-09 06:21:12 +00:00
if ( node = = nullptr )
2015-05-31 16:04:14 +00:00
node = anim - > getNode ( " Bip01 Head " ) ;
2018-10-09 06:21:12 +00:00
if ( node ! = nullptr )
2015-05-31 16:04:14 +00:00
{
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
2020-09-15 21:25:32 +00:00
zAngleRadians = std : : atan2 ( actorDirection . x ( ) , actorDirection . y ( ) ) - std : : atan2 ( direction . x ( ) , direction . y ( ) ) ;
xAngleRadians = std : : asin ( direction . z ( ) ) ;
2014-12-16 19:47:45 +00:00
}
2015-05-31 16:04:14 +00:00
2020-09-15 21:25:32 +00:00
const double xLimit = osg : : DegreesToRadians ( 40.0 ) ;
const double zLimit = osg : : DegreesToRadians ( 30.0 ) ;
double zLimitOffset = mAnimation - > getUpperBodyYawRadians ( ) ;
xAngleRadians = osg : : clampBetween ( Misc : : normalizeAngle ( xAngleRadians ) , - xLimit , xLimit ) ;
zAngleRadians = osg : : clampBetween ( Misc : : normalizeAngle ( zAngleRadians ) ,
- zLimit + zLimitOffset , zLimit + zLimitOffset ) ;
2014-12-16 19:47:45 +00:00
float factor = duration * 5 ;
factor = std : : min ( factor , 1.f ) ;
2020-09-15 21:25:32 +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
}