2012-08-11 15:30:55 +00:00
# include "mechanicsmanagerimp.hpp"
2015-05-29 23:00:24 +00:00
2015-04-22 15:58:55 +00:00
# include <components/misc/rng.hpp>
2015-03-15 01:07:47 +00:00
2015-07-07 17:16:32 +00:00
# include <components/esm/esmwriter.hpp>
2015-02-04 20:18:43 +00:00
# include <components/esm/stolenitems.hpp>
2015-11-20 20:57:04 +00:00
# include <components/sceneutil/positionattitudetransform.hpp>
2017-04-18 09:38:26 +00:00
/*
Start of tes3mp addition
Include additional headers for multiplayer purposes
*/
2019-12-02 17:08:03 +00:00
# include <components/openmw-mp/TimedLog.hpp>
2017-04-20 14:00:40 +00:00
# include "../mwmp/Main.hpp"
2017-05-24 09:19:11 +00:00
# include "../mwmp/LocalPlayer.hpp"
2017-04-30 11:57:43 +00:00
# include "../mwmp/PlayerList.hpp"
2017-04-20 14:00:40 +00:00
# include "../mwmp/CellController.hpp"
2017-04-18 09:38:26 +00:00
/*
End of tes3mp addition
*/
2012-10-01 15:17:04 +00:00
# include "../mwworld/esmstore.hpp"
2013-03-16 21:53:33 +00:00
# include "../mwworld/inventorystore.hpp"
2016-06-17 14:07:16 +00:00
# include "../mwworld/class.hpp"
# include "../mwworld/player.hpp"
2017-08-18 13:06:47 +00:00
# include "../mwworld/ptr.hpp"
2010-07-26 09:15:38 +00:00
2012-04-23 13:27:03 +00:00
# include "../mwbase/environment.hpp"
2018-10-28 11:08:24 +00:00
# include "../mwbase/statemanager.hpp"
2012-07-03 10:30:50 +00:00
# include "../mwbase/world.hpp"
2012-08-12 16:11:09 +00:00
# include "../mwbase/windowmanager.hpp"
2012-11-09 23:42:31 +00:00
# include "../mwbase/dialoguemanager.hpp"
2012-04-23 13:27:03 +00:00
2016-06-17 14:07:16 +00:00
# include "aicombat.hpp"
# include "aipursue.hpp"
2014-01-01 20:20:37 +00:00
# include "spellcasting.hpp"
2014-07-11 05:31:18 +00:00
# include "autocalcspell.hpp"
2015-05-29 23:00:24 +00:00
# include "npcstats.hpp"
2015-08-21 09:12:39 +00:00
# include "actorutil.hpp"
2016-11-16 19:15:25 +00:00
# include "combat.hpp"
2014-08-30 21:11:09 +00:00
2014-12-21 01:45:37 +00:00
namespace
{
float getFightDispositionBias ( float disposition )
{
static const float fFightDispMult = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) . find (
2018-08-29 15:38:12 +00:00
" fFightDispMult " ) - > mValue . getFloat ( ) ;
2014-12-21 01:45:37 +00:00
return ( ( 50.f - disposition ) * fFightDispMult ) ;
}
2015-04-21 23:17:01 +00:00
void getPersuasionRatings ( const MWMechanics : : NpcStats & stats , float & rating1 , float & rating2 , float & rating3 , bool player )
{
const MWWorld : : Store < ESM : : GameSetting > & gmst =
MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
2018-08-29 15:38:12 +00:00
float persTerm = stats . getAttribute ( ESM : : Attribute : : Personality ) . getModified ( ) / gmst . find ( " fPersonalityMod " ) - > mValue . getFloat ( ) ;
float luckTerm = stats . getAttribute ( ESM : : Attribute : : Luck ) . getModified ( ) / gmst . find ( " fLuckMod " ) - > mValue . getFloat ( ) ;
float repTerm = stats . getReputation ( ) * gmst . find ( " fReputationMod " ) - > mValue . getFloat ( ) ;
2015-04-21 23:17:01 +00:00
float fatigueTerm = stats . getFatigueTerm ( ) ;
2018-08-29 15:38:12 +00:00
float levelTerm = stats . getLevel ( ) * gmst . find ( " fLevelMod " ) - > mValue . getFloat ( ) ;
2015-04-21 23:17:01 +00:00
rating1 = ( repTerm + luckTerm + persTerm + stats . getSkill ( ESM : : Skill : : Speechcraft ) . getModified ( ) ) * fatigueTerm ;
if ( player )
{
rating2 = rating1 + levelTerm ;
rating3 = ( stats . getSkill ( ESM : : Skill : : Mercantile ) . getModified ( ) + luckTerm + persTerm ) * fatigueTerm ;
}
else
{
rating2 = ( levelTerm + repTerm + luckTerm + persTerm + stats . getSkill ( ESM : : Skill : : Speechcraft ) . getModified ( ) ) * fatigueTerm ;
rating3 = ( stats . getSkill ( ESM : : Skill : : Mercantile ) . getModified ( ) + repTerm + luckTerm + persTerm ) * fatigueTerm ;
}
}
2014-12-21 01:45:37 +00:00
}
2010-07-26 09:15:38 +00:00
namespace MWMechanics
2010-07-28 16:48:01 +00:00
{
2010-09-15 11:31:26 +00:00
void MechanicsManager : : buildPlayer ( )
{
2015-08-21 09:12:39 +00:00
MWWorld : : Ptr ptr = getPlayer ( ) ;
2010-09-15 11:31:26 +00:00
2014-05-22 18:37:22 +00:00
MWMechanics : : CreatureStats & creatureStats = ptr . getClass ( ) . getCreatureStats ( ptr ) ;
MWMechanics : : NpcStats & npcStats = ptr . getClass ( ) . getNpcStats ( ptr ) ;
2010-09-15 11:31:26 +00:00
2017-04-13 06:13:58 +00:00
npcStats . setNeedRecalcDynamicStats ( true ) ;
2012-11-05 12:07:59 +00:00
const ESM : : NPC * player = ptr . get < ESM : : NPC > ( ) - > mBase ;
2010-09-15 11:31:26 +00:00
// reset
2018-05-08 22:25:07 +00:00
creatureStats . setLevel ( player - > mNpdt . mLevel ) ;
2012-07-22 14:29:54 +00:00
creatureStats . getSpells ( ) . clear ( ) ;
2014-08-17 01:57:26 +00:00
creatureStats . modifyMagicEffects ( MagicEffects ( ) ) ;
2010-09-15 11:31:26 +00:00
2010-09-16 08:45:08 +00:00
for ( int i = 0 ; i < 27 ; + + i )
2018-05-08 22:25:07 +00:00
npcStats . getSkill ( i ) . setBase ( player - > mNpdt . mSkills [ i ] ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Strength , player - > mNpdt . mStrength ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Intelligence , player - > mNpdt . mIntelligence ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Willpower , player - > mNpdt . mWillpower ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Agility , player - > mNpdt . mAgility ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Speed , player - > mNpdt . mSpeed ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Endurance , player - > mNpdt . mEndurance ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Personality , player - > mNpdt . mPersonality ) ;
creatureStats . setAttribute ( ESM : : Attribute : : Luck , player - > mNpdt . mLuck ) ;
2012-11-08 12:37:57 +00:00
const MWWorld : : ESMStore & esmStore =
MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
2012-05-28 09:37:56 +00:00
2010-09-15 11:31:26 +00:00
// race
2010-09-26 08:01:30 +00:00
if ( mRaceSelected )
{
const ESM : : Race * race =
2012-12-08 23:12:24 +00:00
esmStore . get < ESM : : Race > ( ) . find ( player - > mRace ) ;
2010-09-15 13:10:13 +00:00
2012-11-08 12:37:57 +00:00
bool male = ( player - > mFlags & ESM : : NPC : : Female ) = = 0 ;
2010-09-15 13:10:13 +00:00
2010-09-26 08:01:30 +00:00
for ( int i = 0 ; i < 8 ; + + i )
2010-09-15 13:10:13 +00:00
{
2013-04-04 11:50:36 +00:00
const ESM : : Race : : MaleFemale & attribute = race - > mData . mAttributeValues [ i ] ;
2013-12-28 16:19:35 +00:00
creatureStats . setAttribute ( i , male ? attribute . mMale : attribute . mFemale ) ;
2010-09-26 08:01:30 +00:00
}
2010-09-16 08:45:08 +00:00
2012-08-16 13:42:37 +00:00
for ( int i = 0 ; i < 27 ; + + i )
2010-09-16 08:45:08 +00:00
{
2012-08-16 13:42:37 +00:00
int bonus = 0 ;
2012-12-08 23:12:24 +00:00
2012-08-16 13:42:37 +00:00
for ( int i2 = 0 ; i2 < 7 ; + + i2 )
2012-09-17 07:37:50 +00:00
if ( race - > mData . mBonus [ i2 ] . mSkill = = i )
2012-08-16 13:42:37 +00:00
{
2012-09-17 07:37:50 +00:00
bonus = race - > mData . mBonus [ i2 ] . mBonus ;
2012-08-16 13:42:37 +00:00
break ;
}
2012-12-08 23:12:24 +00:00
2012-08-16 13:42:37 +00:00
npcStats . getSkill ( i ) . setBase ( 5 + bonus ) ;
2010-09-16 08:45:08 +00:00
}
2010-09-26 08:01:30 +00:00
2018-11-08 14:03:29 +00:00
for ( const std : : string & power : race - > mPowers . mList )
2010-09-30 12:28:01 +00:00
{
2018-11-05 22:35:20 +00:00
creatureStats . getSpells ( ) . add ( power ) ;
2010-09-30 12:28:01 +00:00
}
2010-09-16 08:45:08 +00:00
}
2010-09-15 13:10:13 +00:00
2010-09-15 11:31:26 +00:00
// birthsign
2012-11-08 14:50:18 +00:00
const std : : string & signId =
MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . getBirthSign ( ) ;
if ( ! signId . empty ( ) )
2010-09-26 08:01:30 +00:00
{
2010-09-30 12:28:01 +00:00
const ESM : : BirthSign * sign =
2012-11-08 14:50:18 +00:00
esmStore . get < ESM : : BirthSign > ( ) . find ( signId ) ;
2010-09-30 12:28:01 +00:00
2018-11-08 14:03:29 +00:00
for ( const std : : string & power : sign - > mPowers . mList )
2010-09-30 12:28:01 +00:00
{
2018-11-05 22:35:20 +00:00
creatureStats . getSpells ( ) . add ( power ) ;
2010-09-30 12:28:01 +00:00
}
2010-09-26 08:01:30 +00:00
}
2010-09-15 11:31:26 +00:00
// class
2010-09-26 08:01:30 +00:00
if ( mClassSelected )
2010-09-15 13:23:38 +00:00
{
2012-11-07 21:36:43 +00:00
const ESM : : Class * class_ =
2012-11-08 12:37:57 +00:00
esmStore . get < ESM : : Class > ( ) . find ( player - > mClass ) ;
2010-09-26 08:01:30 +00:00
for ( int i = 0 ; i < 2 ; + + i )
2010-09-15 13:23:38 +00:00
{
2012-11-07 21:36:43 +00:00
int attribute = class_ - > mData . mAttribute [ i ] ;
2010-09-26 08:01:30 +00:00
if ( attribute > = 0 & & attribute < 8 )
{
2013-12-28 16:19:35 +00:00
creatureStats . setAttribute ( attribute ,
2012-07-22 14:29:54 +00:00
creatureStats . getAttribute ( attribute ) . getBase ( ) + 10 ) ;
2010-09-26 08:01:30 +00:00
}
2010-09-15 13:23:38 +00:00
}
2010-09-26 07:55:00 +00:00
2010-09-26 08:01:30 +00:00
for ( int i = 0 ; i < 2 ; + + i )
2010-09-26 07:55:00 +00:00
{
2010-09-26 08:01:30 +00:00
int bonus = i = = 0 ? 10 : 25 ;
2010-09-26 07:55:00 +00:00
2010-09-26 08:01:30 +00:00
for ( int i2 = 0 ; i2 < 5 ; + + i2 )
2010-09-26 07:55:00 +00:00
{
2012-11-07 21:36:43 +00:00
int index = class_ - > mData . mSkills [ i2 ] [ i ] ;
2010-09-26 08:01:30 +00:00
if ( index > = 0 & & index < 27 )
{
2012-07-06 16:23:48 +00:00
npcStats . getSkill ( index ) . setBase (
npcStats . getSkill ( index ) . getBase ( ) + bonus ) ;
2010-09-26 08:01:30 +00:00
}
2010-09-26 07:55:00 +00:00
}
}
2010-09-15 11:31:26 +00:00
2012-11-05 17:19:22 +00:00
const MWWorld : : Store < ESM : : Skill > & skills =
2012-11-08 12:37:57 +00:00
esmStore . get < ESM : : Skill > ( ) ;
2010-09-26 07:55:00 +00:00
2012-11-05 17:19:22 +00:00
MWWorld : : Store < ESM : : Skill > : : iterator iter = skills . begin ( ) ;
for ( ; iter ! = skills . end ( ) ; + + iter )
2010-09-26 07:55:00 +00:00
{
2014-12-05 01:17:15 +00:00
if ( iter - > second . mData . mSpecialization = = class_ - > mData . mSpecialization )
2010-09-26 07:55:00 +00:00
{
2014-12-05 01:17:15 +00:00
int index = iter - > first ;
2010-09-26 08:01:30 +00:00
if ( index > = 0 & & index < 27 )
{
2012-07-06 16:23:48 +00:00
npcStats . getSkill ( index ) . setBase (
npcStats . getSkill ( index ) . getBase ( ) + 5 ) ;
2010-09-26 08:01:30 +00:00
}
2010-09-26 07:55:00 +00:00
}
}
}
2012-05-18 11:54:07 +00:00
2014-07-11 05:31:18 +00:00
// F_PCStart spells
2018-10-09 06:21:12 +00:00
const ESM : : Race * race = nullptr ;
2014-07-11 05:31:18 +00:00
if ( mRaceSelected )
race = esmStore . get < ESM : : Race > ( ) . find ( player - > mRace ) ;
int skills [ ESM : : Skill : : Length ] ;
for ( int i = 0 ; i < ESM : : Skill : : Length ; + + i )
skills [ i ] = npcStats . getSkill ( i ) . getBase ( ) ;
int attributes [ ESM : : Attribute : : Length ] ;
for ( int i = 0 ; i < ESM : : Attribute : : Length ; + + i )
attributes [ i ] = npcStats . getAttribute ( i ) . getBase ( ) ;
2016-11-18 16:59:05 +00:00
std : : vector < std : : string > selectedSpells = autoCalcPlayerSpells ( skills , attributes , race ) ;
2014-07-11 05:31:18 +00:00
2018-11-08 14:03:29 +00:00
for ( const std : : string & spell : selectedSpells )
2018-11-05 22:35:20 +00:00
creatureStats . getSpells ( ) . add ( spell ) ;
2014-07-11 05:31:18 +00:00
2012-05-18 11:54:07 +00:00
// forced update and current value adjustments
2015-10-05 13:07:13 +00:00
mActors . updateActor ( ptr , 0 ) ;
2012-05-18 11:54:07 +00:00
2013-04-07 14:18:40 +00:00
for ( int i = 0 ; i < 3 ; + + i )
2012-10-19 11:10:06 +00:00
{
DynamicStat < float > stat = creatureStats . getDynamic ( i ) ;
stat . setCurrent ( stat . getModified ( ) ) ;
creatureStats . setDynamic ( i , stat ) ;
}
2013-04-15 00:56:23 +00:00
2014-01-19 10:42:58 +00:00
// auto-equip again. we need this for when the race is changed to a beast race and shoes are no longer equippable
MWWorld : : InventoryStore & invStore = ptr . getClass ( ) . getInventoryStore ( ptr ) ;
2013-04-15 00:56:23 +00:00
for ( int i = 0 ; i < MWWorld : : InventoryStore : : Slots ; + + i )
2013-11-17 12:38:54 +00:00
invStore . unequipAll ( ptr ) ;
2013-05-15 15:54:18 +00:00
invStore . autoEquip ( ptr ) ;
2010-09-15 11:31:26 +00:00
}
2017-03-25 10:15:16 +00:00
// mWatchedTimeToStartDrowning = -1 for correct drowning state check,
// if stats.getTimeToStartDrowning() == 0 already on game start
2012-04-23 13:27:03 +00:00
MechanicsManager : : MechanicsManager ( )
2019-01-24 18:16:42 +00:00
: mWatchedLevel ( - 1 ) , mWatchedTimeToStartDrowning ( - 1 ) , mWatchedStatsEmpty ( true ) , mUpdatePlayer ( true ) , mClassSelected ( false ) ,
2013-11-18 22:03:44 +00:00
mRaceSelected ( false ) , mAI ( true )
2010-07-26 09:15:38 +00:00
{
2016-12-14 15:39:33 +00:00
//buildPlayer no longer here, needs to be done explicitly after all subsystems are up and running
2010-07-26 09:15:38 +00:00
}
2013-01-29 07:39:11 +00:00
void MechanicsManager : : add ( const MWWorld : : Ptr & ptr )
2010-07-27 12:05:05 +00:00
{
2014-05-22 18:37:22 +00:00
if ( ptr . getClass ( ) . isActor ( ) )
2013-01-29 07:39:11 +00:00
mActors . addActor ( ptr ) ;
2013-03-31 22:12:10 +00:00
else
2013-03-31 22:29:41 +00:00
mObjects . addObject ( ptr ) ;
2010-07-27 12:05:05 +00:00
}
2010-07-28 16:48:01 +00:00
2018-06-28 12:58:51 +00:00
void MechanicsManager : : castSpell ( const MWWorld : : Ptr & ptr , const std : : string spellId , bool manualSpell )
{
if ( ptr . getClass ( ) . isActor ( ) )
mActors . castSpell ( ptr , spellId , manualSpell ) ;
}
2013-01-29 07:39:11 +00:00
void MechanicsManager : : remove ( const MWWorld : : Ptr & ptr )
2010-07-27 12:43:46 +00:00
{
2013-01-29 07:39:11 +00:00
if ( ptr = = mWatched )
2011-02-10 09:38:45 +00:00
mWatched = MWWorld : : Ptr ( ) ;
2013-01-29 07:39:11 +00:00
mActors . removeActor ( ptr ) ;
2013-03-31 22:29:41 +00:00
mObjects . removeObject ( ptr ) ;
2010-07-27 12:43:46 +00:00
}
2010-07-28 16:48:01 +00:00
2013-02-25 17:57:34 +00:00
void MechanicsManager : : updateCell ( const MWWorld : : Ptr & old , const MWWorld : : Ptr & ptr )
2013-01-29 08:19:24 +00:00
{
2013-08-28 00:22:07 +00:00
if ( old = = mWatched )
mWatched = ptr ;
2014-05-22 18:37:22 +00:00
if ( ptr . getClass ( ) . isActor ( ) )
2013-02-25 17:57:34 +00:00
mActors . updateActor ( old , ptr ) ;
2013-03-31 22:12:10 +00:00
else
2013-03-31 22:29:41 +00:00
mObjects . updateObject ( old , ptr ) ;
2013-01-29 08:19:24 +00:00
}
2013-01-29 07:39:11 +00:00
void MechanicsManager : : drop ( const MWWorld : : CellStore * cellStore )
2010-07-27 12:05:05 +00:00
{
2013-11-30 09:50:02 +00:00
mActors . dropActors ( cellStore , mWatched ) ;
2013-03-31 22:29:41 +00:00
mObjects . dropObjects ( cellStore ) ;
2010-07-27 12:05:05 +00:00
}
2010-07-28 16:48:01 +00:00
2013-01-29 07:39:11 +00:00
void MechanicsManager : : watchActor ( const MWWorld : : Ptr & ptr )
2010-07-27 12:46:05 +00:00
{
2010-07-27 13:59:41 +00:00
mWatched = ptr ;
}
2010-07-28 16:48:01 +00:00
2013-01-29 07:39:11 +00:00
void MechanicsManager : : update ( float duration , bool paused )
2010-07-27 13:59:41 +00:00
{
2013-08-28 00:56:47 +00:00
if ( ! mWatched . isEmpty ( ) )
2010-07-27 13:59:41 +00:00
{
2013-08-28 00:56:47 +00:00
MWBase : : WindowManager * winMgr = MWBase : : Environment : : get ( ) . getWindowManager ( ) ;
const MWMechanics : : NpcStats & stats = mWatched . getClass ( ) . getNpcStats ( mWatched ) ;
for ( int i = 0 ; i < ESM : : Attribute : : Length ; + + i )
2010-07-27 13:59:41 +00:00
{
2016-06-27 12:17:06 +00:00
if ( stats . getAttribute ( i ) ! = mWatchedAttributes [ i ] | | mWatchedStatsEmpty )
2010-07-27 13:59:41 +00:00
{
2013-08-28 17:50:29 +00:00
std : : stringstream attrname ;
attrname < < " AttribVal " < < ( i + 1 ) ;
2016-06-27 12:17:06 +00:00
mWatchedAttributes [ i ] = stats . getAttribute ( i ) ;
2013-08-28 17:50:29 +00:00
winMgr - > setValue ( attrname . str ( ) , stats . getAttribute ( i ) ) ;
2010-07-27 13:59:41 +00:00
}
}
2010-07-28 16:48:01 +00:00
2016-06-27 12:17:06 +00:00
if ( stats . getHealth ( ) ! = mWatchedHealth | | mWatchedStatsEmpty )
2013-08-28 00:56:47 +00:00
{
2013-08-28 17:50:29 +00:00
static const std : : string hbar ( " HBar " ) ;
2016-06-27 12:17:06 +00:00
mWatchedHealth = stats . getHealth ( ) ;
2013-08-28 17:50:29 +00:00
winMgr - > setValue ( hbar , stats . getHealth ( ) ) ;
2012-07-22 14:29:54 +00:00
}
2016-06-27 12:17:06 +00:00
if ( stats . getMagicka ( ) ! = mWatchedMagicka | | mWatchedStatsEmpty )
2013-08-28 00:56:47 +00:00
{
2013-08-28 17:50:29 +00:00
static const std : : string mbar ( " MBar " ) ;
2016-06-27 12:17:06 +00:00
mWatchedMagicka = stats . getMagicka ( ) ;
2013-08-28 17:50:29 +00:00
winMgr - > setValue ( mbar , stats . getMagicka ( ) ) ;
2012-07-22 14:29:54 +00:00
}
2016-06-27 12:17:06 +00:00
if ( stats . getFatigue ( ) ! = mWatchedFatigue | | mWatchedStatsEmpty )
2013-08-28 00:56:47 +00:00
{
2013-08-28 17:50:29 +00:00
static const std : : string fbar ( " FBar " ) ;
2016-06-27 12:17:06 +00:00
mWatchedFatigue = stats . getFatigue ( ) ;
2013-08-28 17:50:29 +00:00
winMgr - > setValue ( fbar , stats . getFatigue ( ) ) ;
2010-07-28 16:48:01 +00:00
}
2010-09-15 13:32:35 +00:00
2017-04-16 18:15:25 +00:00
float timeToDrown = stats . getTimeToStartDrowning ( ) ;
if ( timeToDrown ! = mWatchedTimeToStartDrowning )
2013-08-07 13:34:11 +00:00
{
2017-04-19 11:52:15 +00:00
static const float fHoldBreathTime = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( )
2018-08-29 15:38:12 +00:00
. find ( " fHoldBreathTime " ) - > mValue . getFloat ( ) ;
2017-04-19 11:52:15 +00:00
2017-04-16 18:15:25 +00:00
mWatchedTimeToStartDrowning = timeToDrown ;
if ( timeToDrown > = fHoldBreathTime | | timeToDrown = = - 1.0 ) // -1.0 is a special value during initialization
2013-08-28 00:56:47 +00:00
winMgr - > setDrowningBarVisibility ( false ) ;
2013-08-07 13:34:11 +00:00
else
{
2013-08-28 00:56:47 +00:00
winMgr - > setDrowningBarVisibility ( true ) ;
2014-04-27 02:27:26 +00:00
winMgr - > setDrowningTimeLeft ( stats . getTimeToStartDrowning ( ) , fHoldBreathTime ) ;
2013-08-07 13:34:11 +00:00
}
}
2011-01-02 16:35:03 +00:00
//Loop over ESM::Skill::SkillEnum
2013-08-28 00:56:47 +00:00
for ( int i = 0 ; i < ESM : : Skill : : Length ; + + i )
2010-09-21 15:42:07 +00:00
{
2016-06-27 12:17:06 +00:00
if ( stats . getSkill ( i ) ! = mWatchedSkills [ i ] | | mWatchedStatsEmpty )
2010-09-21 15:42:07 +00:00
{
2016-06-27 12:17:06 +00:00
mWatchedSkills [ i ] = stats . getSkill ( i ) ;
2013-08-28 00:56:47 +00:00
winMgr - > setValue ( ( ESM : : Skill : : SkillEnum ) i , stats . getSkill ( i ) ) ;
2010-09-21 15:42:07 +00:00
}
}
2019-01-24 18:16:42 +00:00
if ( stats . getLevel ( ) ! = mWatchedLevel )
{
mWatchedLevel = stats . getLevel ( ) ;
winMgr - > setValue ( " level " , mWatchedLevel ) ;
}
2014-04-26 14:44:20 +00:00
2014-09-13 00:56:06 +00:00
mWatchedStatsEmpty = false ;
2014-04-26 14:44:20 +00:00
// Update the equipped weapon icon
MWWorld : : InventoryStore & inv = mWatched . getClass ( ) . getInventoryStore ( mWatched ) ;
MWWorld : : ContainerStoreIterator weapon = inv . getSlot ( MWWorld : : InventoryStore : : Slot_CarriedRight ) ;
if ( weapon = = inv . end ( ) )
winMgr - > unsetSelectedWeapon ( ) ;
else
winMgr - > setSelectedWeapon ( * weapon ) ;
// Update the selected spell icon
MWWorld : : ContainerStoreIterator enchantItem = inv . getSelectedEnchantItem ( ) ;
if ( enchantItem ! = inv . end ( ) )
winMgr - > setSelectedEnchantItem ( * enchantItem ) ;
2014-05-12 19:04:02 +00:00
else
2019-01-24 18:16:42 +00:00
{
const std : : string & spell = winMgr - > getSelectedSpell ( ) ;
if ( ! spell . empty ( ) )
winMgr - > setSelectedSpell ( spell , int ( MWMechanics : : getSpellSuccessChance ( spell , mWatched ) ) ) ;
else
winMgr - > unsetSelectedSpell ( ) ;
}
2010-07-27 13:59:41 +00:00
}
2010-09-15 10:22:06 +00:00
2010-09-15 12:33:02 +00:00
if ( mUpdatePlayer )
2010-09-15 10:22:06 +00:00
{
2013-08-28 00:08:23 +00:00
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
2010-09-15 12:33:02 +00:00
// basic player profile; should not change anymore after the creation phase is finished.
2012-11-05 17:19:22 +00:00
MWBase : : WindowManager * winMgr =
MWBase : : Environment : : get ( ) . getWindowManager ( ) ;
2012-12-08 23:12:24 +00:00
2012-11-08 12:37:57 +00:00
const ESM : : NPC * player =
2014-01-08 17:39:44 +00:00
world - > getPlayerPtr ( ) . get < ESM : : NPC > ( ) - > mBase ;
2012-11-05 17:19:22 +00:00
2012-11-08 12:37:57 +00:00
const ESM : : Race * race =
world - > getStore ( ) . get < ESM : : Race > ( ) . find ( player - > mRace ) ;
const ESM : : Class * cls =
world - > getStore ( ) . get < ESM : : Class > ( ) . find ( player - > mClass ) ;
winMgr - > setValue ( " name " , player - > mName ) ;
winMgr - > setValue ( " race " , race - > mName ) ;
winMgr - > setValue ( " class " , cls - > mName ) ;
2012-11-05 17:19:22 +00:00
2010-09-15 12:33:02 +00:00
mUpdatePlayer = false ;
2010-09-21 15:42:07 +00:00
2012-08-12 16:11:09 +00:00
MWBase : : WindowManager : : SkillList majorSkills ( 5 ) ;
MWBase : : WindowManager : : SkillList minorSkills ( 5 ) ;
2010-09-21 15:42:07 +00:00
for ( int i = 0 ; i < 5 ; + + i )
{
2012-11-08 12:37:57 +00:00
minorSkills [ i ] = cls - > mData . mSkills [ i ] [ 0 ] ;
majorSkills [ i ] = cls - > mData . mSkills [ i ] [ 1 ] ;
2010-09-21 15:42:07 +00:00
}
2012-11-05 17:19:22 +00:00
winMgr - > configureSkills ( majorSkills , minorSkills ) ;
2013-02-04 20:58:06 +00:00
// HACK? The player has been changed, so a new Animation object may
// have been made for them. Make sure they're properly updated.
2015-08-21 09:12:39 +00:00
MWWorld : : Ptr ptr = getPlayer ( ) ;
2015-04-25 13:19:17 +00:00
mActors . removeActor ( ptr ) ;
mActors . addActor ( ptr , true ) ;
2010-09-15 10:22:06 +00:00
}
2011-02-03 10:43:29 +00:00
2015-04-25 13:19:17 +00:00
mActors . update ( duration , paused ) ;
mObjects . update ( duration , paused ) ;
2010-07-27 12:46:05 +00:00
}
2010-09-13 20:59:28 +00:00
2018-09-21 12:34:23 +00:00
void MechanicsManager : : processChangedSettings ( const Settings : : CategorySettingVector & changed )
{
for ( Settings : : CategorySettingVector : : const_iterator it = changed . begin ( ) ; it ! = changed . end ( ) ; + + it )
{
if ( it - > first = = " Game " & & it - > second = = " actors processing range " )
{
2018-10-28 11:08:24 +00:00
int state = MWBase : : Environment : : get ( ) . getStateManager ( ) - > getState ( ) ;
if ( state ! = MWBase : : StateManager : : State_Running )
continue ;
2018-09-21 12:34:23 +00:00
mActors . updateProcessingRange ( ) ;
// Update mechanics for new processing range immediately
update ( 0.f , false ) ;
}
}
}
2019-08-27 18:42:41 +00:00
void MechanicsManager : : notifyDied ( const MWWorld : : Ptr & actor )
{
mActors . notifyDied ( actor ) ;
}
2018-09-21 12:34:23 +00:00
float MechanicsManager : : getActorsProcessingRange ( ) const
{
return mActors . getProcessingRange ( ) ;
}
2017-09-18 17:46:57 +00:00
bool MechanicsManager : : isActorDetected ( const MWWorld : : Ptr & actor , const MWWorld : : Ptr & observer )
{
return mActors . isActorDetected ( actor , observer ) ;
}
2018-08-16 13:47:06 +00:00
bool MechanicsManager : : isAttackPreparing ( const MWWorld : : Ptr & ptr )
2017-09-01 05:34:15 +00:00
{
2018-08-16 13:47:06 +00:00
return mActors . isAttackPreparing ( ptr ) ;
2017-09-01 05:34:15 +00:00
}
2017-08-16 16:30:47 +00:00
bool MechanicsManager : : isRunning ( const MWWorld : : Ptr & ptr )
{
return mActors . isRunning ( ptr ) ;
}
bool MechanicsManager : : isSneaking ( const MWWorld : : Ptr & ptr )
{
2019-07-30 17:58:19 +00:00
CreatureStats & stats = ptr . getClass ( ) . getCreatureStats ( ptr ) ;
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
bool animActive = mActors . isSneaking ( ptr ) ;
bool stanceOn = stats . getStance ( MWMechanics : : CreatureStats : : Stance_Sneak ) ;
bool inair = ! world - > isOnGround ( ptr ) & & ! world - > isSwimming ( ptr ) & & ! world - > isFlying ( ptr ) ;
return stanceOn & & ( animActive | | inair ) ;
2017-08-16 16:30:47 +00:00
}
2019-01-25 16:04:35 +00:00
void MechanicsManager : : rest ( double hours , bool sleep )
2012-09-21 15:53:16 +00:00
{
2018-09-23 18:03:43 +00:00
if ( sleep )
2019-01-25 16:04:35 +00:00
MWBase : : Environment : : get ( ) . getWorld ( ) - > rest ( hours ) ;
2018-09-23 18:03:43 +00:00
2019-01-25 16:04:35 +00:00
mActors . rest ( hours , sleep ) ;
2012-09-21 15:53:16 +00:00
}
2019-01-25 16:04:35 +00:00
void MechanicsManager : : restoreDynamicStats ( MWWorld : : Ptr actor , double hours , bool sleep )
2018-09-23 18:03:43 +00:00
{
2019-01-25 16:04:35 +00:00
mActors . restoreDynamicStats ( actor , hours , sleep ) ;
2018-09-23 18:03:43 +00:00
}
2014-01-14 01:52:34 +00:00
int MechanicsManager : : getHoursToRest ( ) const
{
2015-04-25 13:19:17 +00:00
return mActors . getHoursToRest ( mWatched ) ;
2014-01-14 01:52:34 +00:00
}
2010-09-13 20:59:28 +00:00
void MechanicsManager : : setPlayerName ( const std : : string & name )
{
2012-11-08 12:37:57 +00:00
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
ESM : : NPC player =
2014-01-08 17:39:44 +00:00
* world - > getPlayerPtr ( ) . get < ESM : : NPC > ( ) - > mBase ;
2012-11-08 12:37:57 +00:00
player . mName = name ;
world - > createRecord ( player ) ;
2010-09-15 12:33:02 +00:00
mUpdatePlayer = true ;
2010-09-13 20:59:28 +00:00
}
2012-11-10 07:41:12 +00:00
void MechanicsManager : : setPlayerRace ( const std : : string & race , bool male , const std : : string & head , const std : : string & hair )
2010-09-13 20:59:28 +00:00
{
2012-11-08 12:37:57 +00:00
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
ESM : : NPC player =
2014-01-08 17:39:44 +00:00
* world - > getPlayerPtr ( ) . get < ESM : : NPC > ( ) - > mBase ;
2012-11-08 12:37:57 +00:00
player . mRace = race ;
2012-11-10 07:41:12 +00:00
player . mHead = head ;
player . mHair = hair ;
player . setIsMale ( male ) ;
2012-11-08 12:37:57 +00:00
world - > createRecord ( player ) ;
2010-09-26 08:01:30 +00:00
mRaceSelected = true ;
2010-09-15 11:31:26 +00:00
buildPlayer ( ) ;
2010-09-15 12:33:02 +00:00
mUpdatePlayer = true ;
2010-09-13 20:59:28 +00:00
}
void MechanicsManager : : setPlayerBirthsign ( const std : : string & id )
{
2012-11-08 14:50:18 +00:00
MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . setBirthSign ( id ) ;
2010-09-15 11:31:26 +00:00
buildPlayer ( ) ;
2010-10-22 23:28:30 +00:00
mUpdatePlayer = true ;
2010-09-13 20:59:28 +00:00
}
void MechanicsManager : : setPlayerClass ( const std : : string & id )
{
2012-11-08 12:37:57 +00:00
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
ESM : : NPC player =
2014-01-08 17:39:44 +00:00
* world - > getPlayerPtr ( ) . get < ESM : : NPC > ( ) - > mBase ;
2012-11-08 12:37:57 +00:00
player . mClass = id ;
world - > createRecord ( player ) ;
2010-09-26 08:01:30 +00:00
mClassSelected = true ;
2010-09-15 11:31:26 +00:00
buildPlayer ( ) ;
2010-09-15 12:33:02 +00:00
mUpdatePlayer = true ;
2010-09-13 20:59:28 +00:00
}
2012-11-07 21:36:43 +00:00
void MechanicsManager : : setPlayerClass ( const ESM : : Class & cls )
2010-09-13 20:59:28 +00:00
{
2012-11-07 21:36:43 +00:00
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
2012-11-07 21:52:34 +00:00
const ESM : : Class * ptr = world - > createRecord ( cls ) ;
2012-11-07 21:36:43 +00:00
2012-11-08 12:37:57 +00:00
ESM : : NPC player =
2014-01-08 17:39:44 +00:00
* world - > getPlayerPtr ( ) . get < ESM : : NPC > ( ) - > mBase ;
2012-11-08 12:37:57 +00:00
player . mClass = ptr - > mId ;
world - > createRecord ( player ) ;
2012-11-07 21:36:43 +00:00
2010-09-26 08:01:30 +00:00
mClassSelected = true ;
2010-09-15 11:31:26 +00:00
buildPlayer ( ) ;
2010-09-15 12:33:02 +00:00
mUpdatePlayer = true ;
2010-09-13 20:59:28 +00:00
}
2012-11-05 10:07:43 +00:00
2016-03-17 23:38:02 +00:00
int MechanicsManager : : getDerivedDisposition ( const MWWorld : : Ptr & ptr , bool addTemporaryDispositionChange )
2012-11-05 10:07:43 +00:00
{
2014-05-22 18:37:22 +00:00
const MWMechanics : : NpcStats & npcSkill = ptr . getClass ( ) . getNpcStats ( ptr ) ;
2015-03-08 00:07:29 +00:00
float x = static_cast < float > ( npcSkill . getBaseDisposition ( ) ) ;
2012-11-05 18:55:06 +00:00
MWWorld : : LiveCellRef < ESM : : NPC > * npc = ptr . get < ESM : : NPC > ( ) ;
2015-08-21 09:12:39 +00:00
MWWorld : : Ptr playerPtr = getPlayer ( ) ;
2012-11-05 18:55:06 +00:00
MWWorld : : LiveCellRef < ESM : : NPC > * player = playerPtr . get < ESM : : NPC > ( ) ;
2014-05-22 18:37:22 +00:00
const MWMechanics : : NpcStats & playerStats = playerPtr . getClass ( ) . getNpcStats ( playerPtr ) ;
2012-11-05 18:55:06 +00:00
2014-09-27 20:44:20 +00:00
const MWWorld : : Store < ESM : : GameSetting > & gmst = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
2018-08-29 15:38:12 +00:00
static const float fDispRaceMod = gmst . find ( " fDispRaceMod " ) - > mValue . getFloat ( ) ;
2013-11-21 03:53:53 +00:00
if ( Misc : : StringUtils : : ciEqual ( npc - > mBase - > mRace , player - > mBase - > mRace ) )
2014-09-27 20:44:20 +00:00
x + = fDispRaceMod ;
2012-11-05 18:55:06 +00:00
2018-08-29 15:38:12 +00:00
static const float fDispPersonalityMult = gmst . find ( " fDispPersonalityMult " ) - > mValue . getFloat ( ) ;
static const float fDispPersonalityBase = gmst . find ( " fDispPersonalityBase " ) - > mValue . getFloat ( ) ;
2014-09-27 20:44:20 +00:00
x + = fDispPersonalityMult * ( playerStats . getAttribute ( ESM : : Attribute : : Personality ) . getModified ( ) - fDispPersonalityBase ) ;
2012-11-05 18:55:06 +00:00
float reaction = 0 ;
int rank = 0 ;
2015-01-27 16:32:21 +00:00
std : : string npcFaction = ptr . getClass ( ) . getPrimaryFaction ( ptr ) ;
2012-11-05 18:55:06 +00:00
2015-12-07 21:49:15 +00:00
Misc : : StringUtils : : lowerCaseInPlace ( npcFaction ) ;
2014-01-14 08:46:53 +00:00
if ( playerStats . getFactionRanks ( ) . find ( npcFaction ) ! = playerStats . getFactionRanks ( ) . end ( ) )
2012-11-05 18:55:06 +00:00
{
2014-05-27 12:54:29 +00:00
if ( ! playerStats . getExpelled ( npcFaction ) )
2012-11-05 18:55:06 +00:00
{
2014-06-24 22:11:51 +00:00
// faction reaction towards itself. yes, that exists
2015-03-08 00:07:29 +00:00
reaction = static_cast < float > ( MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > getFactionReaction ( npcFaction , npcFaction ) ) ;
2014-05-27 12:54:29 +00:00
rank = playerStats . getFactionRanks ( ) . find ( npcFaction ) - > second ;
2012-11-05 18:55:06 +00:00
}
}
2014-05-27 12:54:29 +00:00
else if ( ! npcFaction . empty ( ) )
2012-11-05 18:55:06 +00:00
{
2014-05-27 12:54:29 +00:00
std : : map < std : : string , int > : : const_iterator playerFactionIt = playerStats . getFactionRanks ( ) . begin ( ) ;
for ( ; playerFactionIt ! = playerStats . getFactionRanks ( ) . end ( ) ; + + playerFactionIt )
2012-11-05 18:55:06 +00:00
{
2014-05-27 12:54:29 +00:00
std : : string itFaction = playerFactionIt - > first ;
2018-02-10 13:58:30 +00:00
// Ignore the faction, if a player was expelled from it.
if ( playerStats . getExpelled ( itFaction ) )
continue ;
2014-05-27 12:54:29 +00:00
int itReaction = MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > getFactionReaction ( npcFaction , itFaction ) ;
if ( playerFactionIt = = playerStats . getFactionRanks ( ) . begin ( ) | | itReaction < reaction )
2018-02-10 13:58:30 +00:00
{
2015-03-08 00:07:29 +00:00
reaction = static_cast < float > ( itReaction ) ;
2018-02-10 13:58:30 +00:00
rank = playerFactionIt - > second ;
}
2012-11-05 18:55:06 +00:00
}
}
else
{
reaction = 0 ;
rank = 0 ;
}
2012-11-08 12:38:20 +00:00
2018-08-29 15:38:12 +00:00
static const float fDispFactionRankMult = gmst . find ( " fDispFactionRankMult " ) - > mValue . getFloat ( ) ;
static const float fDispFactionRankBase = gmst . find ( " fDispFactionRankBase " ) - > mValue . getFloat ( ) ;
static const float fDispFactionMod = gmst . find ( " fDispFactionMod " ) - > mValue . getFloat ( ) ;
2014-09-27 20:44:20 +00:00
x + = ( fDispFactionRankMult * rank
+ fDispFactionRankBase )
* fDispFactionMod * reaction ;
2018-08-29 15:38:12 +00:00
static const float fDispCrimeMod = gmst . find ( " fDispCrimeMod " ) - > mValue . getFloat ( ) ;
static const float fDispDiseaseMod = gmst . find ( " fDispDiseaseMod " ) - > mValue . getFloat ( ) ;
2014-09-27 20:44:20 +00:00
x - = fDispCrimeMod * playerStats . getBounty ( ) ;
2012-11-09 17:33:11 +00:00
if ( playerStats . hasCommonDisease ( ) | | playerStats . hasBlightDisease ( ) )
2014-09-27 20:44:20 +00:00
x + = fDispDiseaseMod ;
2012-11-09 17:33:11 +00:00
2018-08-29 15:38:12 +00:00
static const float fDispWeaponDrawn = gmst . find ( " fDispWeaponDrawn " ) - > mValue . getFloat ( ) ;
2013-08-09 08:14:08 +00:00
if ( playerStats . getDrawState ( ) = = MWMechanics : : DrawState_Weapon )
2014-09-27 20:44:20 +00:00
x + = fDispWeaponDrawn ;
2012-11-05 18:55:06 +00:00
2014-08-16 20:38:22 +00:00
x + = ptr . getClass ( ) . getCreatureStats ( ptr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Charm ) . getMagnitude ( ) ;
2014-01-05 00:56:36 +00:00
2016-03-08 20:30:28 +00:00
if ( addTemporaryDispositionChange )
x + = MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > getTemporaryDispositionChange ( ) ;
2016-03-08 19:41:39 +00:00
int effective_disposition = std : : max ( 0 , std : : min ( int ( x ) , 100 ) ) ; //, normally clamped to [0..100] when used
2012-11-05 18:55:06 +00:00
return effective_disposition ;
2012-11-05 10:07:43 +00:00
}
2012-11-09 13:42:09 +00:00
int MechanicsManager : : getBarterOffer ( const MWWorld : : Ptr & ptr , int basePrice , bool buying )
2012-11-05 10:07:43 +00:00
{
2012-11-08 21:31:08 +00:00
if ( ptr . getTypeName ( ) = = typeid ( ESM : : Creature ) . name ( ) )
return basePrice ;
2014-05-22 18:37:22 +00:00
const MWMechanics : : NpcStats & sellerStats = ptr . getClass ( ) . getNpcStats ( ptr ) ;
2012-11-05 10:07:43 +00:00
2015-08-21 09:12:39 +00:00
MWWorld : : Ptr playerPtr = getPlayer ( ) ;
2014-05-22 18:37:22 +00:00
const MWMechanics : : NpcStats & playerStats = playerPtr . getClass ( ) . getNpcStats ( playerPtr ) ;
2012-11-05 10:07:43 +00:00
2016-03-08 20:30:28 +00:00
// I suppose the temporary disposition change (second param to getDerivedDisposition()) _has_ to be considered here,
2012-11-09 23:42:31 +00:00
// otherwise one would get different prices when exiting and re-entering the dialogue window...
2016-03-17 23:41:24 +00:00
int clampedDisposition = getDerivedDisposition ( ptr ) ;
2018-10-08 14:06:30 +00:00
float a = static_cast < float > ( std : : min ( playerPtr . getClass ( ) . getSkill ( playerPtr , ESM : : Skill : : Mercantile ) , 100 ) ) ;
2012-11-08 21:37:59 +00:00
float b = std : : min ( 0.1f * playerStats . getAttribute ( ESM : : Attribute : : Luck ) . getModified ( ) , 10.f ) ;
float c = std : : min ( 0.2f * playerStats . getAttribute ( ESM : : Attribute : : Personality ) . getModified ( ) , 10.f ) ;
2018-10-08 14:06:30 +00:00
float d = static_cast < float > ( std : : min ( ptr . getClass ( ) . getSkill ( ptr , ESM : : Skill : : Mercantile ) , 100 ) ) ;
2012-11-08 21:37:59 +00:00
float e = std : : min ( 0.1f * sellerStats . getAttribute ( ESM : : Attribute : : Luck ) . getModified ( ) , 10.f ) ;
float f = std : : min ( 0.2f * sellerStats . getAttribute ( ESM : : Attribute : : Personality ) . getModified ( ) , 10.f ) ;
2012-11-05 10:07:43 +00:00
float pcTerm = ( clampedDisposition - 50 + a + b + c ) * playerStats . getFatigueTerm ( ) ;
float npcTerm = ( d + e + f ) * sellerStats . getFatigueTerm ( ) ;
2019-05-09 17:42:22 +00:00
float buyTerm = 0.01f * ( 100 - 0.5f * ( pcTerm - npcTerm ) ) ;
float sellTerm = 0.01f * ( 50 - 0.5f * ( npcTerm - pcTerm ) ) ;
2018-03-25 10:05:42 +00:00
int offerPrice = int ( basePrice * ( buying ? buyTerm : sellTerm ) ) ;
return std : : max ( 1 , offerPrice ) ;
2012-11-05 10:07:43 +00:00
}
2012-11-08 22:16:40 +00:00
2012-10-27 09:33:18 +00:00
int MechanicsManager : : countDeaths ( const std : : string & id ) const
{
2015-04-25 13:19:17 +00:00
return mActors . countDeaths ( id ) ;
2012-10-27 09:33:18 +00:00
}
2012-11-09 19:18:38 +00:00
2017-06-10 11:04:19 +00:00
/*
Start of tes3mp addition
Make it possible to set the number of deaths for an actor with the given refId
*/
void MechanicsManager : : setDeaths ( const std : : string & refId , int number )
{
mActors . setDeaths ( refId , number ) ;
}
/*
End of tes3mp addition
*/
2016-03-17 23:39:19 +00:00
void MechanicsManager : : getPersuasionDispositionChange ( const MWWorld : : Ptr & npc , PersuasionType type , bool & success , float & tempChange , float & permChange )
2012-11-09 19:18:38 +00:00
{
2012-11-09 23:29:36 +00:00
const MWWorld : : Store < ESM : : GameSetting > & gmst =
MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
2014-05-22 18:37:22 +00:00
MWMechanics : : NpcStats & npcStats = npc . getClass ( ) . getNpcStats ( npc ) ;
2012-11-09 23:29:36 +00:00
2015-08-21 09:12:39 +00:00
MWWorld : : Ptr playerPtr = getPlayer ( ) ;
2014-05-22 18:37:22 +00:00
const MWMechanics : : NpcStats & playerStats = playerPtr . getClass ( ) . getNpcStats ( playerPtr ) ;
2012-11-09 23:29:36 +00:00
2015-04-21 23:17:01 +00:00
float npcRating1 , npcRating2 , npcRating3 ;
getPersuasionRatings ( npcStats , npcRating1 , npcRating2 , npcRating3 , false ) ;
2012-11-09 23:29:36 +00:00
2015-04-21 23:17:01 +00:00
float playerRating1 , playerRating2 , playerRating3 ;
getPersuasionRatings ( playerStats , playerRating1 , playerRating2 , playerRating3 , true ) ;
2012-11-09 23:29:36 +00:00
2016-03-17 23:39:19 +00:00
int currentDisposition = getDerivedDisposition ( npc ) ;
2012-11-09 23:29:36 +00:00
2015-03-08 00:07:29 +00:00
float d = 1 - 0.02f * abs ( currentDisposition - 50 ) ;
2012-11-09 23:29:36 +00:00
float target1 = d * ( playerRating1 - npcRating1 + 50 ) ;
float target2 = d * ( playerRating2 - npcRating2 + 50 ) ;
float bribeMod ;
2018-08-29 15:38:12 +00:00
if ( type = = PT_Bribe10 ) bribeMod = gmst . find ( " fBribe10Mod " ) - > mValue . getFloat ( ) ;
else if ( type = = PT_Bribe100 ) bribeMod = gmst . find ( " fBribe100Mod " ) - > mValue . getFloat ( ) ;
else bribeMod = gmst . find ( " fBribe1000Mod " ) - > mValue . getFloat ( ) ;
2012-11-09 23:29:36 +00:00
float target3 = d * ( playerRating3 - npcRating3 + 50 ) + bribeMod ;
2018-08-29 15:38:12 +00:00
float iPerMinChance = floor ( gmst . find ( " iPerMinChance " ) - > mValue . getFloat ( ) ) ;
float iPerMinChange = floor ( gmst . find ( " iPerMinChange " ) - > mValue . getFloat ( ) ) ;
float fPerDieRollMult = gmst . find ( " fPerDieRollMult " ) - > mValue . getFloat ( ) ;
float fPerTempMult = gmst . find ( " fPerTempMult " ) - > mValue . getFloat ( ) ;
2012-11-09 23:29:36 +00:00
2013-02-25 15:52:31 +00:00
float x = 0 ;
float y = 0 ;
2012-11-09 23:29:36 +00:00
2015-05-23 18:33:44 +00:00
int roll = Misc : : Rng : : roll0to99 ( ) ;
2012-11-09 23:29:36 +00:00
if ( type = = PT_Admire )
{
target1 = std : : max ( iPerMinChance , target1 ) ;
success = ( roll < = target1 ) ;
2015-03-08 00:07:29 +00:00
float c = floor ( fPerDieRollMult * ( target1 - roll ) ) ;
2012-11-09 23:29:36 +00:00
x = success ? std : : max ( iPerMinChange , c ) : c ;
}
else if ( type = = PT_Intimidate )
{
target2 = std : : max ( iPerMinChance , target2 ) ;
success = ( roll < = target2 ) ;
float r ;
if ( roll ! = target2 )
2015-03-08 00:07:29 +00:00
r = floor ( target2 - roll ) ;
2012-11-09 23:29:36 +00:00
else
r = 1 ;
if ( roll < = target2 )
{
2015-03-08 00:07:29 +00:00
float s = floor ( r * fPerDieRollMult * fPerTempMult ) ;
2012-11-09 23:29:36 +00:00
2014-01-05 00:34:35 +00:00
int flee = npcStats . getAiSetting ( MWMechanics : : CreatureStats : : AI_Flee ) . getBase ( ) ;
int fight = npcStats . getAiSetting ( MWMechanics : : CreatureStats : : AI_Fight ) . getBase ( ) ;
npcStats . setAiSetting ( MWMechanics : : CreatureStats : : AI_Flee ,
std : : max ( 0 , std : : min ( 100 , flee + int ( std : : max ( iPerMinChange , s ) ) ) ) ) ;
npcStats . setAiSetting ( MWMechanics : : CreatureStats : : AI_Fight ,
std : : max ( 0 , std : : min ( 100 , fight + int ( std : : min ( - iPerMinChange , - s ) ) ) ) ) ;
2012-11-09 23:29:36 +00:00
}
2015-03-08 00:07:29 +00:00
float c = - std : : abs ( floor ( r * fPerDieRollMult ) ) ;
2012-11-09 23:29:36 +00:00
if ( success )
{
if ( std : : abs ( c ) < iPerMinChange )
{
x = 0 ;
y = - iPerMinChange ;
}
else
{
2015-03-08 00:07:29 +00:00
x = - floor ( c * fPerTempMult ) ;
2012-11-09 23:29:36 +00:00
y = c ;
}
}
else
{
2015-03-08 00:07:29 +00:00
x = floor ( c * fPerTempMult ) ;
2012-11-09 23:29:36 +00:00
y = c ;
}
}
else if ( type = = PT_Taunt )
{
target1 = std : : max ( iPerMinChance , target1 ) ;
success = ( roll < = target1 ) ;
2015-03-08 00:07:29 +00:00
float c = std : : abs ( floor ( target1 - roll ) ) ;
2012-11-09 23:29:36 +00:00
2014-01-08 01:35:36 +00:00
if ( success )
2012-11-09 23:29:36 +00:00
{
float s = c * fPerDieRollMult * fPerTempMult ;
2014-01-05 00:34:35 +00:00
int flee = npcStats . getAiSetting ( CreatureStats : : AI_Flee ) . getBase ( ) ;
int fight = npcStats . getAiSetting ( CreatureStats : : AI_Fight ) . getBase ( ) ;
npcStats . setAiSetting ( CreatureStats : : AI_Flee ,
std : : max ( 0 , std : : min ( 100 , flee + std : : min ( - int ( iPerMinChange ) , int ( - s ) ) ) ) ) ;
npcStats . setAiSetting ( CreatureStats : : AI_Fight ,
std : : max ( 0 , std : : min ( 100 , fight + std : : max ( int ( iPerMinChange ) , int ( s ) ) ) ) ) ;
2012-11-09 23:29:36 +00:00
}
2015-03-08 00:07:29 +00:00
x = floor ( - c * fPerDieRollMult ) ;
2012-11-09 23:29:36 +00:00
if ( success & & std : : abs ( x ) < iPerMinChange )
x = - iPerMinChange ;
}
else // Bribe
{
target3 = std : : max ( iPerMinChance , target3 ) ;
success = ( roll < = target3 ) ;
2015-03-08 00:07:29 +00:00
float c = floor ( ( target3 - roll ) * fPerDieRollMult ) ;
2012-11-09 23:29:36 +00:00
x = success ? std : : max ( iPerMinChange , c ) : c ;
}
tempChange = type = = PT_Intimidate ? x : int ( x * fPerTempMult ) ;
float cappedDispositionChange = tempChange ;
if ( currentDisposition + tempChange > 100.f )
2015-03-08 00:07:29 +00:00
cappedDispositionChange = static_cast < float > ( 100 - currentDisposition ) ;
2012-11-09 23:29:36 +00:00
if ( currentDisposition + tempChange < 0.f )
2015-03-08 00:07:29 +00:00
cappedDispositionChange = static_cast < float > ( - currentDisposition ) ;
2012-11-09 23:29:36 +00:00
2015-03-08 00:07:29 +00:00
permChange = floor ( cappedDispositionChange / fPerTempMult ) ;
2012-11-09 23:29:36 +00:00
if ( type = = PT_Intimidate )
{
permChange = success ? - int ( cappedDispositionChange / fPerTempMult ) : y ;
}
2012-11-09 19:18:38 +00:00
}
2013-01-17 01:53:18 +00:00
2013-04-25 14:08:11 +00:00
void MechanicsManager : : forceStateUpdate ( const MWWorld : : Ptr & ptr )
{
2015-04-25 13:19:17 +00:00
if ( ptr . getClass ( ) . isActor ( ) )
mActors . forceStateUpdate ( ptr ) ;
2013-04-25 14:08:11 +00:00
}
2016-07-30 17:24:03 +00:00
bool MechanicsManager : : playAnimationGroup ( const MWWorld : : Ptr & ptr , const std : : string & groupName , int mode , int number , bool persist )
2013-01-17 01:53:18 +00:00
{
2014-05-22 18:37:22 +00:00
if ( ptr . getClass ( ) . isActor ( ) )
2016-07-30 17:24:03 +00:00
return mActors . playAnimationGroup ( ptr , groupName , mode , number , persist ) ;
2013-03-31 22:12:10 +00:00
else
2016-07-30 17:24:03 +00:00
return mObjects . playAnimationGroup ( ptr , groupName , mode , number , persist ) ;
2013-01-17 01:53:18 +00:00
}
void MechanicsManager : : skipAnimation ( const MWWorld : : Ptr & ptr )
{
2014-05-22 18:37:22 +00:00
if ( ptr . getClass ( ) . isActor ( ) )
2013-01-29 07:39:11 +00:00
mActors . skipAnimation ( ptr ) ;
2013-03-31 22:12:10 +00:00
else
2013-03-31 22:29:41 +00:00
mObjects . skipAnimation ( ptr ) ;
2013-01-17 01:53:18 +00:00
}
2013-05-25 03:10:07 +00:00
bool MechanicsManager : : checkAnimationPlaying ( const MWWorld : : Ptr & ptr , const std : : string & groupName )
{
2015-04-25 13:19:17 +00:00
if ( ptr . getClass ( ) . isActor ( ) )
return mActors . checkAnimationPlaying ( ptr , groupName ) ;
else
2013-05-25 03:10:07 +00:00
return false ;
}
2013-01-17 01:53:18 +00:00
2018-09-22 08:57:50 +00:00
bool MechanicsManager : : onOpen ( const MWWorld : : Ptr & ptr )
{
if ( ptr . getClass ( ) . isActor ( ) )
return true ;
else
return mObjects . onOpen ( ptr ) ;
}
void MechanicsManager : : onClose ( const MWWorld : : Ptr & ptr )
{
if ( ! ptr . getClass ( ) . isActor ( ) )
mObjects . onClose ( ptr ) ;
}
2016-07-30 17:24:03 +00:00
void MechanicsManager : : persistAnimationStates ( )
{
mActors . persistAnimationStates ( ) ;
mObjects . persistAnimationStates ( ) ;
}
2013-11-17 22:15:57 +00:00
void MechanicsManager : : updateMagicEffects ( const MWWorld : : Ptr & ptr )
{
2015-04-25 13:19:17 +00:00
mActors . updateMagicEffects ( ptr ) ;
2013-11-17 22:15:57 +00:00
}
2014-03-26 18:55:52 +00:00
bool MechanicsManager : : toggleAI ( )
2013-11-18 22:03:44 +00:00
{
mAI = ! mAI ;
2014-03-26 18:55:52 +00:00
return mAI ;
2013-11-18 22:03:44 +00:00
}
bool MechanicsManager : : isAIActive ( )
{
return mAI ;
}
2014-01-06 23:51:09 +00:00
2013-12-07 12:17:28 +00:00
void MechanicsManager : : playerLoaded ( )
{
mUpdatePlayer = true ;
mClassSelected = true ;
mRaceSelected = true ;
2018-04-12 14:18:06 +00:00
/*
Start of tes3mp change ( major )
Avoid enabling AI in multiplayer
*/
mAI = false ;
/*
End of tes3mp change ( major )
*/
2013-12-07 12:17:28 +00:00
}
2014-01-24 17:21:52 +00:00
2018-08-11 13:02:09 +00:00
/*
Start of tes3mp change ( major )
Move boundItemIDCache outside of the original isBoundItem ( const MWWorld : : Ptr & item )
method so it can be reused in the new isBoundItem ( std : : string itemId ) method
*/
std : : set < std : : string > boundItemIDCache ;
2017-11-10 05:43:22 +00:00
bool MechanicsManager : : isBoundItem ( const MWWorld : : Ptr & item )
{
2018-08-11 13:02:09 +00:00
/*
End of tes3mp change ( major )
*/
2017-11-10 05:43:22 +00:00
// If this is empty then we haven't executed the GMST cache logic yet; or there isn't any sMagicBound* GMST's for some reason
if ( boundItemIDCache . empty ( ) )
{
// Build a list of known bound item ID's
const MWWorld : : Store < ESM : : GameSetting > & gameSettings = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
2018-11-05 22:35:20 +00:00
for ( const ESM : : GameSetting & currentSetting : gameSettings )
2017-11-10 05:43:22 +00:00
{
std : : string currentGMSTID = currentSetting . mId ;
Misc : : StringUtils : : lowerCaseInPlace ( currentGMSTID ) ;
// Don't bother checking this GMST if it's not a sMagicBound* one.
const std : : string & toFind = " smagicbound " ;
if ( currentGMSTID . compare ( 0 , toFind . length ( ) , toFind ) ! = 0 )
continue ;
// All sMagicBound* GMST's should be of type string
2018-08-29 15:38:12 +00:00
std : : string currentGMSTValue = currentSetting . mValue . getString ( ) ;
2017-11-10 05:43:22 +00:00
Misc : : StringUtils : : lowerCaseInPlace ( currentGMSTValue ) ;
boundItemIDCache . insert ( currentGMSTValue ) ;
}
}
// Perform bound item check and assign the Flag_Bound bit if it passes
std : : string tempItemID = item . getCellRef ( ) . getRefId ( ) ;
Misc : : StringUtils : : lowerCaseInPlace ( tempItemID ) ;
if ( boundItemIDCache . count ( tempItemID ) ! = 0 )
return true ;
return false ;
}
2018-08-11 13:02:09 +00:00
/*
Start of tes3mp addition
Make it possible to check if an itemId corresponds to a bound item
*/
bool MechanicsManager : : isBoundItem ( std : : string itemId )
{
Misc : : StringUtils : : lowerCaseInPlace ( itemId ) ;
if ( boundItemIDCache . count ( itemId ) ! = 0 )
return true ;
return false ;
}
/*
End of tes3mp addition
*/
2017-09-30 16:22:26 +00:00
bool MechanicsManager : : isAllowedToUse ( const MWWorld : : Ptr & ptr , const MWWorld : : Ptr & target , MWWorld : : Ptr & victim )
2014-09-22 13:18:19 +00:00
{
2017-10-03 05:59:31 +00:00
if ( target . isEmpty ( ) )
return true ;
2017-09-30 16:22:26 +00:00
const MWWorld : : CellRef & cellref = target . getCellRef ( ) ;
2017-08-18 13:06:47 +00:00
// there is no harm to use unlocked doors
2018-06-19 10:17:33 +00:00
int lockLevel = cellref . getLockLevel ( ) ;
if ( target . getClass ( ) . isDoor ( ) & &
( lockLevel < = 0 | | lockLevel = = ESM : : UnbreakableLock ) & &
ptr . getCellRef ( ) . getTrap ( ) . empty ( ) )
{
2017-08-18 13:06:47 +00:00
return true ;
2018-06-19 10:17:33 +00:00
}
2017-08-18 13:06:47 +00:00
2019-09-10 18:56:10 +00:00
if ( ! target . getClass ( ) . hasToolTip ( target ) )
2018-12-03 16:21:40 +00:00
return true ;
2017-09-30 16:22:26 +00:00
// TODO: implement a better check to check if target is owned bed
if ( target . getClass ( ) . isActivator ( ) & & target . getClass ( ) . getScript ( target ) . compare ( 0 , 3 , " Bed " ) ! = 0 )
2017-08-18 13:06:47 +00:00
return true ;
2017-09-30 16:22:26 +00:00
if ( target . getClass ( ) . isNpc ( ) )
{
if ( target . getClass ( ) . getCreatureStats ( target ) . isDead ( ) )
return true ;
2017-09-30 17:29:02 +00:00
if ( target . getClass ( ) . getCreatureStats ( target ) . getAiSequence ( ) . isInCombat ( ) )
return true ;
2017-09-30 16:22:26 +00:00
// check if a player tries to pickpocket a target NPC
2019-07-30 17:58:19 +00:00
if ( target . getClass ( ) . getCreatureStats ( target ) . getKnockedDown ( ) | | isSneaking ( ptr ) )
2017-09-30 16:22:26 +00:00
return false ;
return true ;
}
2015-02-04 20:18:43 +00:00
const std : : string & owner = cellref . getOwner ( ) ;
2014-09-22 13:18:19 +00:00
bool isOwned = ! owner . empty ( ) & & owner ! = " player " ;
2015-02-04 20:18:43 +00:00
const std : : string & faction = cellref . getFaction ( ) ;
2014-09-22 13:18:19 +00:00
bool isFactionOwned = false ;
if ( ! faction . empty ( ) & & ptr . getClass ( ) . isNpc ( ) )
{
const std : : map < std : : string , int > & factions = ptr . getClass ( ) . getNpcStats ( ptr ) . getFactionRanks ( ) ;
std : : map < std : : string , int > : : const_iterator found = factions . find ( Misc : : StringUtils : : lowerCase ( faction ) ) ;
if ( found = = factions . end ( )
2015-02-04 20:18:43 +00:00
| | found - > second < cellref . getFactionRank ( ) )
2014-09-22 13:18:19 +00:00
isFactionOwned = true ;
}
2015-02-04 20:18:43 +00:00
const std : : string & globalVariable = cellref . getGlobalVariable ( ) ;
2014-09-22 13:18:19 +00:00
if ( ! globalVariable . empty ( ) & & MWBase : : Environment : : get ( ) . getWorld ( ) - > getGlobalInt ( Misc : : StringUtils : : lowerCase ( globalVariable ) ) = = 1 )
{
isOwned = false ;
isFactionOwned = false ;
}
2015-02-04 20:18:43 +00:00
if ( ! cellref . getOwner ( ) . empty ( ) )
2019-05-06 19:06:13 +00:00
victim = MWBase : : Environment : : get ( ) . getWorld ( ) - > searchPtr ( cellref . getOwner ( ) , true , false ) ;
2014-09-22 13:18:19 +00:00
2019-09-16 10:18:41 +00:00
// A special case for evidence chest - we should not allow to take items even if it is technically permitted
if ( Misc : : StringUtils : : ciEqual ( cellref . getRefId ( ) , " stolen_goods " ) )
return false ;
2014-09-22 13:18:19 +00:00
return ( ! isOwned & & ! isFactionOwned ) ;
}
2014-01-08 16:19:43 +00:00
bool MechanicsManager : : sleepInBed ( const MWWorld : : Ptr & ptr , const MWWorld : : Ptr & bed )
2014-01-07 18:49:16 +00:00
{
2014-09-04 23:58:57 +00:00
if ( ptr . getClass ( ) . getNpcStats ( ptr ) . isWerewolf ( ) )
{
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > messageBox ( " #{sWerewolfRefusal} " ) ;
return true ;
}
2016-06-06 23:53:16 +00:00
if ( MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . enemiesNearby ( ) ) {
2014-04-25 02:47:45 +00:00
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > messageBox ( " #{sNotifyMessage2} " ) ;
return true ;
}
2014-01-08 16:19:43 +00:00
MWWorld : : Ptr victim ;
2017-08-18 13:06:47 +00:00
if ( isAllowedToUse ( ptr , bed , victim ) )
2014-01-10 20:26:24 +00:00
return false ;
2014-01-07 19:24:01 +00:00
2019-09-16 10:20:43 +00:00
if ( commitCrime ( ptr , victim , OT_SleepingInOwnedBed , bed . getCellRef ( ) . getFaction ( ) ) )
2014-01-07 19:24:01 +00:00
{
2014-01-08 16:19:43 +00:00
MWBase : : Environment : : get ( ) . getWindowManager ( ) - > messageBox ( " #{sNotifyMessage64} " ) ;
return true ;
2014-01-07 19:24:01 +00:00
}
2014-01-08 16:19:43 +00:00
else
return false ;
}
2014-01-07 19:24:01 +00:00
2019-09-05 14:36:49 +00:00
void MechanicsManager : : unlockAttempted ( const MWWorld : : Ptr & ptr , const MWWorld : : Ptr & item )
2014-01-08 16:19:43 +00:00
{
2014-01-07 19:24:01 +00:00
MWWorld : : Ptr victim ;
2017-08-18 13:06:47 +00:00
if ( isAllowedToUse ( ptr , item , victim ) )
2014-01-10 20:26:24 +00:00
return ;
2019-09-16 10:20:43 +00:00
commitCrime ( ptr , victim , OT_Trespassing , item . getCellRef ( ) . getFaction ( ) ) ;
2014-01-10 20:26:24 +00:00
}
2014-01-08 00:24:06 +00:00
2015-02-04 20:18:43 +00:00
std : : vector < std : : pair < std : : string , int > > MechanicsManager : : getStolenItemOwners ( const std : : string & itemid )
{
std : : vector < std : : pair < std : : string , int > > result ;
StolenItemsMap : : const_iterator it = mStolenItems . find ( Misc : : StringUtils : : lowerCase ( itemid ) ) ;
if ( it = = mStolenItems . end ( ) )
return result ;
else
{
const OwnerMap & owners = it - > second ;
for ( OwnerMap : : const_iterator ownerIt = owners . begin ( ) ; ownerIt ! = owners . end ( ) ; + + ownerIt )
result . push_back ( std : : make_pair ( ownerIt - > first . first , ownerIt - > second ) ) ;
return result ;
}
}
2018-06-03 06:46:12 +00:00
bool MechanicsManager : : isItemStolenFrom ( const std : : string & itemid , const MWWorld : : Ptr & ptr )
2015-02-04 20:18:43 +00:00
{
StolenItemsMap : : const_iterator it = mStolenItems . find ( Misc : : StringUtils : : lowerCase ( itemid ) ) ;
if ( it = = mStolenItems . end ( ) )
return false ;
2018-06-03 06:46:12 +00:00
2015-02-04 20:18:43 +00:00
const OwnerMap & owners = it - > second ;
2018-06-03 06:46:12 +00:00
const std : : string ownerid = ptr . getCellRef ( ) . getRefId ( ) ;
2015-02-04 20:18:43 +00:00
OwnerMap : : const_iterator ownerFound = owners . find ( std : : make_pair ( Misc : : StringUtils : : lowerCase ( ownerid ) , false ) ) ;
2018-06-03 07:50:10 +00:00
if ( ownerFound ! = owners . end ( ) )
return true ;
2018-06-03 06:46:12 +00:00
2018-06-03 07:50:10 +00:00
const std : : string factionid = ptr . getClass ( ) . getPrimaryFaction ( ptr ) ;
if ( ! factionid . empty ( ) )
{
OwnerMap : : const_iterator factionOwnerFound = owners . find ( std : : make_pair ( Misc : : StringUtils : : lowerCase ( factionid ) , true ) ) ;
return factionOwnerFound ! = owners . end ( ) ;
}
return false ;
2015-02-04 20:18:43 +00:00
}
2017-08-14 06:41:37 +00:00
void MechanicsManager : : confiscateStolenItemToOwner ( const MWWorld : : Ptr & player , const MWWorld : : Ptr & item , const MWWorld : : Ptr & victim , int count )
{
if ( player ! = getPlayer ( ) )
return ;
const std : : string itemId = Misc : : StringUtils : : lowerCase ( item . getCellRef ( ) . getRefId ( ) ) ;
StolenItemsMap : : iterator stolenIt = mStolenItems . find ( itemId ) ;
if ( stolenIt = = mStolenItems . end ( ) )
return ;
Owner owner ;
owner . first = victim . getCellRef ( ) . getRefId ( ) ;
owner . second = false ;
2018-06-03 10:02:27 +00:00
const std : : string victimFaction = victim . getClass ( ) . getPrimaryFaction ( victim ) ;
if ( ! victimFaction . empty ( ) & & Misc : : StringUtils : : ciEqual ( item . getCellRef ( ) . getFaction ( ) , victimFaction ) ) // Is the item faction-owned?
{
owner . first = victimFaction ;
owner . second = true ;
}
2017-08-14 06:41:37 +00:00
Misc : : StringUtils : : lowerCaseInPlace ( owner . first ) ;
// decrease count of stolen items
int toRemove = std : : min ( count , mStolenItems [ itemId ] [ owner ] ) ;
mStolenItems [ itemId ] [ owner ] - = toRemove ;
if ( mStolenItems [ itemId ] [ owner ] = = 0 )
{
// erase owner from stolen items owners
OwnerMap & owners = stolenIt - > second ;
OwnerMap : : iterator ownersIt = owners . find ( owner ) ;
if ( ownersIt ! = owners . end ( ) )
owners . erase ( ownersIt ) ;
}
MWWorld : : ContainerStore & store = player . getClass ( ) . getContainerStore ( player ) ;
// move items from player to owner and report about theft
victim . getClass ( ) . getContainerStore ( victim ) . add ( item , toRemove , victim ) ;
store . remove ( item , toRemove , player ) ;
2019-09-16 10:20:43 +00:00
commitCrime ( player , victim , OT_Theft , item . getCellRef ( ) . getFaction ( ) , item . getClass ( ) . getValue ( item ) * toRemove ) ;
2017-08-14 06:41:37 +00:00
}
2015-02-04 20:18:43 +00:00
void MechanicsManager : : confiscateStolenItems ( const MWWorld : : Ptr & player , const MWWorld : : Ptr & targetContainer )
{
MWWorld : : ContainerStore & store = player . getClass ( ) . getContainerStore ( player ) ;
for ( MWWorld : : ContainerStoreIterator it = store . begin ( ) ; it ! = store . end ( ) ; + + it )
{
2015-12-18 15:58:08 +00:00
StolenItemsMap : : iterator stolenIt = mStolenItems . find ( Misc : : StringUtils : : lowerCase ( it - > getCellRef ( ) . getRefId ( ) ) ) ;
2015-02-04 20:18:43 +00:00
if ( stolenIt = = mStolenItems . end ( ) )
continue ;
OwnerMap & owners = stolenIt - > second ;
int itemCount = it - > getRefData ( ) . getCount ( ) ;
for ( OwnerMap : : iterator ownerIt = owners . begin ( ) ; ownerIt ! = owners . end ( ) ; )
{
int toRemove = std : : min ( itemCount , ownerIt - > second ) ;
itemCount - = toRemove ;
ownerIt - > second - = toRemove ;
if ( ownerIt - > second = = 0 )
owners . erase ( ownerIt + + ) ;
else
+ + ownerIt ;
}
int toMove = it - > getRefData ( ) . getCount ( ) - itemCount ;
targetContainer . getClass ( ) . getContainerStore ( targetContainer ) . add ( * it , toMove , targetContainer ) ;
store . remove ( * it , toMove , player ) ;
}
// TODO: unhardcode the locklevel
2019-09-17 18:30:37 +00:00
targetContainer . getCellRef ( ) . lock ( 50 ) ;
2015-02-04 20:18:43 +00:00
}
2015-02-03 22:43:56 +00:00
void MechanicsManager : : itemTaken ( const MWWorld : : Ptr & ptr , const MWWorld : : Ptr & item , const MWWorld : : Ptr & container ,
2017-10-05 07:12:43 +00:00
int count , bool alarm )
2014-01-10 20:26:24 +00:00
{
2015-08-21 09:12:39 +00:00
if ( ptr ! = getPlayer ( ) )
2015-02-04 20:18:43 +00:00
return ;
2014-01-10 20:26:24 +00:00
MWWorld : : Ptr victim ;
2015-02-03 22:43:56 +00:00
2017-09-27 16:00:41 +00:00
bool isAllowed = true ;
2015-02-04 20:18:43 +00:00
const MWWorld : : CellRef * ownerCellRef = & item . getCellRef ( ) ;
2015-02-03 22:43:56 +00:00
if ( ! container . isEmpty ( ) )
{
// Inherit the owner of the container
2015-02-04 20:18:43 +00:00
ownerCellRef = & container . getCellRef ( ) ;
2017-09-27 16:00:41 +00:00
isAllowed = isAllowedToUse ( ptr , container , victim ) ;
2015-02-03 22:43:56 +00:00
}
else
{
2017-09-27 16:00:41 +00:00
isAllowed = isAllowedToUse ( ptr , item , victim ) ;
2015-02-03 22:43:56 +00:00
if ( ! item . getCellRef ( ) . hasContentFile ( ) )
{
// this is a manually placed item, which means it was already stolen
return ;
}
}
2017-09-27 16:00:41 +00:00
if ( isAllowed )
2015-02-04 20:18:43 +00:00
return ;
Owner owner ;
owner . second = false ;
2017-10-05 07:12:43 +00:00
if ( ! container . isEmpty ( ) & & container . getClass ( ) . isActor ( ) )
2015-02-04 20:18:43 +00:00
{
2017-10-05 07:12:43 +00:00
// "container" is an actor inventory, so just take actor's ID
owner . first = ownerCellRef - > getRefId ( ) ;
2015-02-04 20:18:43 +00:00
}
2017-10-05 07:12:43 +00:00
else
{
owner . first = ownerCellRef - > getOwner ( ) ;
if ( owner . first . empty ( ) )
{
owner . first = ownerCellRef - > getFaction ( ) ;
owner . second = true ;
}
}
2015-12-07 21:49:15 +00:00
Misc : : StringUtils : : lowerCaseInPlace ( owner . first ) ;
2015-03-22 16:25:23 +00:00
if ( ! Misc : : StringUtils : : ciEqual ( item . getCellRef ( ) . getRefId ( ) , MWWorld : : ContainerStore : : sGoldId ) )
2018-02-26 20:21:51 +00:00
{
2019-02-04 21:49:19 +00:00
if ( victim . isEmpty ( ) | | ( victim . getClass ( ) . isActor ( ) & & ! victim . getClass ( ) . getCreatureStats ( victim ) . isDead ( ) ) )
2018-02-26 20:21:51 +00:00
mStolenItems [ Misc : : StringUtils : : lowerCase ( item . getCellRef ( ) . getRefId ( ) ) ] [ owner ] + = count ;
}
2017-10-05 07:12:43 +00:00
if ( alarm )
2019-09-16 10:20:43 +00:00
commitCrime ( ptr , victim , OT_Theft , ownerCellRef - > getFaction ( ) , item . getClass ( ) . getValue ( item ) * count ) ;
2014-01-07 18:49:16 +00:00
}
2015-07-19 14:37:19 +00:00
2019-09-16 10:20:43 +00:00
bool MechanicsManager : : commitCrime ( const MWWorld : : Ptr & player , const MWWorld : : Ptr & victim , OffenseType type , const std : : string & factionId , int arg , bool victimAware )
2014-01-07 18:49:16 +00:00
{
2014-06-16 02:03:53 +00:00
// NOTE: victim may be empty
2014-04-19 23:03:31 +00:00
// Only player can commit crime
2015-08-21 09:12:39 +00:00
if ( player ! = getPlayer ( ) )
2014-01-11 02:08:16 +00:00
return false ;
2014-01-08 00:24:06 +00:00
2014-05-06 16:23:17 +00:00
// Find all the actors within the alarm radius
2014-04-01 18:15:55 +00:00
std : : vector < MWWorld : : Ptr > neighbors ;
2014-06-14 20:14:18 +00:00
2015-06-01 19:41:13 +00:00
osg : : Vec3f from ( player . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
2014-12-19 20:45:26 +00:00
const MWWorld : : ESMStore & esmStore = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
2018-08-29 15:38:12 +00:00
float radius = esmStore . get < ESM : : GameSetting > ( ) . find ( " fAlarmRadius " ) - > mValue . getFloat ( ) ;
2014-06-14 20:14:18 +00:00
2015-04-25 13:19:17 +00:00
mActors . getObjectsInRange ( from , radius , neighbors ) ;
2014-06-14 20:14:18 +00:00
// victim should be considered even beyond alarm radius
2015-06-01 19:41:13 +00:00
if ( ! victim . isEmpty ( ) & & ( from - victim . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) . length2 ( ) > radius * radius )
2014-06-14 20:14:18 +00:00
neighbors . push_back ( victim ) ;
2014-04-05 14:26:14 +00:00
2015-07-19 14:37:19 +00:00
// get the player's followers / allies (works recursively) that will not report crimes
std : : set < MWWorld : : Ptr > playerFollowers ;
2016-12-20 11:38:51 +00:00
getActorsSidingWith ( player , playerFollowers ) ;
2015-07-19 14:37:19 +00:00
2014-12-19 20:45:26 +00:00
// Did anyone see it?
2014-12-01 15:36:01 +00:00
bool crimeSeen = false ;
2018-11-05 22:35:20 +00:00
for ( const MWWorld : : Ptr & neighbor : neighbors )
2014-04-25 02:47:45 +00:00
{
2018-11-05 22:35:20 +00:00
if ( ! canReportCrime ( neighbor , victim , playerFollowers ) )
2014-06-17 01:54:41 +00:00
continue ;
2014-04-01 18:15:55 +00:00
2018-11-05 22:35:20 +00:00
if ( ( neighbor = = victim & & victimAware )
2014-06-17 01:54:41 +00:00
// Murder crime can be reported even if no one saw it (hearing is enough, I guess).
// TODO: Add mod support for stealth executions!
2018-11-05 22:35:20 +00:00
| | ( type = = OT_Murder & & neighbor ! = victim )
| | ( MWBase : : Environment : : get ( ) . getWorld ( ) - > getLOS ( player , neighbor ) & & awarenessCheck ( player , neighbor ) ) )
2014-01-07 18:49:16 +00:00
{
2017-04-18 09:38:26 +00:00
/*
Start of tes3mp addition
We need player - controlled NPCs to not report crimes committed by other players
*/
2019-08-21 06:04:36 +00:00
if ( mwmp : : PlayerList : : isDedicatedPlayer ( neighbor ) )
2016-10-31 09:00:56 +00:00
continue ;
2017-04-18 09:38:26 +00:00
/*
End of tes3mp addition
*/
2016-10-31 09:00:56 +00:00
2017-10-15 07:03:02 +00:00
// NPC will complain about theft even if he will do nothing about it
if ( type = = OT_Theft | | type = = OT_Pickpocket )
2018-11-05 22:35:20 +00:00
MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > say ( neighbor , " thief " ) ;
2017-10-15 07:03:02 +00:00
2014-12-01 15:36:01 +00:00
crimeSeen = true ;
}
2014-01-07 18:49:16 +00:00
}
2014-06-16 23:10:37 +00:00
2014-12-19 20:45:26 +00:00
if ( crimeSeen )
2019-09-16 10:20:43 +00:00
reportCrime ( player , victim , type , factionId , arg ) ;
2014-12-19 20:45:26 +00:00
else if ( type = = OT_Assault & & ! victim . isEmpty ( ) )
2017-04-29 17:40:52 +00:00
{
2018-06-01 08:41:31 +00:00
bool reported = false ;
2017-04-29 17:40:52 +00:00
if ( victim . getClass ( ) . isClass ( victim , " guard " )
& & ! victim . getClass ( ) . getCreatureStats ( victim ) . getAiSequence ( ) . hasPackage ( AiPackage : : TypeIdPursue ) )
2019-09-16 10:20:43 +00:00
reported = reportCrime ( player , victim , type , std : : string ( ) , arg ) ;
2018-06-01 08:41:31 +00:00
if ( ! reported )
2017-04-29 17:40:52 +00:00
startCombat ( victim , player ) ; // TODO: combat should be started with an "unaware" flag, which makes the victim flee?
}
2014-12-19 20:45:26 +00:00
return crimeSeen ;
2014-01-07 18:49:16 +00:00
}
2018-06-01 08:41:31 +00:00
bool MechanicsManager : : canReportCrime ( const MWWorld : : Ptr & actor , const MWWorld : : Ptr & victim , std : : set < MWWorld : : Ptr > & playerFollowers )
{
if ( actor = = getPlayer ( )
| | ! actor . getClass ( ) . isNpc ( ) | | actor . getClass ( ) . getCreatureStats ( actor ) . isDead ( ) )
return false ;
if ( actor . getClass ( ) . getCreatureStats ( actor ) . getAiSequence ( ) . isInCombat ( victim ) )
return false ;
// Unconsious actor can not report about crime and should not become hostile
if ( actor . getClass ( ) . getCreatureStats ( actor ) . getKnockedDown ( ) )
return false ;
// Player's followers should not attack player, or try to arrest him
if ( actor . getClass ( ) . getCreatureStats ( actor ) . getAiSequence ( ) . hasPackage ( AiPackage : : TypeIdFollow ) )
{
if ( playerFollowers . find ( actor ) ! = playerFollowers . end ( ) )
return false ;
}
return true ;
}
2019-09-16 10:20:43 +00:00
bool MechanicsManager : : reportCrime ( const MWWorld : : Ptr & player , const MWWorld : : Ptr & victim , OffenseType type , const std : : string & factionId , int arg )
2014-01-07 18:49:16 +00:00
{
2014-01-09 00:55:49 +00:00
const MWWorld : : Store < ESM : : GameSetting > & store = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
2014-04-25 02:47:45 +00:00
2014-06-17 01:54:41 +00:00
if ( type = = OT_Murder & & ! victim . isEmpty ( ) )
victim . getClass ( ) . getCreatureStats ( victim ) . notifyMurder ( ) ;
2014-12-21 01:45:37 +00:00
// Bounty and disposition penalty for each type of crime
float disp = 0.f , dispVictim = 0.f ;
2014-01-07 18:49:16 +00:00
if ( type = = OT_Trespassing | | type = = OT_SleepingInOwnedBed )
2014-12-18 21:27:09 +00:00
{
2018-08-29 15:38:12 +00:00
arg = store . find ( " iCrimeTresspass " ) - > mValue . getInteger ( ) ;
disp = dispVictim = store . find ( " iDispTresspass " ) - > mValue . getFloat ( ) ;
2014-12-18 21:27:09 +00:00
}
2014-01-07 18:49:16 +00:00
else if ( type = = OT_Pickpocket )
2014-12-18 21:27:09 +00:00
{
2018-08-29 15:38:12 +00:00
arg = store . find ( " iCrimePickPocket " ) - > mValue . getInteger ( ) ;
disp = dispVictim = store . find ( " fDispPickPocketMod " ) - > mValue . getFloat ( ) ;
2014-12-18 21:27:09 +00:00
}
2014-01-07 18:49:16 +00:00
else if ( type = = OT_Assault )
2014-12-18 21:27:09 +00:00
{
2018-08-29 15:38:12 +00:00
arg = store . find ( " iCrimeAttack " ) - > mValue . getInteger ( ) ;
disp = store . find ( " iDispAttackMod " ) - > mValue . getFloat ( ) ;
dispVictim = store . find ( " fDispAttacking " ) - > mValue . getFloat ( ) ;
2014-12-18 21:27:09 +00:00
}
2014-01-07 18:49:16 +00:00
else if ( type = = OT_Murder )
2014-12-18 21:27:09 +00:00
{
2018-08-29 15:38:12 +00:00
arg = store . find ( " iCrimeKilling " ) - > mValue . getInteger ( ) ;
disp = dispVictim = store . find ( " iDispKilling " ) - > mValue . getFloat ( ) ;
2014-12-18 21:27:09 +00:00
}
2014-01-09 00:55:49 +00:00
else if ( type = = OT_Theft )
2014-06-09 01:42:29 +00:00
{
2018-08-29 15:38:12 +00:00
disp = dispVictim = store . find ( " fDispStealing " ) - > mValue . getFloat ( ) * arg ;
arg = static_cast < int > ( arg * store . find ( " fCrimeStealing " ) - > mValue . getFloat ( ) ) ;
2014-06-09 01:42:29 +00:00
arg = std : : max ( 1 , arg ) ; // Minimum bounty of 1, in case items with zero value are stolen
}
2014-01-07 18:49:16 +00:00
2014-06-16 23:10:37 +00:00
// Make surrounding actors within alarm distance respond to the crime
std : : vector < MWWorld : : Ptr > neighbors ;
const MWWorld : : ESMStore & esmStore = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) ;
2015-06-01 19:41:13 +00:00
osg : : Vec3f from ( player . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
2018-08-29 15:38:12 +00:00
float radius = esmStore . get < ESM : : GameSetting > ( ) . find ( " fAlarmRadius " ) - > mValue . getFloat ( ) ;
2014-06-16 23:10:37 +00:00
2015-04-25 13:19:17 +00:00
mActors . getObjectsInRange ( from , radius , neighbors ) ;
2014-06-16 23:10:37 +00:00
// victim should be considered even beyond alarm radius
2015-06-01 19:41:13 +00:00
if ( ! victim . isEmpty ( ) & & ( from - victim . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) . length2 ( ) > radius * radius )
2014-06-16 23:10:37 +00:00
neighbors . push_back ( victim ) ;
int id = MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) . getNewCrimeId ( ) ;
// What amount of provocation did this crime generate?
// Controls whether witnesses will engage combat with the criminal.
2014-12-18 21:27:09 +00:00
int fight = 0 , fightVictim = 0 ;
2014-06-16 23:10:37 +00:00
if ( type = = OT_Trespassing | | type = = OT_SleepingInOwnedBed )
2018-08-29 15:38:12 +00:00
fight = fightVictim = esmStore . get < ESM : : GameSetting > ( ) . find ( " iFightTrespass " ) - > mValue . getInteger ( ) ;
2014-06-16 23:10:37 +00:00
else if ( type = = OT_Pickpocket )
2014-12-18 21:27:09 +00:00
{
2018-08-29 15:38:12 +00:00
fight = esmStore . get < ESM : : GameSetting > ( ) . find ( " iFightPickpocket " ) - > mValue . getInteger ( ) ;
fightVictim = esmStore . get < ESM : : GameSetting > ( ) . find ( " iFightPickpocket " ) - > mValue . getInteger ( ) * 4 ; // *4 according to research wiki
2014-12-18 21:27:09 +00:00
}
else if ( type = = OT_Assault )
{
2018-08-29 15:38:12 +00:00
fight = esmStore . get < ESM : : GameSetting > ( ) . find ( " iFightAttacking " ) - > mValue . getInteger ( ) ;
fightVictim = esmStore . get < ESM : : GameSetting > ( ) . find ( " iFightAttack " ) - > mValue . getInteger ( ) ;
2014-12-18 21:27:09 +00:00
}
2014-06-16 23:10:37 +00:00
else if ( type = = OT_Murder )
2018-08-29 15:38:12 +00:00
fight = fightVictim = esmStore . get < ESM : : GameSetting > ( ) . find ( " iFightKilling " ) - > mValue . getInteger ( ) ;
2014-06-16 23:10:37 +00:00
else if ( type = = OT_Theft )
2018-08-29 15:38:12 +00:00
fight = fightVictim = esmStore . get < ESM : : GameSetting > ( ) . find ( " fFightStealing " ) - > mValue . getInteger ( ) ;
2014-06-16 23:10:37 +00:00
2014-12-19 20:45:26 +00:00
bool reported = false ;
2018-06-01 08:41:31 +00:00
std : : set < MWWorld : : Ptr > playerFollowers ;
getActorsSidingWith ( player , playerFollowers ) ;
2014-06-16 23:10:37 +00:00
// Tell everyone (including the original reporter) in alarm range
2018-11-05 22:35:20 +00:00
for ( const MWWorld : : Ptr & actor : neighbors )
2014-06-16 23:10:37 +00:00
{
2018-11-05 22:35:20 +00:00
if ( ! canReportCrime ( actor , victim , playerFollowers ) )
2014-08-02 22:30:08 +00:00
continue ;
2014-12-19 20:45:26 +00:00
// Will the witness report the crime?
2018-11-05 22:35:20 +00:00
if ( actor . getClass ( ) . getCreatureStats ( actor ) . getAiSetting ( CreatureStats : : AI_Alarm ) . getBase ( ) > = 100 )
2014-12-19 20:45:26 +00:00
{
reported = true ;
2017-08-14 14:42:19 +00:00
2017-10-15 07:03:02 +00:00
if ( type = = OT_Trespassing )
2018-11-05 22:35:20 +00:00
MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > say ( actor , " intruder " ) ;
2014-12-19 20:45:26 +00:00
}
2018-06-01 08:41:31 +00:00
}
2014-12-19 20:45:26 +00:00
2018-11-05 22:35:20 +00:00
for ( const MWWorld : : Ptr & actor : neighbors )
2018-06-01 08:41:31 +00:00
{
2018-11-05 22:35:20 +00:00
if ( ! canReportCrime ( actor , victim , playerFollowers ) )
2018-06-01 08:41:31 +00:00
continue ;
2014-12-19 20:45:26 +00:00
2018-11-05 22:35:20 +00:00
if ( reported & & actor . getClass ( ) . isClass ( actor , " guard " ) )
2014-06-16 23:10:37 +00:00
{
// Mark as Alarmed for dialogue
2018-11-05 22:35:20 +00:00
actor . getClass ( ) . getCreatureStats ( actor ) . setAlarmed ( true ) ;
2014-06-16 23:10:37 +00:00
// Set the crime ID, which we will use to calm down participants
// once the bounty has been paid.
2018-11-05 22:35:20 +00:00
actor . getClass ( ) . getNpcStats ( actor ) . setCrimeId ( id ) ;
2014-06-16 23:10:37 +00:00
2018-11-05 22:35:20 +00:00
if ( ! actor . getClass ( ) . getCreatureStats ( actor ) . getAiSequence ( ) . hasPackage ( AiPackage : : TypeIdPursue ) )
2017-08-05 08:38:26 +00:00
{
2018-11-05 22:35:20 +00:00
actor . getClass ( ) . getCreatureStats ( actor ) . getAiSequence ( ) . stack ( AiPursue ( player ) , actor ) ;
2017-08-05 08:38:26 +00:00
}
2014-06-16 23:10:37 +00:00
}
else
2014-01-08 18:26:03 +00:00
{
2018-11-05 22:35:20 +00:00
float dispTerm = ( actor = = victim ) ? dispVictim : disp ;
2014-12-21 01:45:37 +00:00
2018-11-05 22:35:20 +00:00
float alarmTerm = 0.01f * actor . getClass ( ) . getCreatureStats ( actor ) . getAiSetting ( CreatureStats : : AI_Alarm ) . getBase ( ) ;
2014-12-23 15:38:30 +00:00
if ( type = = OT_Pickpocket & & alarmTerm < = 0 )
alarmTerm = 1.0 ;
2018-11-05 22:35:20 +00:00
if ( actor ! = victim )
2014-12-21 01:45:37 +00:00
dispTerm * = alarmTerm ;
2018-11-05 22:35:20 +00:00
float fightTerm = static_cast < float > ( ( actor = = victim ) ? fightVictim : fight ) ;
2014-12-21 01:45:37 +00:00
fightTerm + = getFightDispositionBias ( dispTerm ) ;
2018-11-05 22:35:20 +00:00
fightTerm + = getFightDistanceBias ( actor , player ) ;
2014-12-23 15:38:30 +00:00
fightTerm * = alarmTerm ;
2014-12-18 21:27:09 +00:00
2018-11-05 22:35:20 +00:00
int observerFightRating = actor . getClass ( ) . getCreatureStats ( actor ) . getAiSetting ( CreatureStats : : AI_Fight ) . getBase ( ) ;
2014-12-21 01:45:37 +00:00
if ( observerFightRating + fightTerm > 100 )
2015-03-08 00:07:29 +00:00
fightTerm = static_cast < float > ( 100 - observerFightRating ) ;
2014-12-21 01:45:37 +00:00
fightTerm = std : : max ( 0.f , fightTerm ) ;
if ( observerFightRating + fightTerm > = 100 )
2014-06-16 23:10:37 +00:00
{
2018-11-05 22:35:20 +00:00
startCombat ( actor , player ) ;
2014-06-16 23:10:37 +00:00
2018-11-05 22:35:20 +00:00
NpcStats & observerStats = actor . getClass ( ) . getNpcStats ( actor ) ;
2014-12-14 18:35:34 +00:00
// Apply aggression value to the base Fight rating, so that the actor can continue fighting
// after a Calm spell wears off
2015-03-08 00:07:29 +00:00
observerStats . setAiSetting ( CreatureStats : : AI_Fight , observerFightRating + static_cast < int > ( fightTerm ) ) ;
2014-12-21 01:45:37 +00:00
2015-03-08 00:07:29 +00:00
observerStats . setBaseDisposition ( observerStats . getBaseDisposition ( ) + static_cast < int > ( dispTerm ) ) ;
2014-12-14 18:35:34 +00:00
2014-06-16 23:10:37 +00:00
// Set the crime ID, which we will use to calm down participants
// once the bounty has been paid.
2014-12-21 01:45:37 +00:00
observerStats . setCrimeId ( id ) ;
2014-06-16 23:10:37 +00:00
// Mark as Alarmed for dialogue
2014-12-21 01:45:37 +00:00
observerStats . setAlarmed ( true ) ;
2014-06-16 23:10:37 +00:00
}
2014-01-08 18:26:03 +00:00
}
}
2014-12-19 20:45:26 +00:00
if ( reported )
{
player . getClass ( ) . getNpcStats ( player ) . setBounty ( player . getClass ( ) . getNpcStats ( player ) . getBounty ( )
+ arg ) ;
// If committing a crime against a faction member, expell from the faction
if ( ! victim . isEmpty ( ) & & victim . getClass ( ) . isNpc ( ) )
{
2015-01-27 16:32:21 +00:00
std : : string factionID = victim . getClass ( ) . getPrimaryFaction ( victim ) ;
const std : : map < std : : string , int > & playerRanks = player . getClass ( ) . getNpcStats ( player ) . getFactionRanks ( ) ;
if ( playerRanks . find ( Misc : : StringUtils : : lowerCase ( factionID ) ) ! = playerRanks . end ( ) )
2014-12-19 20:45:26 +00:00
{
2017-05-24 09:19:11 +00:00
/*
Start of tes3mp addition
Send an ID_PLAYER_FACTION packet every time a player is expelled from a faction
*/
2017-07-05 04:05:52 +00:00
mwmp : : Main : : get ( ) . getLocalPlayer ( ) - > sendFactionExpulsionState ( Misc : : StringUtils : : lowerCase ( factionID ) , true ) ;
2017-05-24 09:19:11 +00:00
/*
End of tes3mp addition
*/
2014-12-19 20:45:26 +00:00
player . getClass ( ) . getNpcStats ( player ) . expell ( factionID ) ;
}
}
2019-09-16 10:20:43 +00:00
else if ( ! factionId . empty ( ) )
{
const std : : map < std : : string , int > & playerRanks = player . getClass ( ) . getNpcStats ( player ) . getFactionRanks ( ) ;
if ( playerRanks . find ( Misc : : StringUtils : : lowerCase ( factionId ) ) ! = playerRanks . end ( ) )
{
player . getClass ( ) . getNpcStats ( player ) . expell ( factionId ) ;
}
}
2015-01-05 18:59:47 +00:00
if ( type = = OT_Assault & & ! victim . isEmpty ( )
& & ! victim . getClass ( ) . getCreatureStats ( victim ) . getAiSequence ( ) . isInCombat ( player )
& & victim . getClass ( ) . isNpc ( ) )
{
// Attacker is in combat with us, but we are not in combat with the attacker yet. Time to fight back.
// Note: accidental or collateral damage attacks are ignored.
2017-04-29 17:40:52 +00:00
if ( ! victim . getClass ( ) . getCreatureStats ( victim ) . getAiSequence ( ) . hasPackage ( AiPackage : : TypeIdPursue ) )
startCombat ( victim , player ) ;
2015-01-05 18:59:47 +00:00
// Set the crime ID, which we will use to calm down participants
// once the bounty has been paid.
victim . getClass ( ) . getNpcStats ( victim ) . setCrimeId ( id ) ;
}
2014-12-19 20:45:26 +00:00
}
2018-06-01 08:41:31 +00:00
return reported ;
2014-01-07 18:49:16 +00:00
}
2017-02-01 17:15:10 +00:00
bool MechanicsManager : : actorAttacked ( const MWWorld : : Ptr & target , const MWWorld : : Ptr & attacker )
2014-08-03 13:20:33 +00:00
{
2019-04-16 11:09:53 +00:00
const MWWorld : : Ptr & player = getPlayer ( ) ;
if ( target = = player | | ! attacker . getClass ( ) . isActor ( ) )
2017-02-11 10:59:42 +00:00
return false ;
2019-12-03 20:40:02 +00:00
/*
Start of tes3mp change ( major )
Don ' t set DedicatedPlayers as being in combat with the attacker , to prevent
AI actors from deciding to reciprocate by also starting combat
*/
if ( mwmp : : PlayerList : : isDedicatedPlayer ( target ) )
return false ;
/*
End of tes3mp change ( major )
*/
2017-02-01 17:15:10 +00:00
MWMechanics : : CreatureStats & statsTarget = target . getClass ( ) . getCreatureStats ( target ) ;
2019-12-02 17:08:03 +00:00
/*
Start of tes3mp change ( major )
Allow collateral damage from dedicated players as well
*/
if ( attacker = = player | | mwmp : : PlayerList : : isDedicatedPlayer ( attacker ) )
/*
End of tes3mp change ( major )
*/
2014-08-03 13:20:33 +00:00
{
2019-04-16 11:09:53 +00:00
std : : set < MWWorld : : Ptr > followersAttacker ;
getActorsSidingWith ( attacker , followersAttacker ) ;
2019-12-02 17:08:03 +00:00
/*
Start of tes3mp change ( major )
Check not only whether the target is on the same side as the attacker ,
but also whether the attacker is on the same side as the target ,
thus allowing for NPC companions of one player to forgive another player
when those players are allied
*/
std : : set < MWWorld : : Ptr > followersTarget ;
getActorsSidingWith ( target , followersTarget ) ;
if ( followersAttacker . find ( target ) ! = followersAttacker . end ( ) | | followersTarget . find ( attacker ) ! = followersTarget . end ( ) )
/*
End of tes3mp change ( major )
*/
2014-09-18 03:13:17 +00:00
{
2019-04-16 11:09:53 +00:00
statsTarget . friendlyHit ( ) ;
2019-12-02 17:08:03 +00:00
/*
Start of tes3mp change ( major )
Due to a greater propensity for collateral damage in multiplayer ,
allow more friendly hits
TODO : Allow the server to change the count of the friendly hits
*/
if ( statsTarget . getFriendlyHits ( ) < 8 )
/*
End of tes3mp change ( major )
*/
2019-04-16 11:09:53 +00:00
{
MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > say ( target , " hit " ) ;
return false ;
}
2014-09-18 03:13:17 +00:00
}
2014-08-03 13:20:33 +00:00
}
2018-08-15 23:26:02 +00:00
if ( canCommitCrimeAgainst ( target , attacker ) )
2017-02-01 17:15:10 +00:00
commitCrime ( attacker , target , MWBase : : MechanicsManager : : OT_Assault ) ;
2014-08-03 13:20:33 +00:00
2018-08-15 23:26:02 +00:00
AiSequence & seq = statsTarget . getAiSequence ( ) ;
2017-04-20 14:00:40 +00:00
/*
Start of tes3mp change ( major )
Make it possible to start combat with DedicatedPlayers and DedicatedActors by
adding additional conditions for them
*/
2019-04-16 11:09:53 +00:00
if ( ! attacker . isEmpty ( )
2019-08-22 05:40:32 +00:00
& & ( attacker . getClass ( ) . getCreatureStats ( attacker ) . getAiSequence ( ) . isInCombat ( target ) | | attacker = = player
| | mwmp : : PlayerList : : isDedicatedPlayer ( attacker ) | | mwmp : : Main : : get ( ) . getCellController ( ) - > isDedicatedActor ( attacker ) )
2019-04-16 11:09:53 +00:00
& & ! seq . isInCombat ( attacker ) )
2017-04-20 14:00:40 +00:00
/*
End of tes3mp change ( major )
*/
2014-08-03 13:20:33 +00:00
{
// Attacker is in combat with us, but we are not in combat with the attacker yet. Time to fight back.
// Note: accidental or collateral damage attacks are ignored.
2017-04-29 17:40:52 +00:00
if ( ! target . getClass ( ) . getCreatureStats ( target ) . getAiSequence ( ) . hasPackage ( AiPackage : : TypeIdPursue ) )
2018-04-02 18:27:04 +00:00
{
// If an actor has OnPCHitMe declared in his script, his Fight = 0 and the attacker is player,
// he will attack the player only if we will force him (e.g. via StartCombat console command)
bool peaceful = false ;
std : : string script = target . getClass ( ) . getScript ( target ) ;
2019-04-16 11:09:53 +00:00
if ( ! script . empty ( ) & & target . getRefData ( ) . getLocals ( ) . hasVar ( script , " onpchitme " ) & & attacker = = player )
2018-04-02 18:27:04 +00:00
{
int fight = std : : max ( 0 , target . getClass ( ) . getCreatureStats ( target ) . getAiSetting ( CreatureStats : : AI_Fight ) . getModified ( ) ) ;
peaceful = ( fight = = 0 ) ;
}
if ( ! peaceful )
startCombat ( target , attacker ) ;
}
2014-08-03 13:20:33 +00:00
}
2014-09-26 20:08:07 +00:00
return true ;
2014-08-03 13:20:33 +00:00
}
2018-08-15 23:26:02 +00:00
bool MechanicsManager : : canCommitCrimeAgainst ( const MWWorld : : Ptr & target , const MWWorld : : Ptr & attacker )
{
2018-11-05 22:35:20 +00:00
const MWMechanics : : AiSequence & seq = target . getClass ( ) . getCreatureStats ( target ) . getAiSequence ( ) ;
2018-08-15 23:26:02 +00:00
return target . getClass ( ) . isNpc ( ) & & ! attacker . isEmpty ( ) & & ! seq . isInCombat ( attacker )
& & ! isAggressive ( target , attacker ) & & ! seq . isEngagedWithActor ( )
& & ! target . getClass ( ) . getCreatureStats ( target ) . getAiSequence ( ) . hasPackage ( AiPackage : : TypeIdPursue ) ;
}
2015-07-18 18:39:45 +00:00
void MechanicsManager : : actorKilled ( const MWWorld : : Ptr & victim , const MWWorld : : Ptr & attacker )
{
2018-06-15 10:03:43 +00:00
if ( attacker . isEmpty ( ) | | victim . isEmpty ( ) )
2015-07-18 18:39:45 +00:00
return ;
if ( victim = = attacker )
return ; // known to happen
if ( ! victim . getClass ( ) . isNpc ( ) )
return ; // TODO: implement animal rights
const MWMechanics : : NpcStats & victimStats = victim . getClass ( ) . getNpcStats ( victim ) ;
2018-08-16 17:58:51 +00:00
const MWWorld : : Ptr & player = getPlayer ( ) ;
2018-08-17 21:35:04 +00:00
bool canCommit = attacker = = player & & canCommitCrimeAgainst ( victim , attacker ) ;
2018-06-15 10:03:43 +00:00
// For now we report only about crimes of player and player's followers
if ( attacker ! = player )
{
std : : set < MWWorld : : Ptr > playerFollowers ;
getActorsSidingWith ( player , playerFollowers ) ;
if ( playerFollowers . find ( attacker ) = = playerFollowers . end ( ) )
return ;
}
2015-07-18 18:39:45 +00:00
2018-08-16 17:58:51 +00:00
if ( ! canCommit & & victimStats . getCrimeId ( ) = = - 1 )
return ;
2015-07-18 18:39:45 +00:00
// Simple check for who attacked first: if the player attacked first, a crimeId should be set
// Doesn't handle possible edge case where no one reported the assault, but in such a case,
// for bystanders it is not possible to tell who attacked first, anyway.
2018-06-15 10:03:43 +00:00
commitCrime ( player , victim , MWBase : : MechanicsManager : : OT_Murder ) ;
2015-07-18 18:39:45 +00:00
}
2014-01-06 23:51:09 +00:00
bool MechanicsManager : : awarenessCheck ( const MWWorld : : Ptr & ptr , const MWWorld : : Ptr & observer )
{
2014-12-27 14:02:05 +00:00
if ( observer . getClass ( ) . getCreatureStats ( observer ) . isDead ( ) | | ! observer . getRefData ( ) . isEnabled ( ) )
2014-01-11 01:06:54 +00:00
return false ;
2014-01-06 23:51:09 +00:00
const MWWorld : : Store < ESM : : GameSetting > & store = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
CreatureStats & stats = ptr . getClass ( ) . getCreatureStats ( ptr ) ;
2014-08-16 20:38:22 +00:00
float invisibility = stats . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Invisibility ) . getMagnitude ( ) ;
2014-01-06 23:51:09 +00:00
if ( invisibility > 0 )
return false ;
float sneakTerm = 0 ;
2019-07-30 17:58:19 +00:00
if ( isSneaking ( ptr ) )
2014-01-06 23:51:09 +00:00
{
2018-08-29 15:38:12 +00:00
static float fSneakSkillMult = store . find ( " fSneakSkillMult " ) - > mValue . getFloat ( ) ;
static float fSneakBootMult = store . find ( " fSneakBootMult " ) - > mValue . getFloat ( ) ;
2015-03-08 00:07:29 +00:00
float sneak = static_cast < float > ( ptr . getClass ( ) . getSkill ( ptr , ESM : : Skill : : Sneak ) ) ;
2014-01-06 23:51:09 +00:00
int agility = stats . getAttribute ( ESM : : Attribute : : Agility ) . getModified ( ) ;
int luck = stats . getAttribute ( ESM : : Attribute : : Luck ) . getModified ( ) ;
float bootWeight = 0 ;
2019-07-30 17:58:19 +00:00
if ( ptr . getClass ( ) . isNpc ( ) & & MWBase : : Environment : : get ( ) . getWorld ( ) - > isOnGround ( ptr ) )
2014-01-06 23:51:09 +00:00
{
2017-02-28 14:31:51 +00:00
const MWWorld : : InventoryStore & inv = ptr . getClass ( ) . getInventoryStore ( ptr ) ;
MWWorld : : ConstContainerStoreIterator it = inv . getSlot ( MWWorld : : InventoryStore : : Slot_Boots ) ;
2014-01-06 23:51:09 +00:00
if ( it ! = inv . end ( ) )
bootWeight = it - > getClass ( ) . getWeight ( * it ) ;
}
2015-03-08 00:07:29 +00:00
sneakTerm = fSneakSkillMult * sneak + 0.2f * agility + 0.1f * luck + bootWeight * fSneakBootMult ;
2014-01-06 23:51:09 +00:00
}
2018-08-29 15:38:12 +00:00
static float fSneakDistBase = store . find ( " fSneakDistanceBase " ) - > mValue . getFloat ( ) ;
static float fSneakDistMult = store . find ( " fSneakDistanceMultiplier " ) - > mValue . getFloat ( ) ;
2014-01-06 23:51:09 +00:00
2015-05-29 23:00:24 +00:00
osg : : Vec3f pos1 ( ptr . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
osg : : Vec3f pos2 ( observer . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
float distTerm = fSneakDistBase + fSneakDistMult * ( pos1 - pos2 ) . length ( ) ;
2014-01-06 23:51:09 +00:00
2014-08-16 20:38:22 +00:00
float chameleon = stats . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Chameleon ) . getMagnitude ( ) ;
2014-01-06 23:51:09 +00:00
float x = sneakTerm * distTerm * stats . getFatigueTerm ( ) + chameleon + invisibility ;
CreatureStats & observerStats = observer . getClass ( ) . getCreatureStats ( observer ) ;
int obsAgility = observerStats . getAttribute ( ESM : : Attribute : : Agility ) . getModified ( ) ;
int obsLuck = observerStats . getAttribute ( ESM : : Attribute : : Luck ) . getModified ( ) ;
2014-08-16 20:38:22 +00:00
float obsBlind = observerStats . getMagicEffects ( ) . get ( ESM : : MagicEffect : : Blind ) . getMagnitude ( ) ;
2014-01-15 14:50:45 +00:00
int obsSneak = observer . getClass ( ) . getSkill ( observer , ESM : : Skill : : Sneak ) ;
2014-01-06 23:51:09 +00:00
2015-03-08 00:07:29 +00:00
float obsTerm = obsSneak + 0.2f * obsAgility + 0.1f * obsLuck - obsBlind ;
2014-01-06 23:51:09 +00:00
// is ptr behind the observer?
2018-08-29 15:38:12 +00:00
static float fSneakNoViewMult = store . find ( " fSneakNoViewMult " ) - > mValue . getFloat ( ) ;
static float fSneakViewMult = store . find ( " fSneakViewMult " ) - > mValue . getFloat ( ) ;
2014-01-06 23:51:09 +00:00
float y = 0 ;
2015-05-29 23:00:24 +00:00
osg : : Vec3f vec = pos1 - pos2 ;
if ( observer . getRefData ( ) . getBaseNode ( ) )
{
2015-06-05 01:41:10 +00:00
osg : : Vec3f observerDir = ( observer . getRefData ( ) . getBaseNode ( ) - > getAttitude ( ) * osg : : Vec3f ( 0 , 1 , 0 ) ) ;
float angleRadians = std : : acos ( observerDir * vec / ( observerDir . length ( ) * vec . length ( ) ) ) ;
2016-01-02 18:24:39 +00:00
if ( angleRadians > osg : : DegreesToRadians ( 90.f ) )
2015-05-29 23:00:24 +00:00
y = obsTerm * observerStats . getFatigueTerm ( ) * fSneakNoViewMult ;
else
y = obsTerm * observerStats . getFatigueTerm ( ) * fSneakViewMult ;
}
2014-01-06 23:51:09 +00:00
float target = x - y ;
2015-04-22 15:58:55 +00:00
return ( Misc : : Rng : : roll0to99 ( ) > = target ) ;
2014-01-06 23:51:09 +00:00
}
2014-01-20 12:00:43 +00:00
2014-05-05 22:13:31 +00:00
void MechanicsManager : : startCombat ( const MWWorld : : Ptr & ptr , const MWWorld : : Ptr & target )
{
2019-06-27 16:50:54 +00:00
CreatureStats & stats = ptr . getClass ( ) . getCreatureStats ( ptr ) ;
2017-11-28 08:05:05 +00:00
2019-06-27 16:50:54 +00:00
// Don't add duplicate packages nor add packages to dead actors.
if ( stats . isDead ( ) | | stats . getAiSequence ( ) . isInCombat ( target ) )
2014-07-27 18:30:52 +00:00
return ;
2017-11-28 08:05:05 +00:00
2019-06-27 16:50:54 +00:00
// The target is somehow the same as the actor. Early-out.
if ( ptr = = target )
{
// We don't care about dialogue filters since the target is invalid.
// We still want to play the combat taunt.
MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > say ( ptr , " attack " ) ;
return ;
}
stats . getAiSequence ( ) . stack ( MWMechanics : : AiCombat ( target ) , ptr ) ;
2015-08-21 09:12:39 +00:00
if ( target = = getPlayer ( ) )
2014-06-14 19:36:57 +00:00
{
// if guard starts combat with player, guards pursuing player should do the same
if ( ptr . getClass ( ) . isClass ( ptr , " Guard " ) )
{
2019-06-27 16:50:54 +00:00
stats . setHitAttemptActorId ( target . getClass ( ) . getCreatureStats ( target ) . getActorId ( ) ) ; // Stops guard from ending combat if player is unreachable
2014-12-21 15:45:30 +00:00
for ( Actors : : PtrActorMap : : const_iterator iter = mActors . begin ( ) ; iter ! = mActors . end ( ) ; + + iter )
2014-06-14 19:36:57 +00:00
{
if ( iter - > first . getClass ( ) . isClass ( iter - > first , " Guard " ) )
{
MWMechanics : : AiSequence & aiSeq = iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . getAiSequence ( ) ;
2014-06-22 01:05:17 +00:00
if ( aiSeq . getTypeId ( ) = = MWMechanics : : AiPackage : : TypeIdPursue )
2014-06-14 19:36:57 +00:00
{
aiSeq . stopPursuit ( ) ;
aiSeq . stack ( MWMechanics : : AiCombat ( target ) , ptr ) ;
2017-02-06 12:32:36 +00:00
iter - > first . getClass ( ) . getCreatureStats ( iter - > first ) . setHitAttemptActorId ( target . getClass ( ) . getCreatureStats ( target ) . getActorId ( ) ) ; // Stops guard from ending combat if player is unreachable
2014-06-14 19:36:57 +00:00
}
}
}
}
}
2014-05-28 11:59:31 +00:00
// Must be done after the target is set up, so that CreatureTargetted dialogue filter works properly
2019-06-27 16:50:54 +00:00
MWBase : : Environment : : get ( ) . getDialogueManager ( ) - > say ( ptr , " attack " ) ;
2014-05-05 22:13:31 +00:00
}
2015-06-01 19:41:13 +00:00
void MechanicsManager : : getObjectsInRange ( const osg : : Vec3f & position , float radius , std : : vector < MWWorld : : Ptr > & objects )
2014-01-20 12:00:43 +00:00
{
2015-04-25 13:19:17 +00:00
mActors . getObjectsInRange ( position , radius , objects ) ;
mObjects . getObjectsInRange ( position , radius , objects ) ;
2014-01-20 12:00:43 +00:00
}
2014-01-28 11:33:31 +00:00
2015-06-01 19:41:13 +00:00
void MechanicsManager : : getActorsInRange ( const osg : : Vec3f & position , float radius , std : : vector < MWWorld : : Ptr > & objects )
2014-01-12 13:02:15 +00:00
{
2015-04-25 13:19:17 +00:00
mActors . getObjectsInRange ( position , radius , objects ) ;
2014-01-12 13:02:15 +00:00
}
2014-04-25 00:40:17 +00:00
2017-11-11 08:31:18 +00:00
bool MechanicsManager : : isAnyActorInRange ( const osg : : Vec3f & position , float radius )
{
return mActors . isAnyObjectInRange ( position , radius ) ;
}
2015-12-06 22:32:49 +00:00
std : : list < MWWorld : : Ptr > MechanicsManager : : getActorsSidingWith ( const MWWorld : : Ptr & actor )
2014-01-12 13:02:15 +00:00
{
2015-12-06 22:32:49 +00:00
return mActors . getActorsSidingWith ( actor ) ;
2014-01-12 13:02:15 +00:00
}
2014-04-25 02:47:45 +00:00
2015-12-19 14:11:07 +00:00
std : : list < MWWorld : : Ptr > MechanicsManager : : getActorsFollowing ( const MWWorld : : Ptr & actor )
{
return mActors . getActorsFollowing ( actor ) ;
}
2014-12-09 15:02:07 +00:00
std : : list < int > MechanicsManager : : getActorsFollowingIndices ( const MWWorld : : Ptr & actor )
{
2015-04-25 13:19:17 +00:00
return mActors . getActorsFollowingIndices ( actor ) ;
2014-12-09 15:02:07 +00:00
}
2014-04-25 02:47:45 +00:00
std : : list < MWWorld : : Ptr > MechanicsManager : : getActorsFighting ( const MWWorld : : Ptr & actor ) {
2015-04-25 13:19:17 +00:00
return mActors . getActorsFighting ( actor ) ;
2014-04-25 02:47:45 +00:00
}
2014-06-12 23:24:58 +00:00
2016-06-06 23:53:16 +00:00
std : : list < MWWorld : : Ptr > MechanicsManager : : getEnemiesNearby ( const MWWorld : : Ptr & actor ) {
return mActors . getEnemiesNearby ( actor ) ;
}
2016-12-20 11:38:51 +00:00
void MechanicsManager : : getActorsFollowing ( const MWWorld : : Ptr & actor , std : : set < MWWorld : : Ptr > & out ) {
mActors . getActorsFollowing ( actor , out ) ;
}
void MechanicsManager : : getActorsSidingWith ( const MWWorld : : Ptr & actor , std : : set < MWWorld : : Ptr > & out ) {
mActors . getActorsSidingWith ( actor , out ) ;
}
2014-06-12 23:24:58 +00:00
int MechanicsManager : : countSavedGameRecords ( ) const
{
2015-02-04 20:18:43 +00:00
return 1 // Death counter
+ 1 ; // Stolen items
2014-06-12 23:24:58 +00:00
}
void MechanicsManager : : write ( ESM : : ESMWriter & writer , Loading : : Listener & listener ) const
{
2015-04-25 13:19:17 +00:00
mActors . write ( writer , listener ) ;
2015-02-04 20:18:43 +00:00
ESM : : StolenItems items ;
items . mStolenItems = mStolenItems ;
writer . startRecord ( ESM : : REC_STLN ) ;
items . write ( writer ) ;
writer . endRecord ( ESM : : REC_STLN ) ;
2014-06-12 23:24:58 +00:00
}
2015-01-22 18:04:59 +00:00
void MechanicsManager : : readRecord ( ESM : : ESMReader & reader , uint32_t type )
2014-06-12 23:24:58 +00:00
{
2015-02-04 20:18:43 +00:00
if ( type = = ESM : : REC_STLN )
{
ESM : : StolenItems items ;
items . load ( reader ) ;
mStolenItems = items . mStolenItems ;
}
2015-04-25 13:19:17 +00:00
else
mActors . readRecord ( reader , type ) ;
2014-06-12 23:24:58 +00:00
}
void MechanicsManager : : clear ( )
{
2015-04-25 13:19:17 +00:00
mActors . clear ( ) ;
2015-02-04 20:18:43 +00:00
mStolenItems . clear ( ) ;
2016-04-03 18:23:28 +00:00
mClassSelected = false ;
mRaceSelected = false ;
2014-06-12 23:24:58 +00:00
}
2014-06-16 17:34:53 +00:00
2014-12-21 01:45:37 +00:00
bool MechanicsManager : : isAggressive ( const MWWorld : : Ptr & ptr , const MWWorld : : Ptr & target )
2014-06-16 17:34:53 +00:00
{
2017-12-05 15:31:48 +00:00
// Don't become aggressive if a calm effect is active, since it would cause combat to cycle on/off as
// combat is activated here and then canceled by the calm effect
if ( ( ptr . getClass ( ) . isNpc ( ) & & ptr . getClass ( ) . getCreatureStats ( ptr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : CalmHumanoid ) . getMagnitude ( ) > 0 )
| | ( ! ptr . getClass ( ) . isNpc ( ) & & ptr . getClass ( ) . getCreatureStats ( ptr ) . getMagicEffects ( ) . get ( ESM : : MagicEffect : : CalmCreature ) . getMagnitude ( ) > 0 ) )
return false ;
2014-06-16 17:34:53 +00:00
int disposition = 50 ;
if ( ptr . getClass ( ) . isNpc ( ) )
2017-08-03 15:12:21 +00:00
disposition = getDerivedDisposition ( ptr , true ) ;
2014-06-16 17:34:53 +00:00
2019-07-30 21:07:03 +00:00
int fight = ptr . getClass ( ) . getCreatureStats ( ptr ) . getAiSetting ( CreatureStats : : AI_Fight ) . getModified ( )
+ static_cast < int > ( getFightDistanceBias ( ptr , target ) + getFightDispositionBias ( static_cast < float > ( disposition ) ) ) ;
2014-06-16 17:34:53 +00:00
2014-09-06 01:09:11 +00:00
if ( ptr . getClass ( ) . isNpc ( ) & & target . getClass ( ) . isNpc ( ) )
2014-09-04 23:58:57 +00:00
{
2014-09-06 01:09:11 +00:00
if ( target . getClass ( ) . getNpcStats ( target ) . isWerewolf ( ) | |
2015-08-21 09:12:39 +00:00
( target = = getPlayer ( ) & &
2014-09-06 01:09:11 +00:00
MWBase : : Environment : : get ( ) . getWorld ( ) - > getGlobalInt ( " pcknownwerewolf " ) ) )
{
2016-01-21 15:08:04 +00:00
const ESM : : GameSetting * iWerewolfFightMod = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " iWerewolfFightMod " ) ;
2018-08-29 15:38:12 +00:00
fight + = iWerewolfFightMod - > mValue . getInteger ( ) ;
2014-09-06 01:09:11 +00:00
}
2014-09-04 23:58:57 +00:00
}
2014-06-16 17:34:53 +00:00
return ( fight > = 100 ) ;
}
2014-09-05 15:17:45 +00:00
2019-09-21 16:22:45 +00:00
void MechanicsManager : : resurrect ( const MWWorld : : Ptr & ptr )
2014-09-05 15:17:45 +00:00
{
2019-09-21 16:22:45 +00:00
CreatureStats & stats = ptr . getClass ( ) . getCreatureStats ( ptr ) ;
2014-09-05 15:17:45 +00:00
if ( stats . isDead ( ) )
2019-09-21 16:22:45 +00:00
{
2014-09-05 15:17:45 +00:00
stats . resurrect ( ) ;
2019-09-21 16:22:45 +00:00
mActors . resurrect ( ptr ) ;
}
2014-09-05 15:17:45 +00:00
}
2014-12-12 15:49:22 +00:00
2018-06-28 12:58:51 +00:00
bool MechanicsManager : : isCastingSpell ( const MWWorld : : Ptr & ptr ) const
{
return mActors . isCastingSpell ( ptr ) ;
}
2014-12-12 15:49:22 +00:00
bool MechanicsManager : : isReadyToBlock ( const MWWorld : : Ptr & ptr ) const
{
2015-04-25 13:19:17 +00:00
return mActors . isReadyToBlock ( ptr ) ;
2014-12-12 15:49:22 +00:00
}
2015-12-26 17:22:21 +00:00
2017-08-18 15:24:34 +00:00
bool MechanicsManager : : isAttackingOrSpell ( const MWWorld : : Ptr & ptr ) const
{
return mActors . isAttackingOrSpell ( ptr ) ;
}
2019-11-30 13:03:51 +00:00
/*
Start of tes3mp addition
Make it possible to set the attackingOrSpell state from elsewhere in the code
*/
void MechanicsManager : : setAttackingOrSpell ( const MWWorld : : Ptr & ptr , bool state ) const
{
return mActors . setAttackingOrSpell ( ptr , state ) ;
}
/*
End of tes3mp addition
*/
2015-12-26 17:22:21 +00:00
void MechanicsManager : : setWerewolf ( const MWWorld : : Ptr & actor , bool werewolf )
{
MWMechanics : : NpcStats & npcStats = actor . getClass ( ) . getNpcStats ( actor ) ;
// The actor does not have to change state
if ( npcStats . isWerewolf ( ) = = werewolf )
return ;
MWWorld : : Player * player = & MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayer ( ) ;
if ( actor = = player - > getPlayer ( ) )
{
if ( werewolf )
{
2018-03-30 11:05:36 +00:00
player - > saveStats ( ) ;
player - > setWerewolfStats ( ) ;
2015-12-26 17:22:21 +00:00
}
else
2018-03-30 11:05:36 +00:00
player - > restoreStats ( ) ;
2015-12-26 17:22:21 +00:00
}
2015-12-26 17:26:55 +00:00
// Werewolfs can not cast spells, so we need to unset the prepared spell if there is one.
if ( npcStats . getDrawState ( ) = = MWMechanics : : DrawState_Spell )
npcStats . setDrawState ( MWMechanics : : DrawState_Nothing ) ;
npcStats . setWerewolf ( werewolf ) ;
2016-11-22 15:37:49 +00:00
MWWorld : : InventoryStore & inv = actor . getClass ( ) . getInventoryStore ( actor ) ;
2015-12-26 17:22:21 +00:00
if ( werewolf )
{
2016-11-22 15:37:49 +00:00
inv . unequipAll ( actor ) ;
2015-12-26 17:22:21 +00:00
inv . equip ( MWWorld : : InventoryStore : : Slot_Robe , inv . ContainerStore : : add ( " werewolfrobe " , 1 , actor ) , actor ) ;
}
else
{
2016-11-22 15:37:49 +00:00
inv . unequipSlot ( MWWorld : : InventoryStore : : Slot_Robe , actor ) ;
inv . ContainerStore : : remove ( " werewolfrobe " , 1 , actor ) ;
2015-12-26 17:22:21 +00:00
}
if ( actor = = player - > getPlayer ( ) )
{
MWBase : : Environment : : get ( ) . getWorld ( ) - > reattachPlayerCamera ( ) ;
// Update the GUI only when called on the player
MWBase : : WindowManager * windowManager = MWBase : : Environment : : get ( ) . getWindowManager ( ) ;
if ( werewolf )
{
windowManager - > forceHide ( MWGui : : GW_Inventory ) ;
windowManager - > forceHide ( MWGui : : GW_Magic ) ;
}
else
{
windowManager - > unsetForceHide ( MWGui : : GW_Inventory ) ;
windowManager - > unsetForceHide ( MWGui : : GW_Magic ) ;
}
windowManager - > setWerewolfOverlay ( werewolf ) ;
// Witnesses of the player's transformation will make them a globally known werewolf
2018-11-05 22:35:20 +00:00
std : : vector < MWWorld : : Ptr > neighbors ;
2015-12-26 17:22:21 +00:00
const MWWorld : : Store < ESM : : GameSetting > & gmst = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
2018-11-05 22:35:20 +00:00
getActorsInRange ( actor . getRefData ( ) . getPosition ( ) . asVec3 ( ) , gmst . find ( " fAlarmRadius " ) - > mValue . getFloat ( ) , neighbors ) ;
2015-12-26 17:22:21 +00:00
bool detected = false , reported = false ;
2018-11-05 22:35:20 +00:00
for ( const MWWorld : : Ptr & neighbor : neighbors )
2015-12-26 17:22:21 +00:00
{
2018-11-05 22:35:20 +00:00
if ( neighbor = = actor | | ! neighbor . getClass ( ) . isNpc ( ) )
2015-12-26 17:22:21 +00:00
continue ;
2018-11-05 22:35:20 +00:00
if ( MWBase : : Environment : : get ( ) . getWorld ( ) - > getLOS ( neighbor , actor ) & & awarenessCheck ( actor , neighbor ) )
{
2015-12-26 17:22:21 +00:00
detected = true ;
2018-11-05 22:35:20 +00:00
if ( neighbor . getClass ( ) . getCreatureStats ( neighbor ) . getAiSetting ( MWMechanics : : CreatureStats : : AI_Alarm ) . getModified ( ) > 0 )
{
reported = true ;
break ;
}
}
2015-12-26 17:22:21 +00:00
}
if ( detected )
{
windowManager - > messageBox ( " #{sWerewolfAlarmMessage} " ) ;
MWBase : : Environment : : get ( ) . getWorld ( ) - > setGlobalInt ( " pcknownwerewolf " , 1 ) ;
if ( reported )
{
npcStats . setBounty ( npcStats . getBounty ( ) +
2018-08-29 15:38:12 +00:00
gmst . find ( " iWereWolfBounty " ) - > mValue . getInteger ( ) ) ;
2015-12-26 17:22:21 +00:00
}
}
}
}
void MechanicsManager : : applyWerewolfAcrobatics ( const MWWorld : : Ptr & actor )
{
const MWWorld : : Store < ESM : : GameSetting > & gmst = MWBase : : Environment : : get ( ) . getWorld ( ) - > getStore ( ) . get < ESM : : GameSetting > ( ) ;
MWMechanics : : NpcStats & stats = actor . getClass ( ) . getNpcStats ( actor ) ;
2018-08-29 15:38:12 +00:00
stats . getSkill ( ESM : : Skill : : Acrobatics ) . setBase ( gmst . find ( " fWerewolfAcrobatics " ) - > mValue . getInteger ( ) ) ;
2015-12-26 17:22:21 +00:00
}
2016-06-12 00:43:33 +00:00
void MechanicsManager : : cleanupSummonedCreature ( const MWWorld : : Ptr & caster , int creatureActorId )
{
mActors . cleanupSummonedCreature ( caster . getClass ( ) . getCreatureStats ( caster ) , creatureActorId ) ;
}
2010-07-26 09:15:38 +00:00
}