2017-08-01 05:05:35 +00:00
|
|
|
#include "weaponpriority.hpp"
|
|
|
|
|
|
|
|
#include <components/esm/loadench.hpp>
|
|
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
2018-01-11 17:08:11 +00:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2017-08-01 05:05:35 +00:00
|
|
|
|
|
|
|
#include "npcstats.hpp"
|
|
|
|
#include "combat.hpp"
|
|
|
|
#include "aicombataction.hpp"
|
|
|
|
#include "spellpriority.hpp"
|
2017-08-29 12:40:55 +00:00
|
|
|
#include "spellcasting.hpp"
|
2017-08-01 05:05:35 +00:00
|
|
|
|
|
|
|
namespace MWMechanics
|
|
|
|
{
|
|
|
|
float rateWeapon (const MWWorld::Ptr &item, const MWWorld::Ptr& actor, const MWWorld::Ptr& enemy, int type,
|
|
|
|
float arrowRating, float boltRating)
|
|
|
|
{
|
2018-08-30 20:04:02 +00:00
|
|
|
if (enemy.isEmpty() || item.getTypeName() != typeid(ESM::Weapon).name())
|
|
|
|
return 0.f;
|
|
|
|
|
|
|
|
if (item.getClass().hasItemHealth(item) && item.getClass().getItemHealth(item) == 0)
|
2017-08-01 05:05:35 +00:00
|
|
|
return 0.f;
|
|
|
|
|
|
|
|
const ESM::Weapon* weapon = item.get<ESM::Weapon>()->mBase;
|
|
|
|
|
|
|
|
if (type != -1 && weapon->mData.mType != type)
|
|
|
|
return 0.f;
|
2018-08-30 21:36:30 +00:00
|
|
|
|
2018-08-30 20:04:02 +00:00
|
|
|
const MWBase::World* world = MWBase::Environment::get().getWorld();
|
|
|
|
const MWWorld::Store<ESM::GameSetting>& gmst = world->getStore().get<ESM::GameSetting>();
|
2017-08-01 05:05:35 +00:00
|
|
|
|
2018-08-27 09:38:53 +00:00
|
|
|
if (type == -1 && (weapon->mData.mType == ESM::Weapon::Arrow || weapon->mData.mType == ESM::Weapon::Bolt))
|
|
|
|
return 0.f;
|
|
|
|
|
2017-08-01 05:05:35 +00:00
|
|
|
float rating=0.f;
|
2018-09-06 14:17:30 +00:00
|
|
|
static const float fAIMeleeWeaponMult = gmst.find("fAIMeleeWeaponMult")->mValue.getFloat();
|
|
|
|
float ratingMult = fAIMeleeWeaponMult;
|
2017-08-01 05:05:35 +00:00
|
|
|
|
|
|
|
if (weapon->mData.mType >= ESM::Weapon::MarksmanBow && weapon->mData.mType <= ESM::Weapon::MarksmanThrown)
|
|
|
|
{
|
2018-08-30 20:04:02 +00:00
|
|
|
// Underwater ranged combat is impossible
|
|
|
|
if (world->isUnderwater(MWWorld::ConstPtr(actor), 0.75f)
|
|
|
|
|| world->isUnderwater(MWWorld::ConstPtr(enemy), 0.75f))
|
2017-08-01 05:05:35 +00:00
|
|
|
return 0.f;
|
|
|
|
|
2018-09-06 14:17:30 +00:00
|
|
|
// Use a higher rating multiplier if the actor is out of enemy's reach, use the normal mult otherwise
|
2018-08-01 16:41:49 +00:00
|
|
|
if (getDistanceMinusHalfExtents(actor, enemy) >= getMaxAttackDistance(enemy))
|
2018-08-30 20:04:02 +00:00
|
|
|
{
|
|
|
|
static const float fAIRangeMeleeWeaponMult = gmst.find("fAIRangeMeleeWeaponMult")->mValue.getFloat();
|
2018-09-06 14:17:30 +00:00
|
|
|
ratingMult = fAIRangeMeleeWeaponMult;
|
2018-08-30 20:04:02 +00:00
|
|
|
}
|
2017-08-01 05:05:35 +00:00
|
|
|
}
|
|
|
|
|
2018-08-30 20:04:02 +00:00
|
|
|
const float chop = (weapon->mData.mChop[0] + weapon->mData.mChop[1]) / 2.f;
|
2018-10-28 15:28:36 +00:00
|
|
|
// We need to account for the fact that thrown weapons have 2x real damage applied to the target
|
|
|
|
// as they're both the weapon and the ammo of the hit
|
|
|
|
if (weapon->mData.mType == ESM::Weapon::MarksmanThrown)
|
|
|
|
{
|
|
|
|
rating = chop * 2;
|
|
|
|
}
|
|
|
|
else if (weapon->mData.mType >= ESM::Weapon::MarksmanBow)
|
|
|
|
{
|
2018-08-30 20:04:02 +00:00
|
|
|
rating = chop;
|
2018-10-28 15:28:36 +00:00
|
|
|
}
|
2017-08-01 05:05:35 +00:00
|
|
|
else
|
|
|
|
{
|
2018-08-30 20:04:02 +00:00
|
|
|
const float slash = (weapon->mData.mSlash[0] + weapon->mData.mSlash[1]) / 2.f;
|
|
|
|
const float thrust = (weapon->mData.mThrust[0] + weapon->mData.mThrust[1]) / 2.f;
|
|
|
|
rating = (slash * slash + thrust * thrust + chop * chop) / (slash + thrust + chop);
|
2017-08-01 05:05:35 +00:00
|
|
|
}
|
|
|
|
|
2018-08-30 20:04:02 +00:00
|
|
|
adjustWeaponDamage(rating, item, actor);
|
2017-08-01 05:05:35 +00:00
|
|
|
|
2018-08-30 20:04:02 +00:00
|
|
|
if (weapon->mData.mType != ESM::Weapon::MarksmanBow && weapon->mData.mType != ESM::Weapon::MarksmanCrossbow)
|
2018-12-08 18:47:39 +00:00
|
|
|
{
|
2018-08-30 20:04:02 +00:00
|
|
|
resistNormalWeapon(enemy, actor, item, rating);
|
2018-12-08 18:47:39 +00:00
|
|
|
applyWerewolfDamageMult(enemy, item, rating);
|
|
|
|
}
|
2018-08-30 20:04:02 +00:00
|
|
|
else if (weapon->mData.mType == ESM::Weapon::MarksmanBow)
|
2017-08-01 05:05:35 +00:00
|
|
|
{
|
|
|
|
if (arrowRating <= 0.f)
|
|
|
|
rating = 0.f;
|
|
|
|
else
|
|
|
|
rating += arrowRating;
|
|
|
|
}
|
|
|
|
else if (weapon->mData.mType == ESM::Weapon::MarksmanCrossbow)
|
|
|
|
{
|
|
|
|
if (boltRating <= 0.f)
|
|
|
|
rating = 0.f;
|
|
|
|
else
|
|
|
|
rating += boltRating;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!weapon->mEnchant.empty())
|
|
|
|
{
|
2018-08-30 20:04:02 +00:00
|
|
|
const ESM::Enchantment* enchantment = world->getStore().get<ESM::Enchantment>().find(weapon->mEnchant);
|
2017-08-29 12:40:55 +00:00
|
|
|
if (enchantment->mData.mType == ESM::Enchantment::WhenStrikes)
|
|
|
|
{
|
|
|
|
int castCost = getEffectiveEnchantmentCastCost(static_cast<float>(enchantment->mData.mCost), actor);
|
2019-07-31 10:10:51 +00:00
|
|
|
float charge = item.getCellRef().getEnchantmentCharge();
|
2017-08-29 12:40:55 +00:00
|
|
|
|
2019-07-31 10:10:51 +00:00
|
|
|
if (charge == -1 || charge >= castCost || weapon->mData.mType >= ESM::Weapon::MarksmanThrown)
|
2017-08-29 12:40:55 +00:00
|
|
|
rating += rateEffects(enchantment->mEffects, actor, enemy);
|
|
|
|
}
|
2017-08-01 05:05:35 +00:00
|
|
|
}
|
|
|
|
|
2018-08-30 22:54:59 +00:00
|
|
|
int value = 50.f;
|
2018-08-20 18:38:57 +00:00
|
|
|
if (actor.getClass().isNpc())
|
2018-08-01 16:27:19 +00:00
|
|
|
{
|
2018-08-20 18:38:57 +00:00
|
|
|
int skill = item.getClass().getEquipmentSkill(item);
|
|
|
|
if (skill != -1)
|
2018-08-30 22:54:59 +00:00
|
|
|
value = actor.getClass().getSkill(actor, skill);
|
2018-08-20 18:38:57 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MWWorld::LiveCellRef<ESM::Creature> *ref = actor.get<ESM::Creature>();
|
2018-08-30 22:54:59 +00:00
|
|
|
value = ref->mBase->mData.mCombat;
|
2018-08-01 16:27:19 +00:00
|
|
|
}
|
2017-08-01 05:05:35 +00:00
|
|
|
|
2019-03-09 18:20:43 +00:00
|
|
|
// Take hit chance in account, but do not allow rating become negative.
|
|
|
|
float chance = getHitChance(actor, enemy, value) / 100.f;
|
|
|
|
rating *= std::min(1.f, std::max(0.01f, chance));
|
2018-08-30 22:54:59 +00:00
|
|
|
|
2018-09-19 08:04:58 +00:00
|
|
|
if (weapon->mData.mType < ESM::Weapon::Arrow)
|
2018-08-30 23:06:29 +00:00
|
|
|
rating *= weapon->mData.mSpeed;
|
|
|
|
|
2018-09-06 14:17:30 +00:00
|
|
|
return rating * ratingMult;
|
2017-08-01 05:05:35 +00:00
|
|
|
}
|
|
|
|
|
2018-01-11 17:08:11 +00:00
|
|
|
float rateAmmo(const MWWorld::Ptr &actor, const MWWorld::Ptr &enemy, MWWorld::Ptr &bestAmmo, ESM::Weapon::Type ammoType)
|
|
|
|
{
|
|
|
|
float bestAmmoRating = 0.f;
|
|
|
|
if (!actor.getClass().hasInventoryStore(actor))
|
|
|
|
return bestAmmoRating;
|
|
|
|
|
|
|
|
MWWorld::InventoryStore& store = actor.getClass().getInventoryStore(actor);
|
|
|
|
|
|
|
|
for (MWWorld::ContainerStoreIterator it = store.begin(); it != store.end(); ++it)
|
|
|
|
{
|
|
|
|
float rating = rateWeapon(*it, actor, enemy, ammoType);
|
|
|
|
if (rating > bestAmmoRating)
|
|
|
|
{
|
|
|
|
bestAmmoRating = rating;
|
|
|
|
bestAmmo = *it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return bestAmmoRating;
|
|
|
|
}
|
|
|
|
|
|
|
|
float rateAmmo(const MWWorld::Ptr &actor, const MWWorld::Ptr &enemy, ESM::Weapon::Type ammoType)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr emptyPtr;
|
|
|
|
return rateAmmo(actor, enemy, emptyPtr, ammoType);
|
|
|
|
}
|
|
|
|
|
2017-08-01 05:05:35 +00:00
|
|
|
float vanillaRateWeaponAndAmmo(const MWWorld::Ptr& weapon, const MWWorld::Ptr& ammo, const MWWorld::Ptr& actor, const MWWorld::Ptr& enemy)
|
|
|
|
{
|
|
|
|
const MWWorld::Store<ESM::GameSetting>& gmst = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>();
|
|
|
|
|
2018-08-29 15:38:12 +00:00
|
|
|
static const float fAIMeleeWeaponMult = gmst.find("fAIMeleeWeaponMult")->mValue.getFloat();
|
|
|
|
static const float fAIMeleeArmorMult = gmst.find("fAIMeleeArmorMult")->mValue.getFloat();
|
|
|
|
static const float fAIRangeMeleeWeaponMult = gmst.find("fAIRangeMeleeWeaponMult")->mValue.getFloat();
|
2017-08-01 05:05:35 +00:00
|
|
|
|
|
|
|
if (weapon.isEmpty())
|
|
|
|
return 0.f;
|
|
|
|
|
|
|
|
float skillMult = actor.getClass().getSkill(actor, weapon.getClass().getEquipmentSkill(weapon)) * 0.01f;
|
|
|
|
float chopMult = fAIMeleeWeaponMult;
|
|
|
|
float bonusDamage = 0.f;
|
|
|
|
|
|
|
|
const ESM::Weapon* esmWeap = weapon.get<ESM::Weapon>()->mBase;
|
|
|
|
|
|
|
|
if (esmWeap->mData.mType >= ESM::Weapon::MarksmanBow)
|
|
|
|
{
|
|
|
|
if (!ammo.isEmpty() && !MWBase::Environment::get().getWorld()->isSwimming(enemy))
|
|
|
|
{
|
|
|
|
bonusDamage = ammo.get<ESM::Weapon>()->mBase->mData.mChop[1];
|
|
|
|
chopMult = fAIRangeMeleeWeaponMult;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
chopMult = 0.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
float chopRating = (esmWeap->mData.mChop[1] + bonusDamage) * skillMult * chopMult;
|
|
|
|
float slashRating = esmWeap->mData.mSlash[1] * skillMult * fAIMeleeWeaponMult;
|
|
|
|
float thrustRating = esmWeap->mData.mThrust[1] * skillMult * fAIMeleeWeaponMult;
|
|
|
|
|
|
|
|
return actor.getClass().getArmorRating(actor) * fAIMeleeArmorMult
|
|
|
|
+ std::max(std::max(chopRating, slashRating), thrustRating);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|