mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-25 06:35:30 +00:00
408 lines
14 KiB
C++
408 lines
14 KiB
C++
|
|
#include "mechanicsmanager.hpp"
|
|
|
|
#include <components/esm_store/store.hpp>
|
|
|
|
#include "../mwgui/window_manager.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
|
#include "../mwworld/environment.hpp"
|
|
#include "../mwworld/world.hpp"
|
|
#include "../mwworld/player.hpp"
|
|
|
|
namespace MWMechanics
|
|
{
|
|
void MechanicsManager::buildPlayer()
|
|
{
|
|
MWWorld::Ptr ptr = mEnvironment.mWorld->getPlayer().getPlayer();
|
|
|
|
MWMechanics::CreatureStats& creatureStats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
|
MWMechanics::NpcStats& npcStats = MWWorld::Class::get (ptr).getNpcStats (ptr);
|
|
|
|
const ESM::NPC *player = ptr.get<ESM::NPC>()->base;
|
|
|
|
// reset
|
|
creatureStats.mLevel = player->npdt52.level;
|
|
creatureStats.mAbilities.clear();
|
|
creatureStats.mMagicEffects = MagicEffects();
|
|
|
|
for (int i=0; i<27; ++i)
|
|
npcStats.mSkill[i].setBase (player->npdt52.skills[i]);
|
|
|
|
// race
|
|
if (mRaceSelected)
|
|
{
|
|
const ESM::Race *race =
|
|
mEnvironment.mWorld->getStore().races.find (
|
|
mEnvironment.mWorld->getPlayer().getRace());
|
|
|
|
bool male = mEnvironment.mWorld->getPlayer().isMale();
|
|
|
|
for (int i=0; i<8; ++i)
|
|
{
|
|
const ESM::Race::MaleFemale *attribute = 0;
|
|
switch (i)
|
|
{
|
|
case 0: attribute = &race->data.strength; break;
|
|
case 1: attribute = &race->data.intelligence; break;
|
|
case 2: attribute = &race->data.willpower; break;
|
|
case 3: attribute = &race->data.agility; break;
|
|
case 4: attribute = &race->data.speed; break;
|
|
case 5: attribute = &race->data.endurance; break;
|
|
case 6: attribute = &race->data.personality; break;
|
|
case 7: attribute = &race->data.luck; break;
|
|
}
|
|
|
|
creatureStats.mAttributes[i].setBase (
|
|
static_cast<int> (male ? attribute->male : attribute->female));
|
|
}
|
|
|
|
for (int i=0; i<7; ++i)
|
|
{
|
|
int index = race->data.bonus[i].skill;
|
|
|
|
if (index>=0 && index<27)
|
|
{
|
|
npcStats.mSkill[index].setBase (
|
|
npcStats.mSkill[index].getBase() + race->data.bonus[i].bonus);
|
|
}
|
|
}
|
|
|
|
for (std::vector<std::string>::const_iterator iter (race->powers.list.begin());
|
|
iter!=race->powers.list.end(); ++iter)
|
|
{
|
|
insertSpell (*iter, ptr);
|
|
}
|
|
}
|
|
|
|
// birthsign
|
|
if (!mEnvironment.mWorld->getPlayer().getBirthsign().empty())
|
|
{
|
|
const ESM::BirthSign *sign =
|
|
mEnvironment.mWorld->getStore().birthSigns.find (
|
|
mEnvironment.mWorld->getPlayer().getBirthsign());
|
|
|
|
for (std::vector<std::string>::const_iterator iter (sign->powers.list.begin());
|
|
iter!=sign->powers.list.end(); ++iter)
|
|
{
|
|
insertSpell (*iter, ptr);
|
|
}
|
|
}
|
|
|
|
// class
|
|
if (mClassSelected)
|
|
{
|
|
const ESM::Class& class_ = mEnvironment.mWorld->getPlayer().getClass();
|
|
|
|
for (int i=0; i<2; ++i)
|
|
{
|
|
int attribute = class_.data.attribute[i];
|
|
if (attribute>=0 && attribute<8)
|
|
{
|
|
creatureStats.mAttributes[attribute].setBase (
|
|
creatureStats.mAttributes[attribute].getBase() + 10);
|
|
}
|
|
}
|
|
|
|
for (int i=0; i<2; ++i)
|
|
{
|
|
int bonus = i==0 ? 10 : 25;
|
|
|
|
for (int i2=0; i2<5; ++i2)
|
|
{
|
|
int index = class_.data.skills[i2][i];
|
|
|
|
if (index>=0 && index<27)
|
|
{
|
|
npcStats.mSkill[index].setBase (
|
|
npcStats.mSkill[index].getBase() + bonus);
|
|
}
|
|
}
|
|
}
|
|
|
|
typedef ESMS::IndexListT<ESM::Skill>::MapType ContainerType;
|
|
const ContainerType& skills = mEnvironment.mWorld->getStore().skills.list;
|
|
|
|
for (ContainerType::const_iterator iter (skills.begin()); iter!=skills.end(); ++iter)
|
|
{
|
|
if (iter->second.data.specialization==class_.data.specialization)
|
|
{
|
|
int index = iter->first;
|
|
|
|
if (index>=0 && index<27)
|
|
{
|
|
npcStats.mSkill[index].setBase (
|
|
npcStats.mSkill[index].getBase() + 5);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// magic effects
|
|
adjustMagicEffects (ptr);
|
|
|
|
// calculate dynamic stats
|
|
int strength = creatureStats.mAttributes[0].getBase();
|
|
int intelligence = creatureStats.mAttributes[1].getBase();
|
|
int willpower = creatureStats.mAttributes[2].getBase();
|
|
int agility = creatureStats.mAttributes[3].getBase();
|
|
int endurance = creatureStats.mAttributes[5].getBase();
|
|
|
|
double magickaFactor = creatureStats.mMagicEffects.get (EffectKey (84)).mMagnitude*0.1 + 0.5;
|
|
|
|
creatureStats.mDynamic[0].setBase (static_cast<int> (0.5 * (strength + endurance)));
|
|
creatureStats.mDynamic[1].setBase (static_cast<int> (intelligence +
|
|
magickaFactor * intelligence));
|
|
creatureStats.mDynamic[2].setBase (strength+willpower+agility+endurance);
|
|
|
|
for (int i=0; i<3; ++i)
|
|
creatureStats.mDynamic[i].setCurrent (creatureStats.mDynamic[i].getModified());
|
|
}
|
|
|
|
void MechanicsManager::insertSpell (const std::string& id, MWWorld::Ptr& creature)
|
|
{
|
|
MWMechanics::CreatureStats& creatureStats =
|
|
MWWorld::Class::get (creature).getCreatureStats (creature);
|
|
|
|
const ESM::Spell *spell = mEnvironment.mWorld->getStore().spells.find (id);
|
|
|
|
switch (spell->data.type)
|
|
{
|
|
case ESM::Spell::ST_Ability:
|
|
|
|
if (creatureStats.mAbilities.find (id)==creatureStats.mAbilities.end())
|
|
{
|
|
creatureStats.mAbilities.insert (id);
|
|
}
|
|
|
|
break;
|
|
|
|
// TODO ST_SPELL, ST_Blight, ST_Disease, ST_Curse, ST_Power
|
|
|
|
default:
|
|
|
|
std::cout
|
|
<< "adding unsupported spell type (" << spell->data.type
|
|
<< ") to creature: " << id << std::endl;
|
|
}
|
|
}
|
|
|
|
void MechanicsManager::adjustMagicEffects (MWWorld::Ptr& creature)
|
|
{
|
|
MWMechanics::CreatureStats& creatureStats =
|
|
MWWorld::Class::get (creature).getCreatureStats (creature);
|
|
|
|
MagicEffects now;
|
|
|
|
for (std::set<std::string>::const_iterator iter (creatureStats.mAbilities.begin());
|
|
iter!=creatureStats.mAbilities.end(); ++iter)
|
|
{
|
|
const ESM::Spell *spell = mEnvironment.mWorld->getStore().spells.find (*iter);
|
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator iter = spell->effects.list.begin();
|
|
iter!=spell->effects.list.end(); ++iter)
|
|
{
|
|
if (iter->range==0) // self
|
|
{
|
|
EffectParam param;
|
|
param.mMagnitude = iter->magnMax; // TODO calculate magnitude
|
|
now.add (EffectKey (*iter), param);
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO add effects from other spell types, active spells and equipment
|
|
|
|
MagicEffects diff = MagicEffects::diff (creatureStats.mMagicEffects, now);
|
|
|
|
creatureStats.mMagicEffects = now;
|
|
|
|
// TODO apply diff to other stats
|
|
}
|
|
|
|
MechanicsManager::MechanicsManager (MWWorld::Environment& environment)
|
|
: mEnvironment (environment), mUpdatePlayer (true), mClassSelected (false),
|
|
mRaceSelected (false)
|
|
{
|
|
buildPlayer();
|
|
}
|
|
|
|
void MechanicsManager::addActor (const MWWorld::Ptr& ptr)
|
|
{
|
|
mActors.insert (ptr);
|
|
}
|
|
|
|
void MechanicsManager::removeActor (const MWWorld::Ptr& ptr)
|
|
{
|
|
if (ptr==mWatched)
|
|
mWatched = MWWorld::Ptr();
|
|
|
|
mActors.erase (ptr);
|
|
}
|
|
|
|
void MechanicsManager::dropActors (const MWWorld::Ptr::CellStore *cellStore)
|
|
{
|
|
if (!mWatched.isEmpty() && mWatched.getCell()==cellStore)
|
|
mWatched = MWWorld::Ptr();
|
|
|
|
std::set<MWWorld::Ptr>::iterator iter = mActors.begin();
|
|
|
|
while (iter!=mActors.end())
|
|
if (iter->getCell()==cellStore)
|
|
{
|
|
mActors.erase (iter++);
|
|
}
|
|
else
|
|
++iter;
|
|
}
|
|
|
|
void MechanicsManager::watchActor (const MWWorld::Ptr& ptr)
|
|
{
|
|
mWatched = ptr;
|
|
}
|
|
|
|
void MechanicsManager::update (std::vector<std::pair<std::string, Ogre::Vector3> >& movement)
|
|
{
|
|
if (!mWatched.isEmpty())
|
|
{
|
|
MWMechanics::CreatureStats& stats =
|
|
MWWorld::Class::get (mWatched).getCreatureStats (mWatched);
|
|
|
|
MWMechanics::NpcStats& npcStats =
|
|
MWWorld::Class::get (mWatched).getNpcStats (mWatched);
|
|
|
|
static const char *attributeNames[8] =
|
|
{
|
|
"AttribVal1", "AttribVal2", "AttribVal3", "AttribVal4", "AttribVal5",
|
|
"AttribVal6", "AttribVal7", "AttribVal8"
|
|
};
|
|
|
|
static const char *dynamicNames[3] =
|
|
{
|
|
"HBar", "MBar", "FBar"
|
|
};
|
|
|
|
static const char *skillNames[27] =
|
|
{
|
|
"SkillBlock", "SkillArmorer", "SkillMediumArmor", "SkillHeavyArmor",
|
|
"SkillBluntWeapon", "SkillLongBlade", "SkillAxe", "SkillSpear",
|
|
"SkillAthletics", "SkillEnchant", "SkillDestruction", "SkillAlteration",
|
|
"SkillIllusion", "SkillConjuration", "SkillMysticism", "SkillRestoration",
|
|
"SkillAlchemy", "SkillUnarmored", "SkillSecurity", "SkillSneak",
|
|
"SkillAcrobatics", "SkillLightArmor", "SkillShortBlade", "SkillMarksman",
|
|
"SkillMercantile", "SkillSpeechcraft", "SkillHandToHand",
|
|
};
|
|
|
|
for (int i=0; i<8; ++i)
|
|
{
|
|
if (stats.mAttributes[i]!=mWatchedCreature.mAttributes[i])
|
|
{
|
|
mWatchedCreature.mAttributes[i] = stats.mAttributes[i];
|
|
|
|
mEnvironment.mWindowManager->setValue (attributeNames[i], stats.mAttributes[i]);
|
|
}
|
|
}
|
|
|
|
for (int i=0; i<3; ++i)
|
|
{
|
|
if (stats.mDynamic[i]!=mWatchedCreature.mDynamic[i])
|
|
{
|
|
mWatchedCreature.mDynamic[i] = stats.mDynamic[i];
|
|
|
|
mEnvironment.mWindowManager->setValue (dynamicNames[i], stats.mDynamic[i]);
|
|
}
|
|
}
|
|
|
|
bool update = false;
|
|
|
|
for (int i=0; i<27; ++i)
|
|
{
|
|
if (npcStats.mSkill[i]!=mWatchedNpc.mSkill[i])
|
|
{
|
|
update = true;
|
|
|
|
mWatchedNpc.mSkill[i] = npcStats.mSkill[i];
|
|
|
|
mEnvironment.mWindowManager->setValue (skillNames[i], npcStats.mSkill[i]);
|
|
|
|
}
|
|
}
|
|
|
|
if (update)
|
|
mEnvironment.mWindowManager->updateSkillArea();
|
|
|
|
mEnvironment.mWindowManager->setValue ("level", stats.mLevel);
|
|
}
|
|
|
|
if (mUpdatePlayer)
|
|
{
|
|
// basic player profile; should not change anymore after the creation phase is finished.
|
|
mEnvironment.mWindowManager->setValue ("name", mEnvironment.mWorld->getPlayer().getName());
|
|
mEnvironment.mWindowManager->setValue ("race",
|
|
mEnvironment.mWorld->getStore().races.find (mEnvironment.mWorld->getPlayer().
|
|
getRace())->name);
|
|
mEnvironment.mWindowManager->setValue ("class",
|
|
mEnvironment.mWorld->getPlayer().getClass().name);
|
|
mUpdatePlayer = false;
|
|
|
|
MWGui::WindowManager::SkillList majorSkills (5);
|
|
MWGui::WindowManager::SkillList minorSkills (5);
|
|
|
|
for (int i=0; i<5; ++i)
|
|
{
|
|
minorSkills[i] = mEnvironment.mWorld->getPlayer().getClass().data.skills[i][0];
|
|
majorSkills[i] = mEnvironment.mWorld->getPlayer().getClass().data.skills[i][1];
|
|
}
|
|
|
|
mEnvironment.mWindowManager->configureSkills (majorSkills, minorSkills);
|
|
}
|
|
|
|
for (std::set<MWWorld::Ptr>::iterator iter (mActors.begin()); iter!=mActors.end();
|
|
++iter)
|
|
{
|
|
Ogre::Vector3 vector = MWWorld::Class::get (*iter).getMovementVector (*iter);
|
|
|
|
if (vector!=Ogre::Vector3::ZERO)
|
|
movement.push_back (std::make_pair (iter->getRefData().getHandle(), vector));
|
|
}
|
|
}
|
|
|
|
void MechanicsManager::setPlayerName (const std::string& name)
|
|
{
|
|
mEnvironment.mWorld->getPlayer().setName (name);
|
|
mUpdatePlayer = true;
|
|
}
|
|
|
|
void MechanicsManager::setPlayerRace (const std::string& race, bool male)
|
|
{
|
|
mEnvironment.mWorld->getPlayer().setGender (male);
|
|
mEnvironment.mWorld->getPlayer().setRace (race);
|
|
mRaceSelected = true;
|
|
buildPlayer();
|
|
mUpdatePlayer = true;
|
|
}
|
|
|
|
void MechanicsManager::setPlayerBirthsign (const std::string& id)
|
|
{
|
|
mEnvironment.mWorld->getPlayer().setBirthsign (id);
|
|
buildPlayer();
|
|
mUpdatePlayer = true;
|
|
}
|
|
|
|
void MechanicsManager::setPlayerClass (const std::string& id)
|
|
{
|
|
mEnvironment.mWorld->getPlayer().setClass (*mEnvironment.mWorld->getStore().classes.find (id));
|
|
mClassSelected = true;
|
|
buildPlayer();
|
|
mUpdatePlayer = true;
|
|
}
|
|
|
|
void MechanicsManager::setPlayerClass (const ESM::Class& class_)
|
|
{
|
|
mEnvironment.mWorld->getPlayer().setClass (class_);
|
|
mClassSelected = true;
|
|
buildPlayer();
|
|
mUpdatePlayer = true;
|
|
}
|
|
}
|