1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-22 04:23:50 +00:00
openmw-tes3mp/apps/openmw/mwmechanics/enchanting.cpp

339 lines
11 KiB
C++
Raw Normal View History

2013-03-28 16:41:00 +00:00
#include "enchanting.hpp"
2015-04-22 15:58:55 +00:00
#include <components/misc/rng.hpp>
[General] Implement RecordDynamic packet, part 1 Spell, potion, enchantment, creature, NPC, armor, book, clothing, miscellaneous and weapon record data can now be sent in a RecordDynamic packet. Additionally, the packets include data related to associated magical effects (for spells, potions and enchantments), data related to default inventory contents (for creatures and NPCs) and data related to body parts affected (for armor and clothing). The server now has associated script functions for setting most of the details of the above, with the main exception being individual creature and NPC stats. Records can either be created entirely from scratch or can use an existing record (set via the baseId variable) as a starting point for their values. In the latter case, only the values that are specifically set override the starting values. Creature and NPC records also have an inventoryBaseId that can be used on top of the baseId to base their inventories on another existing record. The client's RecordHelper class has been heavily expanded to allow for the above mentioned functionality. When players create spells, potions and enchantments as part of regular gameplay, they send RecordDynamic packets that provide the server with the complete details of the records that should be created. When they create enchantments, they also provide the server with armor, book, clothing and weapon records corresponding to the items they've enchanted. This functionality added by this packet was originally supposed to be exclusive to the rewrite, but I've gone ahead and tried to provide it for the pre-rewrite in a way that can mostly be reused for the rewrite.
2018-07-30 07:56:26 +00:00
/*
Start of tes3mp addition
Include additional headers for multiplayer purposes
*/
#include <components/openmw-mp/Log.hpp>
#include "../mwmp/Main.hpp"
#include "../mwmp/Networking.hpp"
#include "../mwmp/Worldstate.hpp"
/*
End of tes3mp addition
*/
2013-03-28 16:41:00 +00:00
#include "../mwworld/manualref.hpp"
#include "../mwworld/class.hpp"
#include "../mwworld/containerstore.hpp"
2015-02-09 14:01:49 +00:00
#include "../mwworld/esmstore.hpp"
2016-06-17 14:07:16 +00:00
2013-04-02 18:46:48 +00:00
#include "../mwbase/mechanicsmanager.hpp"
2013-03-30 18:08:42 +00:00
#include "creaturestats.hpp"
#include "npcstats.hpp"
#include "spellcasting.hpp"
2015-08-21 09:12:39 +00:00
#include "actorutil.hpp"
2013-03-30 18:08:42 +00:00
2013-03-28 16:41:00 +00:00
namespace MWMechanics
{
2013-07-31 16:46:32 +00:00
Enchanting::Enchanting()
: mCastStyle(ESM::Enchantment::CastOnce)
, mSelfEnchanting(false)
2013-03-28 16:41:00 +00:00
{}
2017-04-20 11:36:14 +00:00
void Enchanting::setOldItem(const MWWorld::Ptr& oldItem)
2013-03-28 16:41:00 +00:00
{
mOldItemPtr=oldItem;
if(!itemEmpty())
{
mObjectType = mOldItemPtr.getTypeName();
}
else
{
mObjectType="";
}
}
void Enchanting::setNewItemName(const std::string& s)
2013-03-28 16:41:00 +00:00
{
mNewItemName=s;
}
2017-04-20 11:36:14 +00:00
void Enchanting::setEffect(const ESM::EffectList& effectList)
2013-03-28 16:41:00 +00:00
{
mEffectList=effectList;
}
int Enchanting::getCastStyle() const
2013-03-28 16:41:00 +00:00
{
return mCastStyle;
2013-03-28 16:41:00 +00:00
}
2017-04-20 11:36:14 +00:00
void Enchanting::setSoulGem(const MWWorld::Ptr& soulGem)
2013-03-28 16:41:00 +00:00
{
mSoulGemPtr=soulGem;
}
bool Enchanting::create()
2013-03-28 16:41:00 +00:00
{
2015-08-21 09:12:39 +00:00
const MWWorld::Ptr& player = getPlayer();
MWWorld::ContainerStore& store = player.getClass().getContainerStore(player);
ESM::Enchantment enchantment;
enchantment.mData.mCharge = getGemCharge();
enchantment.mData.mAutocalc = 0;
enchantment.mData.mType = mCastStyle;
enchantment.mData.mCost = getBaseCastCost();
store.remove(mSoulGemPtr, 1, player);
2013-04-08 15:53:41 +00:00
//Exception for Azura Star, new one will be added after enchanting
2014-01-14 08:47:31 +00:00
if(Misc::StringUtils::ciEqual(mSoulGemPtr.get<ESM::Miscellaneous>()->mBase->mId, "Misc_SoulGem_Azura"))
store.add("Misc_SoulGem_Azura", 1, player);
2013-03-29 11:00:09 +00:00
2013-03-30 18:08:42 +00:00
if(mSelfEnchanting)
{
2015-04-22 15:58:55 +00:00
if(getEnchantChance() <= (Misc::Rng::roll0to99()))
return false;
2013-03-30 18:08:42 +00:00
mEnchanter.getClass().skillUsageSucceeded (mEnchanter, ESM::Skill::Enchant, 2);
2013-03-30 18:08:42 +00:00
}
if(mCastStyle==ESM::Enchantment::ConstantEffect)
2013-03-28 16:41:00 +00:00
{
enchantment.mData.mCharge=0;
2013-03-28 16:41:00 +00:00
}
enchantment.mEffects = mEffectList;
2013-03-31 21:18:23 +00:00
// Apply the enchantment
const ESM::Enchantment *enchantmentPtr = MWBase::Environment::get().getWorld()->createRecord (enchantment);
2013-03-31 21:18:23 +00:00
[General] Implement RecordDynamic packet, part 1 Spell, potion, enchantment, creature, NPC, armor, book, clothing, miscellaneous and weapon record data can now be sent in a RecordDynamic packet. Additionally, the packets include data related to associated magical effects (for spells, potions and enchantments), data related to default inventory contents (for creatures and NPCs) and data related to body parts affected (for armor and clothing). The server now has associated script functions for setting most of the details of the above, with the main exception being individual creature and NPC stats. Records can either be created entirely from scratch or can use an existing record (set via the baseId variable) as a starting point for their values. In the latter case, only the values that are specifically set override the starting values. Creature and NPC records also have an inventoryBaseId that can be used on top of the baseId to base their inventories on another existing record. The client's RecordHelper class has been heavily expanded to allow for the above mentioned functionality. When players create spells, potions and enchantments as part of regular gameplay, they send RecordDynamic packets that provide the server with the complete details of the records that should be created. When they create enchantments, they also provide the server with armor, book, clothing and weapon records corresponding to the items they've enchanted. This functionality added by this packet was originally supposed to be exclusive to the rewrite, but I've gone ahead and tried to provide it for the pre-rewrite in a way that can mostly be reused for the rewrite.
2018-07-30 07:56:26 +00:00
/*
Start of tes3mp change (major)
Send the enchantment's record to the server
Don't add the new item to the player's inventory and instead expect the server to
add it
The applyEnchantment() method is where the record of the newly enchanted will be sent
to the server, causing the server to send back the player's inventory with the new item
included
*/
mwmp::Main::get().getNetworking()->getWorldstate()->sendEnchantmentRecord(enchantmentPtr);
store.remove(mOldItemPtr, 1, player);
2013-04-02 18:46:48 +00:00
if(!mSelfEnchanting)
2013-04-03 16:02:30 +00:00
payForEnchantment();
2013-03-28 16:41:00 +00:00
[General] Implement RecordDynamic packet, part 1 Spell, potion, enchantment, creature, NPC, armor, book, clothing, miscellaneous and weapon record data can now be sent in a RecordDynamic packet. Additionally, the packets include data related to associated magical effects (for spells, potions and enchantments), data related to default inventory contents (for creatures and NPCs) and data related to body parts affected (for armor and clothing). The server now has associated script functions for setting most of the details of the above, with the main exception being individual creature and NPC stats. Records can either be created entirely from scratch or can use an existing record (set via the baseId variable) as a starting point for their values. In the latter case, only the values that are specifically set override the starting values. Creature and NPC records also have an inventoryBaseId that can be used on top of the baseId to base their inventories on another existing record. The client's RecordHelper class has been heavily expanded to allow for the above mentioned functionality. When players create spells, potions and enchantments as part of regular gameplay, they send RecordDynamic packets that provide the server with the complete details of the records that should be created. When they create enchantments, they also provide the server with armor, book, clothing and weapon records corresponding to the items they've enchanted. This functionality added by this packet was originally supposed to be exclusive to the rewrite, but I've gone ahead and tried to provide it for the pre-rewrite in a way that can mostly be reused for the rewrite.
2018-07-30 07:56:26 +00:00
std::string newItemId = mOldItemPtr.getClass().applyEnchantment(mOldItemPtr, enchantmentPtr->mId, getGemCharge(), mNewItemName);
/*
End of tes3mp change (major)
*/
return true;
2013-03-28 16:41:00 +00:00
}
void Enchanting::nextCastStyle()
2013-03-28 16:41:00 +00:00
{
if (itemEmpty())
{
mCastStyle = ESM::Enchantment::WhenUsed;
2013-03-28 16:41:00 +00:00
return;
}
const bool powerfulSoul = getGemCharge() >= \
2013-05-27 18:47:53 +00:00
MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find ("iSoulAmountForConstantEffect")->getInt();
if ((mObjectType == typeid(ESM::Armor).name()) || (mObjectType == typeid(ESM::Clothing).name()))
{ // Armor or Clothing
switch(mCastStyle)
2013-03-28 16:41:00 +00:00
{
case ESM::Enchantment::WhenUsed:
if (powerfulSoul)
mCastStyle = ESM::Enchantment::ConstantEffect;
return;
default: // takes care of Constant effect too
mCastStyle = ESM::Enchantment::WhenUsed;
return;
2013-03-28 16:41:00 +00:00
}
}
else if(mObjectType == typeid(ESM::Weapon).name())
{ // Weapon
switch(mCastStyle)
2013-03-28 16:41:00 +00:00
{
case ESM::Enchantment::WhenStrikes:
mCastStyle = ESM::Enchantment::WhenUsed;
2013-05-27 18:47:53 +00:00
return;
case ESM::Enchantment::WhenUsed:
2013-05-27 18:47:53 +00:00
if (powerfulSoul)
mCastStyle = ESM::Enchantment::ConstantEffect;
2013-05-27 18:47:53 +00:00
else
mCastStyle = ESM::Enchantment::WhenStrikes;
2013-05-27 18:47:53 +00:00
return;
default: // takes care of Constant effect too
mCastStyle = ESM::Enchantment::WhenStrikes;
2013-05-27 18:47:53 +00:00
return;
2013-03-28 16:41:00 +00:00
}
}
else if(mObjectType == typeid(ESM::Book).name())
{ // Scroll or Book
mCastStyle = ESM::Enchantment::CastOnce;
return;
2013-03-28 16:41:00 +00:00
}
// Fail case
mCastStyle = ESM::Enchantment::CastOnce;
2013-03-28 16:41:00 +00:00
}
2013-05-27 18:47:53 +00:00
/*
* Vanilla enchant cost formula:
*
* Touch/Self: (min + max) * baseCost * 0.025 * duration + area * baseCost * 0.025
* Target: 1.5 * ((min + max) * baseCost * 0.025 * duration + area * baseCost * 0.025)
* Constant eff: (min + max) * baseCost * 2.5 + area * baseCost * 0.025
*
* For multiple effects - cost of each effect is multiplied by number of effects that follows +1.
*
* Note: Minimal value inside formula for 'min' and 'max' is 1. So in vanilla:
* (0 + 0) == (1 + 0) == (1 + 1) => 2 or (2 + 0) == (1 + 2) => 3
*
* Formula on UESPWiki is not entirely correct.
*/
int Enchanting::getEnchantPoints() const
2013-03-28 16:41:00 +00:00
{
2013-05-27 18:47:53 +00:00
if (mEffectList.mList.empty())
// No effects added, cost = 0
return 0;
2013-03-28 16:41:00 +00:00
const MWWorld::ESMStore &store = MWBase::Environment::get().getWorld()->getStore();
std::vector<ESM::ENAMstruct> mEffects = mEffectList.mList;
2013-03-30 18:08:42 +00:00
int enchantmentCost = 0;
float cost = 0;
2013-03-28 16:41:00 +00:00
for (std::vector<ESM::ENAMstruct>::const_iterator it = mEffects.begin(); it != mEffects.end(); ++it)
{
2014-09-26 15:12:48 +00:00
float baseCost = (store.get<ESM::MagicEffect>().find(it->mEffectID))->mData.mBaseCost;
int magMin = std::max(1, it->mMagnMin);
int magMax = std::max(1, it->mMagnMax);
int area = std::max(1, it->mArea);
2013-03-28 16:41:00 +00:00
float magnitudeCost = (magMin + magMax) * baseCost * 0.05f;
if (mCastStyle == ESM::Enchantment::ConstantEffect)
2013-03-28 16:41:00 +00:00
{
magnitudeCost *= store.get<ESM::GameSetting>().find("fEnchantmentConstantDurationMult")->getFloat();
}
else
{
magnitudeCost *= it->mDuration;
2013-03-28 16:41:00 +00:00
}
float areaCost = area * 0.05f * baseCost;
2013-03-28 16:41:00 +00:00
const float fEffectCostMult = store.get<ESM::GameSetting>().find("fEffectCostMult")->getFloat();
cost += (magnitudeCost + areaCost) * fEffectCostMult;
cost = std::max(1.f, cost);
2015-01-12 22:28:52 +00:00
if (it->mRange == ESM::RT_Target)
cost *= 1.5;
enchantmentCost += static_cast<int>(cost);
2013-03-28 16:41:00 +00:00
}
return enchantmentCost;
2013-03-28 16:41:00 +00:00
}
2013-04-02 18:46:48 +00:00
2013-05-27 16:08:12 +00:00
int Enchanting::getBaseCastCost() const
2013-05-27 16:08:12 +00:00
{
if (mCastStyle == ESM::Enchantment::ConstantEffect)
2013-05-27 18:47:53 +00:00
return 0;
return getEnchantPoints();
}
2013-05-27 16:08:12 +00:00
int Enchanting::getEffectiveCastCost() const
{
int baseCost = getBaseCastCost();
2015-08-21 09:12:39 +00:00
MWWorld::Ptr player = getPlayer();
return getEffectiveEnchantmentCastCost(static_cast<float>(baseCost), player);
2013-05-27 16:08:12 +00:00
}
2013-04-02 18:46:48 +00:00
int Enchanting::getEnchantPrice() const
{
if(mEnchanter.isEmpty())
return 0;
float priceMultipler = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find ("fEnchantmentValueMult")->getFloat();
int price = MWBase::Environment::get().getMechanicsManager()->getBarterOffer(mEnchanter, static_cast<int>(getEnchantPoints() * priceMultipler), true);
2013-04-02 18:46:48 +00:00
return price;
}
int Enchanting::getGemCharge() const
2013-03-28 16:41:00 +00:00
{
const MWWorld::ESMStore &store = MWBase::Environment::get().getWorld()->getStore();
if(soulEmpty())
return 0;
if(mSoulGemPtr.getCellRef().getSoul()=="")
2013-03-28 16:41:00 +00:00
return 0;
const ESM::Creature* soul = store.get<ESM::Creature>().search(mSoulGemPtr.getCellRef().getSoul());
if(soul)
return soul->mData.mSoul;
else
return 0;
2013-03-28 16:41:00 +00:00
}
int Enchanting::getMaxEnchantValue() const
2013-03-28 16:41:00 +00:00
{
if (itemEmpty())
return 0;
2014-01-27 00:58:04 +00:00
const MWWorld::ESMStore &store = MWBase::Environment::get().getWorld()->getStore();
return static_cast<int>(mOldItemPtr.getClass().getEnchantmentPoints(mOldItemPtr) * store.get<ESM::GameSetting>().find("fEnchantmentMult")->getFloat());
2013-03-28 16:41:00 +00:00
}
bool Enchanting::soulEmpty() const
2013-03-28 16:41:00 +00:00
{
return mSoulGemPtr.isEmpty();
2013-03-28 16:41:00 +00:00
}
bool Enchanting::itemEmpty() const
2013-03-28 16:41:00 +00:00
{
return mOldItemPtr.isEmpty();
2013-03-28 16:41:00 +00:00
}
2013-03-30 18:08:42 +00:00
void Enchanting::setSelfEnchanting(bool selfEnchanting)
{
mSelfEnchanting = selfEnchanting;
}
2017-04-20 11:36:14 +00:00
void Enchanting::setEnchanter(const MWWorld::Ptr& enchanter)
2013-03-30 18:08:42 +00:00
{
mEnchanter = enchanter;
}
float Enchanting::getEnchantChance() const
2013-03-30 18:08:42 +00:00
{
const NpcStats& npcStats = mEnchanter.getClass().getNpcStats (mEnchanter);
2013-03-30 18:08:42 +00:00
float chance1 = (npcStats.getSkill (ESM::Skill::Enchant).getModified() +
(0.25f * npcStats.getAttribute (ESM::Attribute::Intelligence).getModified())
+ (0.125f * npcStats.getAttribute (ESM::Attribute::Luck).getModified()));
2014-01-27 00:58:04 +00:00
const MWWorld::Store<ESM::GameSetting>& gmst = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>();
float chance2 = 7.5f / (gmst.find("fEnchantmentChanceMult")->getFloat() * ((mCastStyle == ESM::Enchantment::ConstantEffect) ?
gmst.find("fEnchantmentConstantChanceMult")->getFloat() : 1.0f ))
2014-01-27 00:58:04 +00:00
* getEnchantPoints();
2013-03-30 18:08:42 +00:00
return (chance1-chance2);
}
2013-04-02 18:46:48 +00:00
void Enchanting::payForEnchantment() const
{
2015-08-21 09:12:39 +00:00
const MWWorld::Ptr& player = getPlayer();
MWWorld::ContainerStore& store = player.getClass().getContainerStore(player);
2013-04-02 18:46:48 +00:00
store.remove(MWWorld::ContainerStore::sGoldId, getEnchantPrice(), player);
// add gold to NPC trading gold pool
CreatureStats& enchanterStats = mEnchanter.getClass().getCreatureStats(mEnchanter);
enchanterStats.setGoldPool(enchanterStats.getGoldPool() + getEnchantPrice());
2013-04-02 18:46:48 +00:00
}
2013-03-28 16:41:00 +00:00
}