2010-07-26 09:15:38 +00:00
|
|
|
|
2012-08-11 15:30:55 +00:00
|
|
|
#include "mechanicsmanagerimp.hpp"
|
2014-04-03 04:50:09 +00:00
|
|
|
#include "npcstats.hpp"
|
2010-07-26 09:15:38 +00:00
|
|
|
|
2012-10-01 15:17:04 +00:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2013-03-16 21:53:33 +00:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2010-07-26 09:15:38 +00:00
|
|
|
|
2012-04-23 13:27:03 +00:00
|
|
|
#include "../mwbase/environment.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
|
|
|
|
2010-08-03 09:49:12 +00:00
|
|
|
#include "../mwworld/class.hpp"
|
2011-01-04 14:58:22 +00:00
|
|
|
#include "../mwworld/player.hpp"
|
2010-08-03 09:49:12 +00:00
|
|
|
|
2014-05-05 22:13:31 +00:00
|
|
|
#include "../mwmechanics/aicombat.hpp"
|
|
|
|
|
2014-01-06 23:51:09 +00:00
|
|
|
#include <OgreSceneNode.h>
|
|
|
|
|
2014-01-01 20:20:37 +00:00
|
|
|
#include "spellcasting.hpp"
|
|
|
|
|
2014-01-08 16:19:43 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
/// @return is \a ptr allowed to take/use \a item or is it a crime?
|
2014-01-10 20:26:24 +00:00
|
|
|
bool isAllowedToUse (const MWWorld::Ptr& ptr, const MWWorld::Ptr& item, MWWorld::Ptr& victim)
|
2014-01-08 16:19:43 +00:00
|
|
|
{
|
2014-05-25 12:13:07 +00:00
|
|
|
const std::string& owner = item.getCellRef().getOwner();
|
2014-05-15 02:52:35 +00:00
|
|
|
bool isOwned = !owner.empty() && owner != "player";
|
2014-01-08 16:19:43 +00:00
|
|
|
|
2014-05-25 12:13:07 +00:00
|
|
|
const std::string& faction = item.getCellRef().getFaction();
|
2014-01-08 16:19:43 +00:00
|
|
|
bool isFactionOwned = false;
|
2014-05-15 00:37:20 +00:00
|
|
|
if (!faction.empty() && ptr.getClass().isNpc())
|
2014-01-08 16:19:43 +00:00
|
|
|
{
|
|
|
|
const std::map<std::string, int>& factions = ptr.getClass().getNpcStats(ptr).getFactionRanks();
|
|
|
|
if (factions.find(Misc::StringUtils::lowerCase(faction)) == factions.end())
|
|
|
|
isFactionOwned = true;
|
|
|
|
}
|
|
|
|
|
2014-05-25 12:13:07 +00:00
|
|
|
if (!item.getCellRef().getOwner().empty())
|
|
|
|
victim = MWBase::Environment::get().getWorld()->searchPtr(item.getCellRef().getOwner(), true);
|
2014-01-10 20:26:24 +00:00
|
|
|
|
2014-01-08 16:19:43 +00:00
|
|
|
return (!isOwned && !isFactionOwned);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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()
|
|
|
|
{
|
2014-01-08 17:39:44 +00:00
|
|
|
MWWorld::Ptr ptr = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
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
|
|
|
|
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
|
2012-09-17 07:37:50 +00:00
|
|
|
creatureStats.setLevel(player->mNpdt52.mLevel);
|
2012-07-22 14:29:54 +00:00
|
|
|
creatureStats.getSpells().clear();
|
|
|
|
creatureStats.setMagicEffects(MagicEffects());
|
2010-09-15 11:31:26 +00:00
|
|
|
|
2010-09-16 08:45:08 +00:00
|
|
|
for (int i=0; i<27; ++i)
|
2012-09-17 07:37:50 +00:00
|
|
|
npcStats.getSkill (i).setBase (player->mNpdt52.mSkills[i]);
|
2010-09-16 08:45:08 +00:00
|
|
|
|
2013-12-28 16:19:35 +00:00
|
|
|
creatureStats.setAttribute(ESM::Attribute::Strength, player->mNpdt52.mStrength);
|
|
|
|
creatureStats.setAttribute(ESM::Attribute::Intelligence, player->mNpdt52.mIntelligence);
|
|
|
|
creatureStats.setAttribute(ESM::Attribute::Willpower, player->mNpdt52.mWillpower);
|
|
|
|
creatureStats.setAttribute(ESM::Attribute::Agility, player->mNpdt52.mAgility);
|
|
|
|
creatureStats.setAttribute(ESM::Attribute::Speed, player->mNpdt52.mSpeed);
|
|
|
|
creatureStats.setAttribute(ESM::Attribute::Endurance, player->mNpdt52.mEndurance);
|
|
|
|
creatureStats.setAttribute(ESM::Attribute::Personality, player->mNpdt52.mPersonality);
|
|
|
|
creatureStats.setAttribute(ESM::Attribute::Luck, player->mNpdt52.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
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
for (std::vector<std::string>::const_iterator iter (race->mPowers.mList.begin());
|
|
|
|
iter!=race->mPowers.mList.end(); ++iter)
|
2010-09-30 12:28:01 +00:00
|
|
|
{
|
2012-07-22 14:29:54 +00:00
|
|
|
creatureStats.getSpells().add (*iter);
|
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
|
|
|
|
2012-09-17 07:37:50 +00:00
|
|
|
for (std::vector<std::string>::const_iterator iter (sign->mPowers.mList.begin());
|
|
|
|
iter!=sign->mPowers.mList.end(); ++iter)
|
2010-09-30 12:28:01 +00:00
|
|
|
{
|
2012-07-22 14:29:54 +00:00
|
|
|
creatureStats.getSpells().add (*iter);
|
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
|
|
|
}
|
2014-01-01 20:20:37 +00:00
|
|
|
|
|
|
|
if (i==1)
|
|
|
|
{
|
|
|
|
// Major skill - add starting spells for this skill if existing
|
|
|
|
const MWWorld::ESMStore& store = MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
MWWorld::Store<ESM::Spell>::iterator it = store.get<ESM::Spell>().begin();
|
|
|
|
for (; it != store.get<ESM::Spell>().end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->mData.mFlags & ESM::Spell::F_PCStart
|
|
|
|
&& spellSchoolToSkill(getSpellSchool(&*it, ptr)) == index)
|
|
|
|
creatureStats.getSpells().add(it->mId);
|
|
|
|
}
|
|
|
|
}
|
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
|
|
|
{
|
2012-11-07 21:36:43 +00:00
|
|
|
if (iter->mData.mSpecialization==class_->mData.mSpecialization)
|
2010-09-26 07:55:00 +00:00
|
|
|
{
|
2012-11-05 17:19:22 +00:00
|
|
|
int index = iter->mIndex;
|
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
|
|
|
|
|
|
|
// forced update and current value adjustments
|
|
|
|
mActors.updateActor (ptr, 0);
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2012-04-23 13:27:03 +00:00
|
|
|
MechanicsManager::MechanicsManager()
|
|
|
|
: mUpdatePlayer (true), mClassSelected (false),
|
2013-11-18 22:03:44 +00:00
|
|
|
mRaceSelected (false), mAI(true)
|
2010-07-26 09:15:38 +00:00
|
|
|
{
|
2013-11-14 13:41:10 +00:00
|
|
|
//buildPlayer no longer here, needs to be done explicitely 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
|
|
|
|
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-11-19 15:42:24 +00:00
|
|
|
void MechanicsManager::advanceTime (float duration)
|
|
|
|
{
|
|
|
|
// Uses ingame time, but scaled to real time
|
|
|
|
duration /= MWBase::Environment::get().getWorld()->getTimeScaleFactor();
|
2014-01-08 17:39:44 +00:00
|
|
|
MWWorld::Ptr player = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
2013-11-19 15:42:24 +00:00
|
|
|
player.getClass().getInventoryStore(player).rechargeItems(duration);
|
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
2013-08-28 00:56:47 +00:00
|
|
|
if(stats.getAttribute(i) != mWatchedStats.getAttribute(i))
|
2010-07-27 13:59:41 +00:00
|
|
|
{
|
2013-08-28 17:50:29 +00:00
|
|
|
std::stringstream attrname;
|
|
|
|
attrname << "AttribVal"<<(i+1);
|
|
|
|
|
2013-08-28 00:56:47 +00:00
|
|
|
mWatchedStats.setAttribute(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
|
|
|
|
2013-08-28 00:56:47 +00:00
|
|
|
if(stats.getHealth() != mWatchedStats.getHealth())
|
|
|
|
{
|
2013-08-28 17:50:29 +00:00
|
|
|
static const std::string hbar("HBar");
|
2013-08-28 00:56:47 +00:00
|
|
|
mWatchedStats.setHealth(stats.getHealth());
|
2013-08-28 17:50:29 +00:00
|
|
|
winMgr->setValue(hbar, stats.getHealth());
|
2012-07-22 14:29:54 +00:00
|
|
|
}
|
2013-08-28 00:56:47 +00:00
|
|
|
if(stats.getMagicka() != mWatchedStats.getMagicka())
|
|
|
|
{
|
2013-08-28 17:50:29 +00:00
|
|
|
static const std::string mbar("MBar");
|
2013-08-28 00:56:47 +00:00
|
|
|
mWatchedStats.setMagicka(stats.getMagicka());
|
2013-08-28 17:50:29 +00:00
|
|
|
winMgr->setValue(mbar, stats.getMagicka());
|
2012-07-22 14:29:54 +00:00
|
|
|
}
|
2013-08-28 00:56:47 +00:00
|
|
|
if(stats.getFatigue() != mWatchedStats.getFatigue())
|
|
|
|
{
|
2013-08-28 17:50:29 +00:00
|
|
|
static const std::string fbar("FBar");
|
2013-08-28 00:56:47 +00:00
|
|
|
mWatchedStats.setFatigue(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
|
|
|
|
2013-08-28 00:56:47 +00:00
|
|
|
if(stats.getTimeToStartDrowning() != mWatchedStats.getTimeToStartDrowning())
|
2013-08-07 13:34:11 +00:00
|
|
|
{
|
2014-04-27 02:27:26 +00:00
|
|
|
const float fHoldBreathTime = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>()
|
|
|
|
.find("fHoldBreathTime")->getFloat();
|
2013-08-28 00:56:47 +00:00
|
|
|
mWatchedStats.setTimeToStartDrowning(stats.getTimeToStartDrowning());
|
2014-04-27 02:27:26 +00:00
|
|
|
if(stats.getTimeToStartDrowning() >= fHoldBreathTime)
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-21 15:42:07 +00:00
|
|
|
bool update = false;
|
|
|
|
|
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
|
|
|
{
|
2013-08-28 00:56:47 +00:00
|
|
|
if(stats.getSkill(i) != mWatchedStats.getSkill(i))
|
2010-09-21 15:42:07 +00:00
|
|
|
{
|
|
|
|
update = true;
|
2013-08-28 00:56:47 +00:00
|
|
|
mWatchedStats.getSkill(i) = stats.getSkill(i);
|
|
|
|
winMgr->setValue((ESM::Skill::SkillEnum)i, stats.getSkill(i));
|
2010-09-21 15:42:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-28 00:56:47 +00:00
|
|
|
if(update)
|
|
|
|
winMgr->updateSkillArea();
|
2010-09-21 15:42:07 +00:00
|
|
|
|
2013-08-28 00:56:47 +00:00
|
|
|
winMgr->setValue("level", stats.getLevel());
|
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 if (!winMgr->getSelectedSpell().empty())
|
|
|
|
winMgr->setSelectedSpell(winMgr->getSelectedSpell(), int(MWMechanics::getSpellSuccessChance(winMgr->getSelectedSpell(), mWatched)));
|
|
|
|
else
|
2014-04-27 10:54:18 +00:00
|
|
|
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.
|
2014-01-08 17:39:44 +00:00
|
|
|
MWWorld::Ptr ptr = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
2013-02-04 20:58:06 +00:00
|
|
|
mActors.removeActor(ptr);
|
2014-05-13 17:01:02 +00:00
|
|
|
mActors.addActor(ptr, true);
|
2010-09-15 10:22:06 +00:00
|
|
|
}
|
2011-02-03 10:43:29 +00:00
|
|
|
|
2013-01-29 07:39:11 +00:00
|
|
|
mActors.update(duration, paused);
|
2013-03-31 22:29:41 +00:00
|
|
|
mObjects.update(duration, paused);
|
2010-07-27 12:46:05 +00:00
|
|
|
}
|
2010-09-13 20:59:28 +00:00
|
|
|
|
2014-01-14 01:20:13 +00:00
|
|
|
void MechanicsManager::rest(bool sleep)
|
2012-09-21 15:53:16 +00:00
|
|
|
{
|
2014-01-14 01:20:13 +00:00
|
|
|
mActors.restoreDynamicStats (sleep);
|
2012-09-21 15:53:16 +00:00
|
|
|
}
|
|
|
|
|
2014-01-14 01:52:34 +00:00
|
|
|
int MechanicsManager::getHoursToRest() const
|
|
|
|
{
|
|
|
|
return mActors.getHoursToRest(mWatched);
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2012-11-09 13:42:09 +00:00
|
|
|
int MechanicsManager::getDerivedDisposition(const MWWorld::Ptr& ptr)
|
2012-11-05 10:07:43 +00:00
|
|
|
{
|
2014-05-22 18:37:22 +00:00
|
|
|
const MWMechanics::NpcStats& npcSkill = ptr.getClass().getNpcStats(ptr);
|
2012-11-09 19:18:38 +00:00
|
|
|
float x = npcSkill.getBaseDisposition();
|
2012-11-05 18:55:06 +00:00
|
|
|
|
|
|
|
MWWorld::LiveCellRef<ESM::NPC>* npc = ptr.get<ESM::NPC>();
|
2014-01-08 17:39:44 +00:00
|
|
|
MWWorld::Ptr playerPtr = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
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
|
|
|
|
2013-11-21 03:53:53 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(npc->mBase->mRace, player->mBase->mRace))
|
2013-08-09 08:14:08 +00:00
|
|
|
x += MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispRaceMod")->getFloat();
|
2012-11-05 18:55:06 +00:00
|
|
|
|
2012-11-08 22:16:40 +00:00
|
|
|
x += MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispPersonalityMult")->getFloat()
|
|
|
|
* (playerStats.getAttribute(ESM::Attribute::Personality).getModified() - MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispPersonalityBase")->getFloat());
|
2012-11-05 18:55:06 +00:00
|
|
|
|
|
|
|
float reaction = 0;
|
|
|
|
int rank = 0;
|
2012-11-06 12:10:54 +00:00
|
|
|
std::string npcFaction = "";
|
|
|
|
if(!npcSkill.getFactionRanks().empty()) npcFaction = npcSkill.getFactionRanks().begin()->first;
|
2012-11-05 18:55:06 +00:00
|
|
|
|
2014-01-14 08:46:53 +00:00
|
|
|
Misc::StringUtils::toLower(npcFaction);
|
|
|
|
|
|
|
|
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-05-27 12:54:29 +00:00
|
|
|
reaction = playerStats.getFactionReputation(npcFaction);
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
int itReaction = MWBase::Environment::get().getDialogueManager()->getFactionReaction(npcFaction, itFaction);
|
|
|
|
if (playerFactionIt == playerStats.getFactionRanks().begin() || itReaction < reaction)
|
|
|
|
reaction = itReaction;
|
2012-11-05 18:55:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reaction = 0;
|
|
|
|
rank = 0;
|
|
|
|
}
|
2012-11-08 22:16:40 +00:00
|
|
|
x += (MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispFactionRankMult")->getFloat() * rank
|
|
|
|
+ MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispFactionRankBase")->getFloat())
|
|
|
|
* MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispFactionMod")->getFloat() * reaction;
|
2012-11-08 12:38:20 +00:00
|
|
|
|
2013-08-09 08:14:08 +00:00
|
|
|
x -= MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispCrimeMod")->getFloat() * playerStats.getBounty();
|
2012-11-09 17:33:11 +00:00
|
|
|
if (playerStats.hasCommonDisease() || playerStats.hasBlightDisease())
|
|
|
|
x += MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispDiseaseMod")->getFloat();
|
|
|
|
|
2013-08-09 08:14:08 +00:00
|
|
|
if (playerStats.getDrawState() == MWMechanics::DrawState_Weapon)
|
2012-11-09 17:33:11 +00:00
|
|
|
x += MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fDispWeaponDrawn")->getFloat();
|
2012-11-05 18:55:06 +00:00
|
|
|
|
2014-01-05 00:56:36 +00:00
|
|
|
x += ptr.getClass().getCreatureStats(ptr).getMagicEffects().get(ESM::MagicEffect::Charm).mMagnitude;
|
|
|
|
|
2012-11-05 18:55:06 +00:00
|
|
|
int effective_disposition = std::max(0,std::min(int(x),100));//, normally clamped to [0..100] when used
|
|
|
|
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
|
|
|
|
2014-01-08 17:39:44 +00:00
|
|
|
MWWorld::Ptr playerPtr = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
2014-05-22 18:37:22 +00:00
|
|
|
const MWMechanics::NpcStats &playerStats = playerPtr.getClass().getNpcStats(playerPtr);
|
2012-11-05 10:07:43 +00:00
|
|
|
|
2012-11-09 23:42:31 +00:00
|
|
|
// I suppose the temporary disposition change _has_ to be considered here,
|
|
|
|
// otherwise one would get different prices when exiting and re-entering the dialogue window...
|
|
|
|
int clampedDisposition = std::max(0, std::min(getDerivedDisposition(ptr)
|
|
|
|
+ MWBase::Environment::get().getDialogueManager()->getTemporaryDispositionChange(),100));
|
2014-01-03 02:46:30 +00:00
|
|
|
float a = std::min(playerStats.getSkill(ESM::Skill::Mercantile).getModified(), 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);
|
2014-01-03 02:46:30 +00:00
|
|
|
float d = std::min(sellerStats.getSkill(ESM::Skill::Mercantile).getModified(), 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-12-08 23:12:24 +00:00
|
|
|
|
2012-11-05 10:07:43 +00:00
|
|
|
float pcTerm = (clampedDisposition - 50 + a + b + c) * playerStats.getFatigueTerm();
|
|
|
|
float npcTerm = (d + e + f) * sellerStats.getFatigueTerm();
|
|
|
|
float buyTerm = 0.01 * (100 - 0.5 * (pcTerm - npcTerm));
|
|
|
|
float sellTerm = 0.01 * (50 - 0.5 * (npcTerm - pcTerm));
|
|
|
|
|
2012-12-08 23:12:24 +00:00
|
|
|
float x;
|
2012-11-05 10:07:43 +00:00
|
|
|
if(buying) x = buyTerm;
|
2012-11-05 18:55:06 +00:00
|
|
|
else x = std::min(buyTerm, sellTerm);
|
2012-11-05 10:07:43 +00:00
|
|
|
int offerPrice;
|
2013-02-25 15:31:48 +00:00
|
|
|
if (x < 1)
|
|
|
|
offerPrice = int(x * basePrice);
|
|
|
|
else
|
|
|
|
offerPrice = basePrice + int((x - 1) * basePrice);
|
2012-11-05 18:55:06 +00:00
|
|
|
offerPrice = std::max(1, offerPrice);
|
2012-11-05 10:07:43 +00:00
|
|
|
return offerPrice;
|
|
|
|
}
|
2012-11-08 22:16:40 +00:00
|
|
|
|
2012-10-27 09:33:18 +00:00
|
|
|
int MechanicsManager::countDeaths (const std::string& id) const
|
|
|
|
{
|
|
|
|
return mActors.countDeaths (id);
|
|
|
|
}
|
2012-11-09 19:18:38 +00:00
|
|
|
|
|
|
|
|
2012-11-09 23:29:36 +00:00
|
|
|
void MechanicsManager::getPersuasionDispositionChange (const MWWorld::Ptr& npc, PersuasionType type,
|
|
|
|
float currentTemporaryDispositionDelta, 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
|
|
|
|
2014-01-08 17:39:44 +00:00
|
|
|
MWWorld::Ptr playerPtr = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
2014-05-22 18:37:22 +00:00
|
|
|
const MWMechanics::NpcStats &playerStats = playerPtr.getClass().getNpcStats(playerPtr);
|
2012-11-09 23:29:36 +00:00
|
|
|
|
|
|
|
float persTerm = playerStats.getAttribute(ESM::Attribute::Personality).getModified()
|
|
|
|
/ gmst.find("fPersonalityMod")->getFloat();
|
|
|
|
|
|
|
|
float luckTerm = playerStats.getAttribute(ESM::Attribute::Luck).getModified()
|
|
|
|
/ gmst.find("fLuckMod")->getFloat();
|
|
|
|
|
2013-08-09 08:14:08 +00:00
|
|
|
float repTerm = playerStats.getReputation() * gmst.find("fReputationMod")->getFloat();
|
2012-11-09 23:29:36 +00:00
|
|
|
float levelTerm = playerStats.getLevel() * gmst.find("fLevelMod")->getFloat();
|
|
|
|
|
|
|
|
float fatigueTerm = playerStats.getFatigueTerm();
|
|
|
|
|
2013-08-09 08:14:08 +00:00
|
|
|
float playerRating1 = (repTerm + luckTerm + persTerm + playerStats.getSkill(ESM::Skill::Speechcraft).getModified()) * fatigueTerm;
|
2012-11-09 23:29:36 +00:00
|
|
|
float playerRating2 = playerRating1 + levelTerm;
|
2013-08-09 08:14:08 +00:00
|
|
|
float playerRating3 = (playerStats.getSkill(ESM::Skill::Mercantile).getModified() + luckTerm + persTerm) * fatigueTerm;
|
2012-11-09 23:29:36 +00:00
|
|
|
|
2013-08-09 08:14:08 +00:00
|
|
|
float npcRating1 = (repTerm + luckTerm + persTerm + playerStats.getSkill(ESM::Skill::Speechcraft).getModified()) * fatigueTerm;
|
|
|
|
float npcRating2 = (levelTerm + repTerm + luckTerm + persTerm + npcStats.getSkill(ESM::Skill::Speechcraft).getModified()) * fatigueTerm;
|
|
|
|
float npcRating3 = (playerStats.getSkill(ESM::Skill::Mercantile).getModified() + repTerm + luckTerm + persTerm) * fatigueTerm;
|
2012-11-09 23:29:36 +00:00
|
|
|
|
|
|
|
int currentDisposition = std::min(100, std::max(0, int(getDerivedDisposition(npc) + currentTemporaryDispositionDelta)));
|
|
|
|
|
|
|
|
float d = 1 - 0.02 * abs(currentDisposition - 50);
|
|
|
|
float target1 = d * (playerRating1 - npcRating1 + 50);
|
|
|
|
float target2 = d * (playerRating2 - npcRating2 + 50);
|
|
|
|
|
|
|
|
float bribeMod;
|
|
|
|
if (type == PT_Bribe10) bribeMod = gmst.find("fBribe10Mod")->getFloat();
|
2013-04-18 19:37:58 +00:00
|
|
|
else if (type == PT_Bribe100) bribeMod = gmst.find("fBribe100Mod")->getFloat();
|
2012-11-09 23:29:36 +00:00
|
|
|
else bribeMod = gmst.find("fBribe1000Mod")->getFloat();
|
|
|
|
|
|
|
|
float target3 = d * (playerRating3 - npcRating3 + 50) + bribeMod;
|
|
|
|
|
|
|
|
float iPerMinChance = gmst.find("iPerMinChance")->getInt();
|
|
|
|
float iPerMinChange = gmst.find("iPerMinChange")->getInt();
|
|
|
|
float fPerDieRollMult = gmst.find("fPerDieRollMult")->getFloat();
|
|
|
|
float fPerTempMult = gmst.find("fPerTempMult")->getFloat();
|
|
|
|
|
2013-02-25 15:52:31 +00:00
|
|
|
float x = 0;
|
|
|
|
float y = 0;
|
2012-11-09 23:29:36 +00:00
|
|
|
|
|
|
|
float roll = static_cast<float> (std::rand()) / RAND_MAX * 100;
|
|
|
|
|
|
|
|
if (type == PT_Admire)
|
|
|
|
{
|
|
|
|
target1 = std::max(iPerMinChance, target1);
|
|
|
|
success = (roll <= target1);
|
|
|
|
float c = int(fPerDieRollMult * (target1 - roll));
|
|
|
|
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)
|
|
|
|
r = int(target2 - roll);
|
|
|
|
else
|
|
|
|
r = 1;
|
|
|
|
|
|
|
|
if (roll <= target2)
|
|
|
|
{
|
|
|
|
float s = int(r * fPerDieRollMult * fPerTempMult);
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
float c = -std::abs(int(r * fPerDieRollMult));
|
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
if (std::abs(c) < iPerMinChange)
|
|
|
|
{
|
|
|
|
x = 0;
|
|
|
|
y = -iPerMinChange;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
x = -int(c * fPerTempMult);
|
|
|
|
y = c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
x = int(c * fPerTempMult);
|
|
|
|
y = c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (type == PT_Taunt)
|
|
|
|
{
|
|
|
|
target1 = std::max(iPerMinChance, target1);
|
|
|
|
success = (roll <= target1);
|
|
|
|
|
|
|
|
float c = std::abs(int(target1 - roll));
|
|
|
|
|
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
|
|
|
}
|
|
|
|
x = int(-c * fPerDieRollMult);
|
|
|
|
|
|
|
|
if (success && std::abs(x) < iPerMinChange)
|
|
|
|
x = -iPerMinChange;
|
|
|
|
}
|
|
|
|
else // Bribe
|
|
|
|
{
|
|
|
|
target3 = std::max(iPerMinChance, target3);
|
|
|
|
success = (roll <= target3);
|
|
|
|
float c = int((target3 - roll) * fPerDieRollMult);
|
|
|
|
|
|
|
|
x = success ? std::max(iPerMinChange, c) : c;
|
|
|
|
}
|
|
|
|
|
|
|
|
tempChange = type == PT_Intimidate ? x : int(x * fPerTempMult);
|
|
|
|
|
|
|
|
|
|
|
|
float cappedDispositionChange = tempChange;
|
|
|
|
if (currentDisposition + tempChange > 100.f)
|
|
|
|
cappedDispositionChange = 100 - currentDisposition;
|
|
|
|
if (currentDisposition + tempChange < 0.f)
|
|
|
|
cappedDispositionChange = -currentDisposition;
|
|
|
|
|
|
|
|
permChange = int(cappedDispositionChange / fPerTempMult);
|
|
|
|
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)
|
|
|
|
{
|
2014-05-22 18:37:22 +00:00
|
|
|
if(ptr.getClass().isActor())
|
2013-04-25 14:08:11 +00:00
|
|
|
mActors.forceStateUpdate(ptr);
|
|
|
|
}
|
|
|
|
|
2013-01-17 01:53:18 +00:00
|
|
|
void MechanicsManager::playAnimationGroup(const MWWorld::Ptr& ptr, const std::string& groupName, int mode, int number)
|
|
|
|
{
|
2014-05-22 18:37:22 +00:00
|
|
|
if(ptr.getClass().isActor())
|
2013-01-29 07:39:11 +00:00
|
|
|
mActors.playAnimationGroup(ptr, groupName, mode, number);
|
2013-03-31 22:12:10 +00:00
|
|
|
else
|
2013-03-31 22:29:41 +00:00
|
|
|
mObjects.playAnimationGroup(ptr, groupName, mode, number);
|
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)
|
|
|
|
{
|
2014-05-22 18:37:22 +00:00
|
|
|
if(ptr.getClass().isActor())
|
2013-05-25 03:10:07 +00:00
|
|
|
return mActors.checkAnimationPlaying(ptr, groupName);
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
2013-01-17 01:53:18 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
void MechanicsManager::updateMagicEffects(const MWWorld::Ptr &ptr)
|
|
|
|
{
|
|
|
|
mActors.updateMagicEffects(ptr);
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
mAI = true;
|
|
|
|
}
|
2014-01-24 17:21:52 +00:00
|
|
|
|
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-04-25 02:47:45 +00:00
|
|
|
if(MWBase::Environment::get().getWorld()->getPlayer().isInCombat()) {
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sNotifyMessage2}");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-01-08 16:19:43 +00:00
|
|
|
MWWorld::Ptr victim;
|
2014-01-10 20:26:24 +00:00
|
|
|
if (isAllowedToUse(ptr, bed, victim))
|
|
|
|
return false;
|
2014-01-07 19:24:01 +00:00
|
|
|
|
2014-01-08 16:19:43 +00:00
|
|
|
if(commitCrime(ptr, victim, OT_SleepingInOwnedBed))
|
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
|
|
|
|
2014-01-10 20:26:24 +00:00
|
|
|
void MechanicsManager::objectOpened(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;
|
2014-01-10 20:26:24 +00:00
|
|
|
if (isAllowedToUse(ptr, item, victim))
|
|
|
|
return;
|
|
|
|
commitCrime(ptr, victim, OT_Trespassing);
|
|
|
|
}
|
2014-01-08 00:24:06 +00:00
|
|
|
|
2014-01-10 20:26:24 +00:00
|
|
|
void MechanicsManager::itemTaken(const MWWorld::Ptr &ptr, const MWWorld::Ptr &item, int count)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr victim;
|
|
|
|
if (isAllowedToUse(ptr, item, victim))
|
|
|
|
return;
|
2014-01-07 18:49:16 +00:00
|
|
|
commitCrime(ptr, victim, OT_Theft, item.getClass().getValue(item) * count);
|
|
|
|
}
|
|
|
|
|
2014-05-09 16:45:49 +00:00
|
|
|
bool MechanicsManager::commitCrime(const MWWorld::Ptr &player, const MWWorld::Ptr &victim, OffenseType type, int arg)
|
2014-01-07 18:49:16 +00:00
|
|
|
{
|
2014-04-01 18:15:55 +00:00
|
|
|
// NOTE: int arg can be from itemTaken() so DON'T modify it, since it is
|
|
|
|
// passed to reportCrime later on in this function.
|
|
|
|
|
2014-04-19 23:03:31 +00:00
|
|
|
// Only player can commit crime
|
2014-05-09 16:45:49 +00:00
|
|
|
if (player.getRefData().getHandle() != "player")
|
2014-01-11 02:08:16 +00:00
|
|
|
return false;
|
2014-01-08 00:24:06 +00:00
|
|
|
|
2014-04-01 18:15:55 +00:00
|
|
|
const MWWorld::ESMStore& esmStore = MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
|
|
|
|
// What amount of alarm did this crime generate?
|
|
|
|
int alarm;
|
|
|
|
if (type == OT_Trespassing || type == OT_SleepingInOwnedBed)
|
|
|
|
alarm = esmStore.get<ESM::GameSetting>().find("iAlarmTresspass")->getInt();
|
|
|
|
else if (type == OT_Pickpocket)
|
|
|
|
alarm = esmStore.get<ESM::GameSetting>().find("iAlarmPickPocket")->getInt();
|
|
|
|
else if (type == OT_Assault)
|
|
|
|
alarm = esmStore.get<ESM::GameSetting>().find("iAlarmAttack")->getInt();
|
|
|
|
else if (type == OT_Murder)
|
|
|
|
alarm = esmStore.get<ESM::GameSetting>().find("iAlarmKilling")->getInt();
|
|
|
|
else if (type == OT_Theft)
|
|
|
|
alarm = esmStore.get<ESM::GameSetting>().find("iAlarmStealing")->getInt();
|
2014-04-19 23:42:49 +00:00
|
|
|
else
|
|
|
|
return false;
|
2014-04-01 18:15:55 +00:00
|
|
|
|
|
|
|
// Innocent until proven guilty
|
|
|
|
bool reported = false;
|
|
|
|
|
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
|
|
|
|
|
|
|
Ogre::Vector3 from = Ogre::Vector3(player.getRefData().getPosition().pos);
|
|
|
|
int radius = esmStore.get<ESM::GameSetting>().find("fAlarmRadius")->getInt();
|
|
|
|
|
|
|
|
mActors.getObjectsInRange(from, radius, neighbors);
|
|
|
|
|
|
|
|
// victim should be considered even beyond alarm radius
|
|
|
|
if (from.squaredDistance(Ogre::Vector3(victim.getRefData().getPosition().pos)) > radius*radius)
|
|
|
|
neighbors.push_back(victim);
|
2014-04-05 14:26:14 +00:00
|
|
|
|
2014-05-06 16:23:17 +00:00
|
|
|
int id = MWBase::Environment::get().getWorld()->getPlayer().getNewCrimeId();
|
|
|
|
|
|
|
|
// Find actors who witnessed the crime
|
2014-04-01 18:15:55 +00:00
|
|
|
for (std::vector<MWWorld::Ptr>::iterator it = neighbors.begin(); it != neighbors.end(); ++it)
|
2014-04-25 02:47:45 +00:00
|
|
|
{
|
2014-05-09 16:45:49 +00:00
|
|
|
if (*it == player) continue; // not the player
|
2014-04-01 18:15:55 +00:00
|
|
|
|
2014-04-03 20:13:14 +00:00
|
|
|
// Was the crime seen?
|
2014-05-09 16:45:49 +00:00
|
|
|
if (MWBase::Environment::get().getWorld()->getLOS(player, *it) && awarenessCheck(player, *it) )
|
2014-01-07 18:49:16 +00:00
|
|
|
{
|
2014-05-06 16:23:17 +00:00
|
|
|
// TODO: Add more messages
|
|
|
|
if (type == OT_Theft)
|
|
|
|
MWBase::Environment::get().getDialogueManager()->say(*it, "thief");
|
|
|
|
|
|
|
|
if (*it == victim)
|
|
|
|
{
|
|
|
|
// Self-defense
|
|
|
|
// The victim is aware of the criminal/assailant. If being assaulted, fight back now
|
|
|
|
// (regardless of whether the assault is reported or not)
|
|
|
|
// This applies to both NPCs and creatures
|
|
|
|
|
|
|
|
// ... except if this is a guard: then the player is given a chance to pay a fine / go to jail instead
|
2014-05-09 16:45:49 +00:00
|
|
|
if (type == OT_Assault && !it->getClass().isClass(*it, "guard"))
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->startCombat(victim, player);
|
2014-05-06 16:23:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Crime reporting only applies to NPCs
|
|
|
|
if (!it->getClass().isNpc())
|
|
|
|
continue;
|
2014-01-07 18:49:16 +00:00
|
|
|
|
2014-04-01 18:15:55 +00:00
|
|
|
// Will the witness report the crime?
|
2014-04-05 14:26:14 +00:00
|
|
|
if (it->getClass().getCreatureStats(*it).getAiSetting(CreatureStats::AI_Alarm).getBase() >= alarm)
|
2014-01-07 18:49:16 +00:00
|
|
|
{
|
2014-04-01 18:15:55 +00:00
|
|
|
reported = true;
|
|
|
|
|
2014-04-05 14:26:14 +00:00
|
|
|
// Tell everyone, including yourself
|
2014-04-01 18:15:55 +00:00
|
|
|
for (std::vector<MWWorld::Ptr>::iterator it1 = neighbors.begin(); it1 != neighbors.end(); ++it1)
|
2014-04-25 02:47:45 +00:00
|
|
|
{
|
2014-05-09 16:45:49 +00:00
|
|
|
if ( *it1 == player
|
2014-04-30 11:44:29 +00:00
|
|
|
|| !it1->getClass().isNpc()) continue; // not the player and is an NPC
|
2014-04-06 02:45:40 +00:00
|
|
|
|
2014-04-05 14:26:14 +00:00
|
|
|
// Will other witnesses paticipate in crime
|
2014-04-25 02:47:45 +00:00
|
|
|
if ( it1->getClass().getCreatureStats(*it1).getAiSetting(CreatureStats::AI_Alarm).getBase() >= alarm
|
2014-04-05 14:26:14 +00:00
|
|
|
|| type == OT_Assault )
|
|
|
|
{
|
|
|
|
it1->getClass().getNpcStats(*it1).setCrimeId(id);
|
|
|
|
}
|
2014-04-18 11:44:09 +00:00
|
|
|
|
|
|
|
// Mark as Alarmed for dialogue
|
|
|
|
it1->getClass().getCreatureStats(*it1).setAlarmed(true);
|
2014-04-01 18:15:55 +00:00
|
|
|
}
|
2014-01-07 18:49:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (reported)
|
2014-05-09 16:45:49 +00:00
|
|
|
reportCrime(player, victim, type, arg);
|
2014-01-08 16:19:43 +00:00
|
|
|
return reported;
|
2014-01-07 18:49:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MechanicsManager::reportCrime(const MWWorld::Ptr &ptr, const MWWorld::Ptr &victim, OffenseType type, int arg)
|
|
|
|
{
|
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-01-07 18:49:16 +00:00
|
|
|
// Bounty for each type of crime
|
|
|
|
if (type == OT_Trespassing || type == OT_SleepingInOwnedBed)
|
2014-01-09 00:55:49 +00:00
|
|
|
arg = store.find("iCrimeTresspass")->getInt();
|
2014-01-07 18:49:16 +00:00
|
|
|
else if (type == OT_Pickpocket)
|
2014-01-09 00:55:49 +00:00
|
|
|
arg = store.find("iCrimePickPocket")->getInt();
|
2014-01-07 18:49:16 +00:00
|
|
|
else if (type == OT_Assault)
|
2014-01-09 00:55:49 +00:00
|
|
|
arg = store.find("iCrimeAttack")->getInt();
|
2014-01-07 18:49:16 +00:00
|
|
|
else if (type == OT_Murder)
|
2014-01-09 00:55:49 +00:00
|
|
|
arg = store.find("iCrimeKilling")->getInt();
|
|
|
|
else if (type == OT_Theft)
|
2014-06-09 01:42:29 +00:00
|
|
|
{
|
2014-01-09 00:55:49 +00:00
|
|
|
arg *= store.find("fCrimeStealing")->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
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sCrimeMessage}");
|
|
|
|
ptr.getClass().getNpcStats(ptr).setBounty(ptr.getClass().getNpcStats(ptr).getBounty()
|
|
|
|
+ arg);
|
|
|
|
|
2014-01-08 18:26:03 +00:00
|
|
|
// If committing a crime against a faction member, expell from the faction
|
|
|
|
if (!victim.isEmpty() && victim.getClass().isNpc())
|
|
|
|
{
|
|
|
|
std::string factionID;
|
|
|
|
if(!victim.getClass().getNpcStats(victim).getFactionRanks().empty())
|
|
|
|
factionID = victim.getClass().getNpcStats(victim).getFactionRanks().begin()->first;
|
|
|
|
if (ptr.getClass().getNpcStats(ptr).isSameFaction(victim.getClass().getNpcStats(victim)))
|
|
|
|
{
|
|
|
|
ptr.getClass().getNpcStats(ptr).expell(factionID);
|
|
|
|
}
|
|
|
|
}
|
2014-01-07 18:49:16 +00:00
|
|
|
}
|
|
|
|
|
2014-01-06 23:51:09 +00:00
|
|
|
bool MechanicsManager::awarenessCheck(const MWWorld::Ptr &ptr, const MWWorld::Ptr &observer)
|
|
|
|
{
|
2014-01-11 01:06:54 +00:00
|
|
|
if (observer.getClass().getCreatureStats(observer).isDead())
|
|
|
|
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);
|
|
|
|
|
|
|
|
float invisibility = stats.getMagicEffects().get(ESM::MagicEffect::Invisibility).mMagnitude;
|
|
|
|
if (invisibility > 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
float sneakTerm = 0;
|
2014-01-15 06:47:21 +00:00
|
|
|
if (ptr.getClass().getCreatureStats(ptr).getStance(CreatureStats::Stance_Sneak)
|
2014-01-07 00:20:13 +00:00
|
|
|
&& !MWBase::Environment::get().getWorld()->isSwimming(ptr)
|
|
|
|
&& MWBase::Environment::get().getWorld()->isOnGround(ptr))
|
2014-01-06 23:51:09 +00:00
|
|
|
{
|
|
|
|
static float fSneakSkillMult = store.find("fSneakSkillMult")->getFloat();
|
|
|
|
static float fSneakBootMult = store.find("fSneakBootMult")->getFloat();
|
2014-01-15 14:50:45 +00:00
|
|
|
float sneak = 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;
|
|
|
|
if (ptr.getClass().isNpc())
|
|
|
|
{
|
|
|
|
MWWorld::InventoryStore& inv = ptr.getClass().getInventoryStore(ptr);
|
|
|
|
MWWorld::ContainerStoreIterator it = inv.getSlot(MWWorld::InventoryStore::Slot_Boots);
|
|
|
|
if (it != inv.end())
|
|
|
|
bootWeight = it->getClass().getWeight(*it);
|
|
|
|
}
|
|
|
|
sneakTerm = fSneakSkillMult * sneak + 0.2 * agility + 0.1 * luck + bootWeight * fSneakBootMult;
|
|
|
|
}
|
|
|
|
|
|
|
|
static float fSneakDistBase = store.find("fSneakDistanceBase")->getFloat();
|
|
|
|
static float fSneakDistMult = store.find("fSneakDistanceMultiplier")->getFloat();
|
|
|
|
|
|
|
|
Ogre::Vector3 pos1 (ptr.getRefData().getPosition().pos);
|
|
|
|
Ogre::Vector3 pos2 (observer.getRefData().getPosition().pos);
|
|
|
|
float distTerm = fSneakDistBase + fSneakDistMult * pos1.distance(pos2);
|
|
|
|
|
|
|
|
float chameleon = stats.getMagicEffects().get(ESM::MagicEffect::Chameleon).mMagnitude;
|
|
|
|
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();
|
|
|
|
float obsBlind = observerStats.getMagicEffects().get(ESM::MagicEffect::Blind).mMagnitude;
|
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
|
|
|
|
|
|
|
float obsTerm = obsSneak + 0.2 * obsAgility + 0.1 * obsLuck - obsBlind;
|
|
|
|
|
|
|
|
// is ptr behind the observer?
|
|
|
|
static float fSneakNoViewMult = store.find("fSneakNoViewMult")->getFloat();
|
|
|
|
static float fSneakViewMult = store.find("fSneakViewMult")->getFloat();
|
|
|
|
float y = 0;
|
|
|
|
Ogre::Vector3 vec = pos1 - pos2;
|
|
|
|
Ogre::Radian angle = observer.getRefData().getBaseNode()->getOrientation().yAxis().angleBetween(vec);
|
|
|
|
if (angle < Ogre::Degree(90))
|
|
|
|
y = obsTerm * observerStats.getFatigueTerm() * fSneakNoViewMult;
|
|
|
|
else
|
|
|
|
y = obsTerm * observerStats.getFatigueTerm() * fSneakViewMult;
|
|
|
|
|
|
|
|
float target = x - y;
|
|
|
|
int roll = std::rand()/ (static_cast<double> (RAND_MAX) + 1) * 100; // [0, 99]
|
|
|
|
|
|
|
|
return (roll >= target);
|
|
|
|
}
|
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)
|
|
|
|
{
|
|
|
|
ptr.getClass().getCreatureStats(ptr).getAiSequence().stack(MWMechanics::AiCombat(target), ptr);
|
|
|
|
if (target == MWBase::Environment::get().getWorld()->getPlayerPtr())
|
2014-06-14 19:36:57 +00:00
|
|
|
{
|
2014-05-05 22:13:31 +00:00
|
|
|
ptr.getClass().getCreatureStats(ptr).setHostile(true);
|
2014-05-28 11:59:31 +00:00
|
|
|
|
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"))
|
|
|
|
{
|
|
|
|
for (Actors::PtrControllerMap::const_iterator iter = mActors.begin(); iter != mActors.end(); ++iter)
|
|
|
|
{
|
|
|
|
if (iter->first.getClass().isClass(iter->first, "Guard"))
|
|
|
|
{
|
|
|
|
MWMechanics::AiSequence& aiSeq = iter->first.getClass().getCreatureStats(iter->first).getAiSequence();
|
|
|
|
if (aiSeq.getActivePackage()->getTypeId() == MWMechanics::AiPackage::TypeIdPursue)
|
|
|
|
{
|
|
|
|
aiSeq.stopPursuit();
|
|
|
|
aiSeq.stack(MWMechanics::AiCombat(target), ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-28 11:59:31 +00:00
|
|
|
// Must be done after the target is set up, so that CreatureTargetted dialogue filter works properly
|
|
|
|
if (ptr.getClass().isNpc())
|
|
|
|
MWBase::Environment::get().getDialogueManager()->say(ptr, "attack");
|
2014-05-05 22:13:31 +00:00
|
|
|
}
|
|
|
|
|
2014-01-20 12:00:43 +00:00
|
|
|
void MechanicsManager::getObjectsInRange(const Ogre::Vector3 &position, float radius, std::vector<MWWorld::Ptr> &objects)
|
|
|
|
{
|
|
|
|
mActors.getObjectsInRange(position, radius, objects);
|
|
|
|
mObjects.getObjectsInRange(position, radius, objects);
|
|
|
|
}
|
2014-01-28 11:33:31 +00:00
|
|
|
|
2014-04-25 02:41:05 +00:00
|
|
|
void MechanicsManager::getActorsInRange(const Ogre::Vector3 &position, float radius, std::vector<MWWorld::Ptr> &objects)
|
2014-01-12 13:02:15 +00:00
|
|
|
{
|
2014-04-25 02:41:05 +00:00
|
|
|
mActors.getObjectsInRange(position, radius, objects);
|
2014-01-12 13:02:15 +00:00
|
|
|
}
|
2014-04-25 00:40:17 +00:00
|
|
|
|
2014-01-12 13:02:15 +00:00
|
|
|
std::list<MWWorld::Ptr> MechanicsManager::getActorsFollowing(const MWWorld::Ptr& actor)
|
|
|
|
{
|
|
|
|
return mActors.getActorsFollowing(actor);
|
|
|
|
}
|
2014-04-25 02:47:45 +00:00
|
|
|
|
|
|
|
std::list<MWWorld::Ptr> MechanicsManager::getActorsFighting(const MWWorld::Ptr& actor) {
|
|
|
|
return mActors.getActorsFighting(actor);
|
|
|
|
}
|
2014-06-12 23:24:58 +00:00
|
|
|
|
|
|
|
int MechanicsManager::countSavedGameRecords() const
|
|
|
|
{
|
|
|
|
return 1; // Death counter
|
|
|
|
}
|
|
|
|
|
|
|
|
void MechanicsManager::write(ESM::ESMWriter &writer, Loading::Listener &listener) const
|
|
|
|
{
|
|
|
|
mActors.write(writer, listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MechanicsManager::readRecord(ESM::ESMReader &reader, int32_t type)
|
|
|
|
{
|
|
|
|
mActors.readRecord(reader, type);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MechanicsManager::clear()
|
|
|
|
{
|
|
|
|
mActors.clear();
|
|
|
|
}
|
2010-07-26 09:15:38 +00:00
|
|
|
}
|