#include "mechanicsmanager.hpp" #include #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()->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 (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::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::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::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 (0.5 * (strength + endurance))); creatureStats.mDynamic[1].setBase (static_cast (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::const_iterator iter (creatureStats.mAbilities.begin()); iter!=creatureStats.mAbilities.end(); ++iter) { const ESM::Spell *spell = mEnvironment.mWorld->getStore().spells.find (*iter); for (std::vector::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::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 >& 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::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; } }