1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 06:35:30 +00:00

495 lines
14 KiB
C++
Raw Normal View History

2012-09-29 10:02:46 +02:00
#include "alchemy.hpp"
2012-10-18 15:33:27 +02:00
#include <cassert>
#include <cstdlib>
#include <algorithm>
#include <stdexcept>
2012-10-28 14:07:36 +01:00
#include <map>
#include <components/esm/loadskil.hpp>
#include <components/esm/loadappa.hpp>
#include <components/esm/loadgmst.hpp>
#include <components/esm/loadmgef.hpp>
#include "../mwbase/environment.hpp"
#include "../mwbase/world.hpp"
#include "../mwworld/esmstore.hpp"
#include "../mwworld/containerstore.hpp"
#include "../mwworld/class.hpp"
#include "../mwworld/cellstore.hpp"
2012-10-18 15:33:27 +02:00
#include "../mwworld/manualref.hpp"
#include "magiceffects.hpp"
#include "creaturestats.hpp"
#include "npcstats.hpp"
2014-09-26 17:12:48 +02:00
MWMechanics::Alchemy::Alchemy()
: mValue(0)
{
}
std::set<MWMechanics::EffectKey> MWMechanics::Alchemy::listEffects() const
{
2012-10-28 14:07:36 +01:00
std::map<EffectKey, int> effects;
2013-11-10 21:45:27 +01:00
for (TIngredientsIterator iter (mIngredients.begin()); iter!=mIngredients.end(); ++iter)
{
if (!iter->isEmpty())
{
const MWWorld::LiveCellRef<ESM::Ingredient> *ingredient = iter->get<ESM::Ingredient>();
2013-11-10 21:45:27 +01:00
for (int i=0; i<4; ++i)
2012-11-05 16:07:59 +04:00
if (ingredient->mBase->mData.mEffectID[i]!=-1)
2012-10-28 14:07:36 +01:00
{
EffectKey key (
2012-11-05 16:07:59 +04:00
ingredient->mBase->mData.mEffectID[i], ingredient->mBase->mData.mSkills[i]!=-1 ?
ingredient->mBase->mData.mSkills[i] : ingredient->mBase->mData.mAttributes[i]);
2012-10-28 14:07:36 +01:00
++effects[key];
}
}
}
2013-11-10 21:45:27 +01:00
2012-10-28 14:07:36 +01:00
std::set<EffectKey> effects2;
2013-11-10 21:45:27 +01:00
2012-10-28 14:07:36 +01:00
for (std::map<EffectKey, int>::const_iterator iter (effects.begin()); iter!=effects.end(); ++iter)
if (iter->second>1)
effects2.insert (iter->first);
2013-11-10 21:45:27 +01:00
2012-10-28 14:07:36 +01:00
return effects2;
}
void MWMechanics::Alchemy::applyTools (int flags, float& value) const
{
bool magnitude = !(flags & ESM::MagicEffect::NoMagnitude);
bool duration = !(flags & ESM::MagicEffect::NoDuration);
bool negative = flags & (ESM::MagicEffect::Harmful);
int tool = negative ? ESM::Apparatus::Retort : ESM::Apparatus::Albemic;
int setup = 0;
2013-11-10 21:45:27 +01:00
if (!mTools[tool].isEmpty() && !mTools[ESM::Apparatus::Calcinator].isEmpty())
setup = 1;
else if (!mTools[tool].isEmpty())
setup = 2;
else if (!mTools[ESM::Apparatus::Calcinator].isEmpty())
setup = 3;
else
return;
2012-11-05 16:07:59 +04:00
float toolQuality = setup==1 || setup==2 ? mTools[tool].get<ESM::Apparatus>()->mBase->mData.mQuality : 0;
float calcinatorQuality = setup==1 || setup==3 ?
2012-11-05 16:07:59 +04:00
mTools[ESM::Apparatus::Calcinator].get<ESM::Apparatus>()->mBase->mData.mQuality : 0;
float quality = 1;
2013-11-10 21:45:27 +01:00
switch (setup)
{
case 1:
2013-11-10 21:45:27 +01:00
quality = negative ? 2 * toolQuality + 3 * calcinatorQuality :
(magnitude && duration ?
2 * toolQuality + calcinatorQuality : 2/3.0 * (toolQuality + calcinatorQuality) + 0.5);
break;
2013-11-10 21:45:27 +01:00
case 2:
2013-11-10 21:45:27 +01:00
quality = negative ? 1+toolQuality : (magnitude && duration ? toolQuality : toolQuality + 0.5);
break;
2013-11-10 21:45:27 +01:00
case 3:
2013-11-10 21:45:27 +01:00
quality = magnitude && duration ? calcinatorQuality : calcinatorQuality + 0.5;
break;
}
if (setup==3 || !negative)
{
value += quality;
}
else
{
if (quality==0)
2013-11-10 21:45:27 +01:00
throw std::runtime_error ("invalid derived alchemy apparatus quality");
value /= quality;
}
}
void MWMechanics::Alchemy::updateEffects()
{
mEffects.clear();
2012-10-18 15:33:27 +02:00
mValue = 0;
if (countIngredients()<2 || mAlchemist.isEmpty() || mTools[ESM::Apparatus::MortarPestle].isEmpty())
return;
// find effects
std::set<EffectKey> effects (listEffects());
// general alchemy factor
2014-10-02 13:54:56 +02:00
float x = getAlchemyFactor();
2012-11-05 16:07:59 +04:00
x *= mTools[ESM::Apparatus::MortarPestle].get<ESM::Apparatus>()->mBase->mData.mQuality;
x *= MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find ("fPotionStrengthMult")->getFloat();
2012-10-18 15:33:27 +02:00
// value
mValue = static_cast<int> (
x * MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find ("iAlchemyMod")->getFloat());
2012-10-18 15:33:27 +02:00
// build quantified effect list
for (std::set<EffectKey>::const_iterator iter (effects.begin()); iter!=effects.end(); ++iter)
{
const ESM::MagicEffect *magicEffect =
2013-11-10 21:45:27 +01:00
MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().find (iter->mId);
if (magicEffect->mData.mBaseCost<=0)
{
std::ostringstream os;
os << "invalid base cost for magic effect " << iter->mId;
throw std::runtime_error (os.str());
}
2013-11-10 21:45:27 +01:00
float fPotionT1MagMul =
MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find ("fPotionT1MagMult")->getFloat();
if (fPotionT1MagMul<=0)
throw std::runtime_error ("invalid gmst: fPotionT1MagMul");
2013-11-10 21:45:27 +01:00
float fPotionT1DurMult =
MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find ("fPotionT1DurMult")->getFloat();
if (fPotionT1DurMult<=0)
throw std::runtime_error ("invalid gmst: fPotionT1DurMult");
float magnitude = magicEffect->mData.mFlags & ESM::MagicEffect::NoMagnitude ?
1 : (x / fPotionT1MagMul) / magicEffect->mData.mBaseCost;
float duration = magicEffect->mData.mFlags & ESM::MagicEffect::NoDuration ?
1 : (x / fPotionT1DurMult) / magicEffect->mData.mBaseCost;
if (!(magicEffect->mData.mFlags & ESM::MagicEffect::NoMagnitude))
applyTools (magicEffect->mData.mFlags, magnitude);
if (!(magicEffect->mData.mFlags & ESM::MagicEffect::NoDuration))
applyTools (magicEffect->mData.mFlags, duration);
2013-11-10 21:45:27 +01:00
duration = static_cast<int> (duration+0.5);
magnitude = static_cast<int> (magnitude+0.5);
if (magnitude>0 && duration>0)
{
ESM::ENAMstruct effect;
effect.mEffectID = iter->mId;
2013-11-10 21:45:27 +01:00
effect.mAttribute = -1;
effect.mSkill = -1;
if (magicEffect->mData.mFlags & ESM::MagicEffect::TargetSkill)
effect.mSkill = iter->mArg;
else if (magicEffect->mData.mFlags & ESM::MagicEffect::TargetAttribute)
effect.mAttribute = iter->mArg;
2013-11-10 21:45:27 +01:00
effect.mRange = 0;
2013-11-10 21:45:27 +01:00
effect.mArea = 0;
effect.mDuration = duration;
effect.mMagnMin = effect.mMagnMax = magnitude;
mEffects.push_back (effect);
2013-11-10 21:45:27 +01:00
}
}
}
const ESM::Potion *MWMechanics::Alchemy::getRecord(const ESM::Potion& toFind) const
{
const MWWorld::Store<ESM::Potion> &potions =
MWBase::Environment::get().getWorld()->getStore().get<ESM::Potion>();
MWWorld::Store<ESM::Potion>::iterator iter = potions.begin();
for (; iter != potions.end(); ++iter)
2012-10-18 15:33:27 +02:00
{
if (iter->mEffects.mList.size() != mEffects.size())
2012-10-18 15:33:27 +02:00
continue;
2013-11-10 21:45:27 +01:00
if (iter->mName != toFind.mName
|| iter->mScript != toFind.mScript
|| iter->mData.mWeight != toFind.mData.mWeight
|| iter->mData.mValue != toFind.mData.mValue
|| iter->mData.mAutoCalc != toFind.mData.mAutoCalc)
continue;
// Don't choose an ID that came from the content files, would have unintended side effects
// where alchemy can be used to produce quest-relevant items
if (!potions.isDynamic(iter->mId))
continue;
2013-11-10 21:45:27 +01:00
bool mismatch = false;
2012-12-02 21:18:59 +04:00
for (int i=0; i<static_cast<int> (iter->mEffects.mList.size()); ++i)
2012-10-18 15:33:27 +02:00
{
const ESM::ENAMstruct& first = iter->mEffects.mList[i];
2012-10-18 15:33:27 +02:00
const ESM::ENAMstruct& second = mEffects[i];
2013-11-10 21:45:27 +01:00
2012-10-18 15:33:27 +02:00
if (first.mEffectID!=second.mEffectID ||
first.mArea!=second.mArea ||
first.mRange!=second.mRange ||
first.mSkill!=second.mSkill ||
first.mAttribute!=second.mAttribute ||
first.mMagnMin!=second.mMagnMin ||
first.mMagnMax!=second.mMagnMax ||
first.mDuration!=second.mDuration)
{
mismatch = true;
break;
}
}
2013-11-10 21:45:27 +01:00
2012-10-18 15:33:27 +02:00
if (!mismatch)
return &(*iter);
2012-10-18 15:33:27 +02:00
}
2013-11-10 21:45:27 +01:00
2012-10-18 15:33:27 +02:00
return 0;
}
void MWMechanics::Alchemy::removeIngredients()
{
bool needsUpdate = false;
2013-11-10 21:45:27 +01:00
for (TIngredientsContainer::iterator iter (mIngredients.begin()); iter!=mIngredients.end(); ++iter)
if (!iter->isEmpty())
{
iter->getContainerStore()->remove(*iter, 1, mAlchemist);
if (iter->getRefData().getCount()<1)
{
needsUpdate = true;
*iter = MWWorld::Ptr();
}
}
2013-11-10 21:45:27 +01:00
if (needsUpdate)
updateEffects();
}
void MWMechanics::Alchemy::addPotion (const std::string& name)
{
ESM::Potion newRecord;
2013-11-10 21:45:27 +01:00
newRecord.mData.mWeight = 0;
2013-11-10 21:45:27 +01:00
for (TIngredientsIterator iter (beginIngredients()); iter!=endIngredients(); ++iter)
if (!iter->isEmpty())
newRecord.mData.mWeight += iter->get<ESM::Ingredient>()->mBase->mData.mWeight;
2013-11-10 21:45:27 +01:00
if (countIngredients() > 0)
newRecord.mData.mWeight /= countIngredients();
2013-11-10 21:45:27 +01:00
newRecord.mData.mValue = mValue;
newRecord.mData.mAutoCalc = 0;
2013-11-10 21:45:27 +01:00
newRecord.mName = name;
int index = static_cast<int> (std::rand()/(static_cast<double> (RAND_MAX)+1)*6);
assert (index>=0 && index<6);
2013-11-10 21:45:27 +01:00
static const char *meshes[] = { "standard", "bargain", "cheap", "fresh", "exclusive", "quality" };
2013-11-10 21:45:27 +01:00
newRecord.mModel = "m\\misc_potion_" + std::string (meshes[index]) + "_01.nif";
newRecord.mIcon = "m\\tx_potion_" + std::string (meshes[index]) + "_01.dds";
2013-11-10 21:45:27 +01:00
newRecord.mEffects.mList = mEffects;
2013-11-10 21:45:27 +01:00
const ESM::Potion* record = getRecord(newRecord);
if (!record)
record = MWBase::Environment::get().getWorld()->createRecord (newRecord);
2013-11-10 21:45:27 +01:00
mAlchemist.getClass().getContainerStore (mAlchemist).add (record->mId, 1, mAlchemist);
}
void MWMechanics::Alchemy::increaseSkill()
{
mAlchemist.getClass().skillUsageSucceeded (mAlchemist, ESM::Skill::Alchemy, 0);
}
2014-10-02 13:54:56 +02:00
float MWMechanics::Alchemy::getAlchemyFactor() const
{
const CreatureStats& creatureStats = mAlchemist.getClass().getCreatureStats (mAlchemist);
const NpcStats& npcStats = mAlchemist.getClass().getNpcStats (mAlchemist);
2013-11-10 21:45:27 +01:00
return
(npcStats.getSkill (ESM::Skill::Alchemy).getModified() +
2014-10-02 13:54:56 +02:00
0.1 * creatureStats.getAttribute (ESM::Attribute::Intelligence).getModified()
+ 0.1 * creatureStats.getAttribute (ESM::Attribute::Luck).getModified());
}
int MWMechanics::Alchemy::countIngredients() const
{
int ingredients = 0;
for (TIngredientsIterator iter (beginIngredients()); iter!=endIngredients(); ++iter)
if (!iter->isEmpty())
++ingredients;
return ingredients;
}
2012-09-29 10:02:46 +02:00
void MWMechanics::Alchemy::setAlchemist (const MWWorld::Ptr& npc)
{
mAlchemist = npc;
2013-11-10 21:45:27 +01:00
mIngredients.resize (4);
std::fill (mIngredients.begin(), mIngredients.end(), MWWorld::Ptr());
2013-11-10 21:45:27 +01:00
mTools.resize (4);
2013-11-10 21:45:27 +01:00
std::fill (mTools.begin(), mTools.end(), MWWorld::Ptr());
2013-11-10 21:45:27 +01:00
mEffects.clear();
2013-11-10 21:45:27 +01:00
MWWorld::ContainerStore& store = npc.getClass().getContainerStore (npc);
2013-11-10 21:45:27 +01:00
for (MWWorld::ContainerStoreIterator iter (store.begin (MWWorld::ContainerStore::Type_Apparatus));
iter!=store.end(); ++iter)
2013-11-10 21:45:27 +01:00
{
MWWorld::LiveCellRef<ESM::Apparatus>* ref = iter->get<ESM::Apparatus>();
2013-11-10 21:45:27 +01:00
2012-11-05 16:07:59 +04:00
int type = ref->mBase->mData.mType;
2013-11-10 21:45:27 +01:00
if (type<0 || type>=static_cast<int> (mTools.size()))
throw std::runtime_error ("invalid apparatus type");
2013-11-10 21:45:27 +01:00
if (!mTools[type].isEmpty())
2012-11-05 16:07:59 +04:00
if (ref->mBase->mData.mQuality<=mTools[type].get<ESM::Apparatus>()->mBase->mData.mQuality)
continue;
2013-11-10 21:45:27 +01:00
mTools[type] = *iter;
}
}
MWMechanics::Alchemy::TToolsIterator MWMechanics::Alchemy::beginTools() const
{
return mTools.begin();
}
MWMechanics::Alchemy::TToolsIterator MWMechanics::Alchemy::endTools() const
{
return mTools.end();
2012-09-29 10:02:46 +02:00
}
MWMechanics::Alchemy::TIngredientsIterator MWMechanics::Alchemy::beginIngredients() const
{
return mIngredients.begin();
}
MWMechanics::Alchemy::TIngredientsIterator MWMechanics::Alchemy::endIngredients() const
{
return mIngredients.end();
}
void MWMechanics::Alchemy::clear()
{
mAlchemist = MWWorld::Ptr();
mTools.clear();
mIngredients.clear();
mEffects.clear();
}
int MWMechanics::Alchemy::addIngredient (const MWWorld::Ptr& ingredient)
{
// find a free slot
int slot = -1;
for (int i=0; i<static_cast<int> (mIngredients.size()); ++i)
if (mIngredients[i].isEmpty())
{
slot = i;
break;
2013-11-10 21:45:27 +01:00
}
if (slot==-1)
return -1;
2013-11-10 21:45:27 +01:00
for (TIngredientsIterator iter (mIngredients.begin()); iter!=mIngredients.end(); ++iter)
if (!iter->isEmpty() && Misc::StringUtils::ciEqual(ingredient.getClass().getId(ingredient),
iter->getClass().getId(*iter)))
return -1;
2013-11-10 21:45:27 +01:00
mIngredients[slot] = ingredient;
2013-11-10 21:45:27 +01:00
updateEffects();
2013-11-10 21:45:27 +01:00
return slot;
}
void MWMechanics::Alchemy::removeIngredient (int index)
{
if (index>=0 && index<static_cast<int> (mIngredients.size()))
{
mIngredients[index] = MWWorld::Ptr();
updateEffects();
}
}
MWMechanics::Alchemy::TEffectsIterator MWMechanics::Alchemy::beginEffects() const
{
return mEffects.begin();
}
MWMechanics::Alchemy::TEffectsIterator MWMechanics::Alchemy::endEffects() const
{
return mEffects.end();
}
MWMechanics::Alchemy::Result MWMechanics::Alchemy::create (const std::string& name)
{
if (mTools[ESM::Apparatus::MortarPestle].isEmpty())
return Result_NoMortarAndPestle;
2013-11-10 21:45:27 +01:00
if (countIngredients()<2)
return Result_LessThanTwoIngredients;
if (name.empty())
return Result_NoName;
2013-11-10 21:45:27 +01:00
if (listEffects().empty())
return Result_NoEffects;
2014-10-02 13:54:56 +02:00
if (beginEffects() == endEffects())
{
// all effects were nullified due to insufficient skill
removeIngredients();
return Result_RandomFailure;
}
if (getAlchemyFactor()<std::rand()/static_cast<double> (RAND_MAX)*100)
{
removeIngredients();
return Result_RandomFailure;
}
addPotion (name);
removeIngredients();
2013-11-10 21:45:27 +01:00
increaseSkill();
return Result_Success;
}
2014-10-20 17:28:22 +02:00
std::string MWMechanics::Alchemy::suggestPotionName()
{
std::set<MWMechanics::EffectKey> effects = listEffects();
if (effects.empty())
return "";
int effectId = effects.begin()->mId;
return MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find(
2014-10-20 17:42:51 +02:00
ESM::MagicEffect::effectIdToString(effectId))->getString();
2014-10-20 17:28:22 +02:00
}