2014-01-21 00:01:21 +00:00
|
|
|
#include "combat.hpp"
|
|
|
|
|
2015-04-22 15:58:55 +00:00
|
|
|
#include <components/misc/rng.hpp>
|
2018-03-30 23:38:36 +00:00
|
|
|
#include <components/settings/settings.hpp>
|
2015-03-15 01:07:47 +00:00
|
|
|
|
2015-11-20 20:57:04 +00:00
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
|
|
|
|
2014-01-21 00:01:21 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
2014-03-08 04:51:47 +00:00
|
|
|
#include "../mwbase/mechanicsmanager.hpp"
|
|
|
|
#include "../mwbase/soundmanager.hpp"
|
2016-06-17 14:07:16 +00:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2014-01-21 00:01:21 +00:00
|
|
|
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2014-02-23 19:11:05 +00:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2014-01-21 00:01:21 +00:00
|
|
|
|
2016-06-17 14:07:16 +00:00
|
|
|
#include "npcstats.hpp"
|
|
|
|
#include "movement.hpp"
|
|
|
|
#include "spellcasting.hpp"
|
2020-04-04 15:28:53 +00:00
|
|
|
#include "spellresistance.hpp"
|
2016-06-17 14:07:16 +00:00
|
|
|
#include "difficultyscaling.hpp"
|
2015-08-21 09:12:39 +00:00
|
|
|
#include "actorutil.hpp"
|
2019-10-19 10:40:34 +00:00
|
|
|
#include "pathfinding.hpp"
|
2014-01-22 11:09:44 +00:00
|
|
|
|
2014-01-21 00:01:21 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
2015-05-29 23:00:24 +00:00
|
|
|
float signedAngleRadians (const osg::Vec3f& v1, const osg::Vec3f& v2, const osg::Vec3f& normal)
|
2014-01-21 00:01:21 +00:00
|
|
|
{
|
2015-05-29 23:00:24 +00:00
|
|
|
return std::atan2((normal * (v1 ^ v2)), (v1 * v2));
|
2014-01-21 00:01:21 +00:00
|
|
|
}
|
|
|
|
|
2016-02-22 18:37:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace MWMechanics
|
2014-03-08 04:51:47 +00:00
|
|
|
{
|
2016-02-22 18:37:19 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
bool applyOnStrikeEnchantment(const MWWorld::Ptr& attacker, const MWWorld::Ptr& victim, const MWWorld::Ptr& object, const osg::Vec3f& hitPosition, const bool fromProjectile)
|
2014-03-08 04:51:47 +00:00
|
|
|
{
|
2016-02-22 18:37:19 +00:00
|
|
|
std::string enchantmentName = !object.isEmpty() ? object.getClass().getEnchantment(object) : "";
|
|
|
|
if (!enchantmentName.empty())
|
2014-03-08 04:51:47 +00:00
|
|
|
{
|
2016-02-22 18:37:19 +00:00
|
|
|
const ESM::Enchantment* enchantment = MWBase::Environment::get().getWorld()->getStore().get<ESM::Enchantment>().find(
|
|
|
|
enchantmentName);
|
|
|
|
if (enchantment->mData.mType == ESM::Enchantment::WhenStrikes)
|
|
|
|
{
|
2016-12-19 09:15:19 +00:00
|
|
|
MWMechanics::CastSpell cast(attacker, victim, fromProjectile);
|
2016-02-22 18:37:19 +00:00
|
|
|
cast.mHitPosition = hitPosition;
|
|
|
|
cast.cast(object, false);
|
|
|
|
return true;
|
|
|
|
}
|
2014-03-08 04:51:47 +00:00
|
|
|
}
|
2016-02-22 18:37:19 +00:00
|
|
|
return false;
|
2014-03-08 04:51:47 +00:00
|
|
|
}
|
2014-01-21 00:01:21 +00:00
|
|
|
|
2015-06-26 03:15:07 +00:00
|
|
|
bool blockMeleeAttack(const MWWorld::Ptr &attacker, const MWWorld::Ptr &blocker, const MWWorld::Ptr &weapon, float damage, float attackStrength)
|
2014-01-21 00:01:21 +00:00
|
|
|
{
|
|
|
|
if (!blocker.getClass().hasInventoryStore(blocker))
|
|
|
|
return false;
|
|
|
|
|
2014-07-20 14:52:57 +00:00
|
|
|
MWMechanics::CreatureStats& blockerStats = blocker.getClass().getCreatureStats(blocker);
|
|
|
|
|
|
|
|
if (blockerStats.getKnockedDown() // Used for both knockout or knockdown
|
|
|
|
|| blockerStats.getHitRecovery()
|
2015-08-20 06:12:37 +00:00
|
|
|
|| blockerStats.isParalyzed())
|
2014-07-20 14:52:57 +00:00
|
|
|
return false;
|
|
|
|
|
2014-12-12 15:49:22 +00:00
|
|
|
if (!MWBase::Environment::get().getMechanicsManager()->isReadyToBlock(blocker))
|
2014-01-21 00:01:21 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
MWWorld::InventoryStore& inv = blocker.getClass().getInventoryStore(blocker);
|
|
|
|
MWWorld::ContainerStoreIterator shield = inv.getSlot(MWWorld::InventoryStore::Slot_CarriedLeft);
|
|
|
|
if (shield == inv.end() || shield->getTypeName() != typeid(ESM::Armor).name())
|
|
|
|
return false;
|
|
|
|
|
2015-05-29 23:00:24 +00:00
|
|
|
if (!blocker.getRefData().getBaseNode())
|
|
|
|
return false; // shouldn't happen
|
|
|
|
|
|
|
|
float angleDegrees = osg::RadiansToDegrees(
|
|
|
|
signedAngleRadians (
|
|
|
|
(attacker.getRefData().getPosition().asVec3() - blocker.getRefData().getPosition().asVec3()),
|
|
|
|
blocker.getRefData().getBaseNode()->getAttitude() * osg::Vec3f(0,1,0),
|
|
|
|
osg::Vec3f(0,0,1)));
|
2014-01-21 00:01:21 +00:00
|
|
|
|
|
|
|
const MWWorld::Store<ESM::GameSetting>& gmst = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>();
|
2018-08-29 15:38:12 +00:00
|
|
|
if (angleDegrees < gmst.find("fCombatBlockLeftAngle")->mValue.getFloat())
|
2014-01-21 00:01:21 +00:00
|
|
|
return false;
|
2018-08-29 15:38:12 +00:00
|
|
|
if (angleDegrees > gmst.find("fCombatBlockRightAngle")->mValue.getFloat())
|
2014-01-21 00:01:21 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
MWMechanics::CreatureStats& attackerStats = attacker.getClass().getCreatureStats(attacker);
|
|
|
|
|
2015-03-08 04:42:07 +00:00
|
|
|
float blockTerm = blocker.getClass().getSkill(blocker, ESM::Skill::Block) + 0.2f * blockerStats.getAttribute(ESM::Attribute::Agility).getModified()
|
|
|
|
+ 0.1f * blockerStats.getAttribute(ESM::Attribute::Luck).getModified();
|
2015-06-26 03:15:07 +00:00
|
|
|
float enemySwing = attackStrength;
|
2018-08-29 15:38:12 +00:00
|
|
|
float swingTerm = enemySwing * gmst.find("fSwingBlockMult")->mValue.getFloat() + gmst.find("fSwingBlockBase")->mValue.getFloat();
|
2014-01-21 00:01:21 +00:00
|
|
|
|
|
|
|
float blockerTerm = blockTerm * swingTerm;
|
|
|
|
if (blocker.getClass().getMovementSettings(blocker).mPosition[1] <= 0)
|
2018-08-29 15:38:12 +00:00
|
|
|
blockerTerm *= gmst.find("fBlockStillBonus")->mValue.getFloat();
|
2014-01-21 00:01:21 +00:00
|
|
|
blockerTerm *= blockerStats.getFatigueTerm();
|
|
|
|
|
2018-12-23 11:18:33 +00:00
|
|
|
float attackerSkill = 0;
|
2014-12-17 15:07:50 +00:00
|
|
|
if (weapon.isEmpty())
|
|
|
|
attackerSkill = attacker.getClass().getSkill(attacker, ESM::Skill::HandToHand);
|
|
|
|
else
|
|
|
|
attackerSkill = attacker.getClass().getSkill(attacker, weapon.getClass().getEquipmentSkill(weapon));
|
2015-03-08 04:42:07 +00:00
|
|
|
float attackerTerm = attackerSkill + 0.2f * attackerStats.getAttribute(ESM::Attribute::Agility).getModified()
|
|
|
|
+ 0.1f * attackerStats.getAttribute(ESM::Attribute::Luck).getModified();
|
2014-01-21 00:01:21 +00:00
|
|
|
attackerTerm *= attackerStats.getFatigueTerm();
|
|
|
|
|
|
|
|
int x = int(blockerTerm - attackerTerm);
|
2018-08-29 15:38:12 +00:00
|
|
|
int iBlockMaxChance = gmst.find("iBlockMaxChance")->mValue.getInteger();
|
|
|
|
int iBlockMinChance = gmst.find("iBlockMinChance")->mValue.getInteger();
|
2014-01-21 00:01:21 +00:00
|
|
|
x = std::min(iBlockMaxChance, std::max(iBlockMinChance, x));
|
|
|
|
|
2015-04-22 15:58:55 +00:00
|
|
|
if (Misc::Rng::roll0to99() < x)
|
2014-01-21 00:01:21 +00:00
|
|
|
{
|
2019-06-20 17:45:52 +00:00
|
|
|
// Reduce shield durability by incoming damage
|
|
|
|
int shieldhealth = shield->getClass().getItemHealth(*shield);
|
2014-01-21 00:01:21 +00:00
|
|
|
|
2019-06-20 17:45:52 +00:00
|
|
|
shieldhealth -= std::min(shieldhealth, int(damage));
|
|
|
|
shield->getCellRef().setCharge(shieldhealth);
|
|
|
|
if (shieldhealth == 0)
|
|
|
|
inv.unequipItem(*shield, blocker);
|
2014-01-21 00:01:21 +00:00
|
|
|
// Reduce blocker fatigue
|
2018-08-29 15:38:12 +00:00
|
|
|
const float fFatigueBlockBase = gmst.find("fFatigueBlockBase")->mValue.getFloat();
|
|
|
|
const float fFatigueBlockMult = gmst.find("fFatigueBlockMult")->mValue.getFloat();
|
|
|
|
const float fWeaponFatigueBlockMult = gmst.find("fWeaponFatigueBlockMult")->mValue.getFloat();
|
2014-01-21 00:01:21 +00:00
|
|
|
MWMechanics::DynamicStat<float> fatigue = blockerStats.getFatigue();
|
2014-10-05 13:50:01 +00:00
|
|
|
float normalizedEncumbrance = blocker.getClass().getNormalizedEncumbrance(blocker);
|
2014-01-21 00:01:21 +00:00
|
|
|
normalizedEncumbrance = std::min(1.f, normalizedEncumbrance);
|
|
|
|
float fatigueLoss = fFatigueBlockBase + normalizedEncumbrance * fFatigueBlockMult;
|
2014-12-17 15:07:50 +00:00
|
|
|
if (!weapon.isEmpty())
|
2015-06-26 03:15:07 +00:00
|
|
|
fatigueLoss += weapon.getClass().getWeight(weapon) * attackStrength * fWeaponFatigueBlockMult;
|
2014-01-21 00:01:21 +00:00
|
|
|
fatigue.setCurrent(fatigue.getCurrent() - fatigueLoss);
|
|
|
|
blockerStats.setFatigue(fatigue);
|
|
|
|
|
|
|
|
blockerStats.setBlock(true);
|
|
|
|
|
2015-08-21 09:12:39 +00:00
|
|
|
if (blocker == getPlayer())
|
2014-01-21 00:01:21 +00:00
|
|
|
blocker.getClass().skillUsageSucceeded(blocker, ESM::Skill::Block, 0);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-12-08 18:17:15 +00:00
|
|
|
bool isNormalWeapon(const MWWorld::Ptr &weapon)
|
2014-01-22 11:09:44 +00:00
|
|
|
{
|
2018-12-08 18:17:15 +00:00
|
|
|
if (weapon.isEmpty())
|
|
|
|
return false;
|
2014-01-22 11:09:44 +00:00
|
|
|
|
2018-12-08 18:55:08 +00:00
|
|
|
const int flags = weapon.get<ESM::Weapon>()->mBase->mData.mFlags;
|
2018-12-08 18:17:15 +00:00
|
|
|
bool isSilver = flags & ESM::Weapon::Silver;
|
|
|
|
bool isMagical = flags & ESM::Weapon::Magical;
|
|
|
|
bool isEnchanted = !weapon.getClass().getEnchantment(weapon).empty();
|
2014-01-22 11:09:44 +00:00
|
|
|
|
2018-12-08 18:17:15 +00:00
|
|
|
return !isSilver && !isMagical && (!isEnchanted || !Settings::Manager::getBool("enchanted weapons are magical", "Game"));
|
|
|
|
}
|
|
|
|
|
2018-12-08 18:55:08 +00:00
|
|
|
void resistNormalWeapon(const MWWorld::Ptr &actor, const MWWorld::Ptr& attacker, const MWWorld::Ptr &weapon, float &damage)
|
2018-12-08 18:17:15 +00:00
|
|
|
{
|
2018-12-08 18:47:39 +00:00
|
|
|
if (damage == 0 || weapon.isEmpty() || !isNormalWeapon(weapon))
|
2018-12-08 18:55:08 +00:00
|
|
|
return;
|
2018-12-08 18:17:15 +00:00
|
|
|
|
2018-12-08 18:47:39 +00:00
|
|
|
const MWMechanics::MagicEffects& effects = actor.getClass().getCreatureStats(actor).getMagicEffects();
|
|
|
|
const float resistance = effects.get(ESM::MagicEffect::ResistNormalWeapons).getMagnitude() / 100.f;
|
|
|
|
const float weakness = effects.get(ESM::MagicEffect::WeaknessToNormalWeapons).getMagnitude() / 100.f;
|
2014-01-22 11:09:44 +00:00
|
|
|
|
2018-12-08 18:47:39 +00:00
|
|
|
damage *= 1.f - std::min(1.f, resistance-weakness);
|
2014-01-22 11:09:44 +00:00
|
|
|
|
2018-12-08 18:55:08 +00:00
|
|
|
if (damage == 0 && attacker == getPlayer())
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sMagicTargetResistsWeapons}");
|
2014-01-22 11:09:44 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 18:47:39 +00:00
|
|
|
void applyWerewolfDamageMult(const MWWorld::Ptr &actor, const MWWorld::Ptr &weapon, float &damage)
|
|
|
|
{
|
2018-12-08 18:55:08 +00:00
|
|
|
if (damage == 0 || weapon.isEmpty() || !actor.getClass().isNpc())
|
2018-12-08 18:47:39 +00:00
|
|
|
return;
|
|
|
|
|
2018-12-08 18:55:08 +00:00
|
|
|
const int flags = weapon.get<ESM::Weapon>()->mBase->mData.mFlags;
|
2018-12-08 18:47:39 +00:00
|
|
|
bool isSilver = flags & ESM::Weapon::Silver;
|
|
|
|
|
2018-12-08 18:55:08 +00:00
|
|
|
if (isSilver && actor.getClass().getNpcStats(actor).isWerewolf())
|
|
|
|
{
|
|
|
|
const MWWorld::ESMStore& store = MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
damage *= store.get<ESM::GameSetting>().find("fWereWolfSilverWeaponDamageMult")->mValue.getFloat();
|
|
|
|
}
|
2018-12-08 18:47:39 +00:00
|
|
|
}
|
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
void projectileHit(const MWWorld::Ptr& attacker, const MWWorld::Ptr& victim, MWWorld::Ptr weapon, const MWWorld::Ptr& projectile,
|
2015-06-26 00:32:41 +00:00
|
|
|
const osg::Vec3f& hitPosition, float attackStrength)
|
2014-03-08 04:51:47 +00:00
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
|
|
|
const MWWorld::Store<ESM::GameSetting> &gmst = world->getStore().get<ESM::GameSetting>();
|
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
bool validVictim = !victim.isEmpty() && victim.getClass().isActor();
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
float damage = 0.f;
|
|
|
|
if (validVictim)
|
|
|
|
{
|
|
|
|
if (attacker == getPlayer())
|
|
|
|
MWBase::Environment::get().getWindowManager()->setEnemy(victim);
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
int weaponSkill = ESM::Skill::Marksman;
|
|
|
|
if (!weapon.isEmpty())
|
|
|
|
weaponSkill = weapon.getClass().getEquipmentSkill(weapon);
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
int skillValue = attacker.getClass().getSkill(attacker, weapon.getClass().getEquipmentSkill(weapon));
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
if (Misc::Rng::roll0to99() >= getHitChance(attacker, victim, skillValue))
|
|
|
|
{
|
|
|
|
victim.getClass().onHit(victim, damage, false, projectile, attacker, osg::Vec3f(), false);
|
|
|
|
MWMechanics::reduceWeaponCondition(damage, false, weapon, attacker);
|
|
|
|
return;
|
|
|
|
}
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
const unsigned char* attack = weapon.get<ESM::Weapon>()->mBase->mData.mChop;
|
|
|
|
damage = attack[0] + ((attack[1] - attack[0]) * attackStrength); // Bow/crossbow damage
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
// Arrow/bolt damage
|
|
|
|
// NB in case of thrown weapons, we are applying the damage twice since projectile == weapon
|
|
|
|
attack = projectile.get<ESM::Weapon>()->mBase->mData.mChop;
|
|
|
|
damage += attack[0] + ((attack[1] - attack[0]) * attackStrength);
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
adjustWeaponDamage(damage, weapon, attacker);
|
2019-02-22 14:18:23 +00:00
|
|
|
if (weapon == projectile || Settings::Manager::getBool("only appropriate ammunition bypasses resistance", "Game") || isNormalWeapon(weapon))
|
2018-12-08 18:55:08 +00:00
|
|
|
resistNormalWeapon(victim, attacker, projectile, damage);
|
2018-12-08 18:47:39 +00:00
|
|
|
applyWerewolfDamageMult(victim, projectile, damage);
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2018-07-29 16:27:13 +00:00
|
|
|
if (attacker == getPlayer())
|
2016-12-19 09:15:19 +00:00
|
|
|
attacker.getClass().skillUsageSucceeded(attacker, weaponSkill, 0);
|
2018-07-29 16:27:13 +00:00
|
|
|
|
2019-07-31 10:54:27 +00:00
|
|
|
const MWMechanics::AiSequence& sequence = victim.getClass().getCreatureStats(victim).getAiSequence();
|
|
|
|
bool unaware = attacker == getPlayer() && !sequence.isInCombat()
|
|
|
|
&& !MWBase::Environment::get().getMechanicsManager()->awarenessCheck(attacker, victim);
|
|
|
|
bool knockedDown = victim.getClass().getCreatureStats(victim).getKnockedDown();
|
|
|
|
if (knockedDown || unaware)
|
|
|
|
{
|
|
|
|
damage *= gmst.find("fCombatKODamageMult")->mValue.getFloat();
|
|
|
|
if (!knockedDown)
|
2018-07-29 16:27:13 +00:00
|
|
|
MWBase::Environment::get().getSoundManager()->playSound3D(victim, "critical damage", 1.0f, 1.0f);
|
|
|
|
}
|
2016-12-19 09:15:19 +00:00
|
|
|
}
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
reduceWeaponCondition(damage, validVictim, weapon, attacker);
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2019-10-20 10:30:52 +00:00
|
|
|
// Apply "On hit" effect of the projectile
|
|
|
|
bool appliedEnchantment = applyOnStrikeEnchantment(attacker, victim, projectile, hitPosition, true);
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
if (validVictim)
|
2014-06-15 14:00:29 +00:00
|
|
|
{
|
2016-12-19 09:15:19 +00:00
|
|
|
// Non-enchanted arrows shot at enemies have a chance to turn up in their inventory
|
|
|
|
if (victim != getPlayer() && !appliedEnchantment)
|
|
|
|
{
|
2018-08-29 15:38:12 +00:00
|
|
|
float fProjectileThrownStoreChance = gmst.find("fProjectileThrownStoreChance")->mValue.getFloat();
|
2016-12-19 09:15:19 +00:00
|
|
|
if (Misc::Rng::rollProbability() < fProjectileThrownStoreChance / 100.f)
|
|
|
|
victim.getClass().getContainerStore(victim).add(projectile, 1, victim);
|
|
|
|
}
|
2014-03-08 04:51:47 +00:00
|
|
|
|
2016-12-19 09:15:19 +00:00
|
|
|
victim.getClass().onHit(victim, damage, true, projectile, attacker, hitPosition, true);
|
|
|
|
}
|
2014-03-08 04:51:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
float getHitChance(const MWWorld::Ptr &attacker, const MWWorld::Ptr &victim, int skillValue)
|
|
|
|
{
|
|
|
|
MWMechanics::CreatureStats &stats = attacker.getClass().getCreatureStats(attacker);
|
|
|
|
const MWMechanics::MagicEffects &mageffects = stats.getMagicEffects();
|
2015-03-12 01:44:41 +00:00
|
|
|
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
|
|
|
const MWWorld::Store<ESM::GameSetting> &gmst = world->getStore().get<ESM::GameSetting>();
|
|
|
|
|
|
|
|
float defenseTerm = 0;
|
2015-08-20 06:17:02 +00:00
|
|
|
MWMechanics::CreatureStats& victimStats = victim.getClass().getCreatureStats(victim);
|
|
|
|
if (victimStats.getFatigue().getCurrent() >= 0)
|
2015-03-12 01:44:41 +00:00
|
|
|
{
|
|
|
|
// Maybe we should keep an aware state for actors updated every so often instead of testing every time
|
|
|
|
bool unaware = (!victimStats.getAiSequence().isInCombat())
|
2015-08-21 09:12:39 +00:00
|
|
|
&& (attacker == getPlayer())
|
2015-03-12 01:44:41 +00:00
|
|
|
&& (!MWBase::Environment::get().getMechanicsManager()->awarenessCheck(attacker, victim));
|
|
|
|
if (!(victimStats.getKnockedDown() ||
|
2015-08-20 06:12:37 +00:00
|
|
|
victimStats.isParalyzed()
|
2015-03-12 01:44:41 +00:00
|
|
|
|| unaware ))
|
|
|
|
{
|
|
|
|
defenseTerm = victimStats.getEvasion();
|
|
|
|
}
|
|
|
|
defenseTerm += std::min(100.f,
|
2018-08-29 15:38:12 +00:00
|
|
|
gmst.find("fCombatInvisoMult")->mValue.getFloat() *
|
2015-03-12 01:44:41 +00:00
|
|
|
victimStats.getMagicEffects().get(ESM::MagicEffect::Chameleon).getMagnitude());
|
|
|
|
defenseTerm += std::min(100.f,
|
2018-08-29 15:38:12 +00:00
|
|
|
gmst.find("fCombatInvisoMult")->mValue.getFloat() *
|
2015-03-12 01:44:41 +00:00
|
|
|
victimStats.getMagicEffects().get(ESM::MagicEffect::Invisibility).getMagnitude());
|
|
|
|
}
|
|
|
|
float attackTerm = skillValue +
|
2014-03-08 04:51:47 +00:00
|
|
|
(stats.getAttribute(ESM::Attribute::Agility).getModified() / 5.0f) +
|
|
|
|
(stats.getAttribute(ESM::Attribute::Luck).getModified() / 10.0f);
|
2015-03-12 01:44:41 +00:00
|
|
|
attackTerm *= stats.getFatigueTerm();
|
|
|
|
attackTerm += mageffects.get(ESM::MagicEffect::FortifyAttack).getMagnitude() -
|
2014-08-16 20:38:22 +00:00
|
|
|
mageffects.get(ESM::MagicEffect::Blind).getMagnitude();
|
2015-03-12 01:44:41 +00:00
|
|
|
|
2015-03-14 19:49:03 +00:00
|
|
|
return round(attackTerm - defenseTerm);
|
2014-03-08 04:51:47 +00:00
|
|
|
}
|
|
|
|
|
2014-07-15 19:53:11 +00:00
|
|
|
void applyElementalShields(const MWWorld::Ptr &attacker, const MWWorld::Ptr &victim)
|
|
|
|
{
|
2021-03-01 21:37:30 +00:00
|
|
|
// Don't let elemental shields harm the player in god mode.
|
|
|
|
bool godmode = attacker == getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState();
|
|
|
|
if (godmode)
|
|
|
|
return;
|
2014-07-15 19:53:11 +00:00
|
|
|
for (int i=0; i<3; ++i)
|
|
|
|
{
|
2014-08-16 20:38:22 +00:00
|
|
|
float magnitude = victim.getClass().getCreatureStats(victim).getMagicEffects().get(ESM::MagicEffect::FireShield+i).getMagnitude();
|
2014-07-15 19:53:11 +00:00
|
|
|
|
|
|
|
if (!magnitude)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CreatureStats& attackerStats = attacker.getClass().getCreatureStats(attacker);
|
|
|
|
float saveTerm = attacker.getClass().getSkill(attacker, ESM::Skill::Destruction)
|
|
|
|
+ 0.2f * attackerStats.getAttribute(ESM::Attribute::Willpower).getModified()
|
|
|
|
+ 0.1f * attackerStats.getAttribute(ESM::Attribute::Luck).getModified();
|
|
|
|
|
2015-03-08 04:42:07 +00:00
|
|
|
float fatigueMax = attackerStats.getFatigue().getModified();
|
|
|
|
float fatigueCurrent = attackerStats.getFatigue().getCurrent();
|
2014-07-15 19:53:11 +00:00
|
|
|
|
2015-03-08 04:42:07 +00:00
|
|
|
float normalisedFatigue = floor(fatigueMax)==0 ? 1 : std::max (0.0f, (fatigueCurrent/fatigueMax));
|
2014-07-15 19:53:11 +00:00
|
|
|
|
|
|
|
saveTerm *= 1.25f * normalisedFatigue;
|
|
|
|
|
2015-04-22 15:58:55 +00:00
|
|
|
float x = std::max(0.f, saveTerm - Misc::Rng::roll0to99());
|
2014-07-15 19:53:11 +00:00
|
|
|
|
|
|
|
int element = ESM::MagicEffect::FireDamage;
|
|
|
|
if (i == 1)
|
|
|
|
element = ESM::MagicEffect::ShockDamage;
|
|
|
|
if (i == 2)
|
|
|
|
element = ESM::MagicEffect::FrostDamage;
|
|
|
|
|
2014-07-15 20:06:46 +00:00
|
|
|
float elementResistance = MWMechanics::getEffectResistanceAttribute(element, &attackerStats.getMagicEffects());
|
2014-07-15 19:53:11 +00:00
|
|
|
|
|
|
|
x = std::min(100.f, x + elementResistance);
|
|
|
|
|
2018-08-29 15:38:12 +00:00
|
|
|
static const float fElementalShieldMult = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fElementalShieldMult")->mValue.getFloat();
|
2014-07-15 19:53:11 +00:00
|
|
|
x = fElementalShieldMult * magnitude * (1.f - 0.01f * x);
|
2014-07-20 14:22:52 +00:00
|
|
|
|
|
|
|
// Note swapped victim and attacker, since the attacker takes the damage here.
|
|
|
|
x = scaleDamage(x, victim, attacker);
|
|
|
|
|
2014-07-15 19:53:11 +00:00
|
|
|
MWMechanics::DynamicStat<float> health = attackerStats.getHealth();
|
|
|
|
health.setCurrent(health.getCurrent() - x);
|
|
|
|
attackerStats.setHealth(health);
|
2021-02-07 11:55:10 +00:00
|
|
|
|
|
|
|
MWBase::Environment::get().getSoundManager()->playSound3D(attacker, "Health Damage", 1.0f, 1.0f);
|
2014-07-15 19:53:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-02 21:14:17 +00:00
|
|
|
void reduceWeaponCondition(float damage, bool hit, MWWorld::Ptr &weapon, const MWWorld::Ptr &attacker)
|
|
|
|
{
|
|
|
|
if (weapon.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!hit)
|
|
|
|
damage = 0.f;
|
|
|
|
|
|
|
|
const bool weaphashealth = weapon.getClass().hasItemHealth(weapon);
|
|
|
|
if(weaphashealth)
|
|
|
|
{
|
|
|
|
int weaphealth = weapon.getClass().getItemHealth(weapon);
|
|
|
|
|
2017-03-25 18:40:11 +00:00
|
|
|
bool godmode = attacker == MWMechanics::getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState();
|
2014-08-02 21:14:17 +00:00
|
|
|
|
2017-03-25 18:40:11 +00:00
|
|
|
// weapon condition does not degrade when godmode is on
|
|
|
|
if (!godmode)
|
|
|
|
{
|
2018-08-29 15:38:12 +00:00
|
|
|
const float fWeaponDamageMult = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fWeaponDamageMult")->mValue.getFloat();
|
2017-03-25 18:40:11 +00:00
|
|
|
float x = std::max(1.f, fWeaponDamageMult * damage);
|
|
|
|
|
|
|
|
weaphealth -= std::min(int(x), weaphealth);
|
|
|
|
weapon.getCellRef().setCharge(weaphealth);
|
|
|
|
}
|
2014-08-02 21:14:17 +00:00
|
|
|
|
|
|
|
// Weapon broken? unequip it
|
|
|
|
if (weaphealth == 0)
|
|
|
|
weapon = *attacker.getClass().getInventoryStore(attacker).unequipItem(weapon, attacker);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-12 02:08:58 +00:00
|
|
|
void adjustWeaponDamage(float &damage, const MWWorld::Ptr &weapon, const MWWorld::Ptr& attacker)
|
2014-08-02 21:14:17 +00:00
|
|
|
{
|
|
|
|
if (weapon.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const bool weaphashealth = weapon.getClass().hasItemHealth(weapon);
|
2018-10-24 15:51:34 +00:00
|
|
|
if (weaphashealth)
|
2014-08-02 21:14:17 +00:00
|
|
|
{
|
2018-10-25 12:45:31 +00:00
|
|
|
damage *= weapon.getClass().getItemNormalizedHealth(weapon);
|
2014-08-02 21:14:17 +00:00
|
|
|
}
|
2015-03-12 02:08:58 +00:00
|
|
|
|
|
|
|
static const float fDamageStrengthBase = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>()
|
2018-08-29 15:38:12 +00:00
|
|
|
.find("fDamageStrengthBase")->mValue.getFloat();
|
2015-03-12 02:08:58 +00:00
|
|
|
static const float fDamageStrengthMult = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>()
|
2018-08-29 15:38:12 +00:00
|
|
|
.find("fDamageStrengthMult")->mValue.getFloat();
|
2015-03-12 02:08:58 +00:00
|
|
|
damage *= fDamageStrengthBase +
|
|
|
|
(attacker.getClass().getCreatureStats(attacker).getAttribute(ESM::Attribute::Strength).getModified() * fDamageStrengthMult * 0.1f);
|
2014-08-02 21:14:17 +00:00
|
|
|
}
|
2014-12-31 22:13:36 +00:00
|
|
|
|
2015-06-26 03:15:07 +00:00
|
|
|
void getHandToHandDamage(const MWWorld::Ptr &attacker, const MWWorld::Ptr &victim, float &damage, bool &healthdmg, float attackStrength)
|
2014-12-31 22:13:36 +00:00
|
|
|
{
|
|
|
|
const MWWorld::ESMStore& store = MWBase::Environment::get().getWorld()->getStore();
|
2018-08-29 15:38:12 +00:00
|
|
|
float minstrike = store.get<ESM::GameSetting>().find("fMinHandToHandMult")->mValue.getFloat();
|
|
|
|
float maxstrike = store.get<ESM::GameSetting>().find("fMaxHandToHandMult")->mValue.getFloat();
|
2015-03-08 04:42:07 +00:00
|
|
|
damage = static_cast<float>(attacker.getClass().getSkill(attacker, ESM::Skill::HandToHand));
|
2015-06-26 03:15:07 +00:00
|
|
|
damage *= minstrike + ((maxstrike-minstrike)*attackStrength);
|
2014-12-31 22:13:36 +00:00
|
|
|
|
|
|
|
MWMechanics::CreatureStats& otherstats = victim.getClass().getCreatureStats(victim);
|
2015-08-20 06:12:37 +00:00
|
|
|
healthdmg = otherstats.isParalyzed()
|
2014-12-31 22:13:36 +00:00
|
|
|
|| otherstats.getKnockedDown();
|
|
|
|
bool isWerewolf = (attacker.getClass().isNpc() && attacker.getClass().getNpcStats(attacker).isWerewolf());
|
2018-07-28 13:10:01 +00:00
|
|
|
|
2018-07-29 02:13:56 +00:00
|
|
|
// Options in the launcher's combo box: unarmedFactorsStrengthComboBox
|
|
|
|
// 0 = Do not factor strength into hand-to-hand combat.
|
|
|
|
// 1 = Factor into werewolf hand-to-hand combat.
|
|
|
|
// 2 = Ignore werewolves.
|
|
|
|
int factorStrength = Settings::Manager::getInt("strength influences hand to hand", "Game");
|
|
|
|
if (factorStrength == 1 || (factorStrength == 2 && !isWerewolf)) {
|
2018-07-28 13:10:01 +00:00
|
|
|
damage *= attacker.getClass().getCreatureStats(attacker).getAttribute(ESM::Attribute::Strength).getModified() / 40.0f;
|
|
|
|
}
|
|
|
|
|
2014-12-31 22:13:36 +00:00
|
|
|
if(isWerewolf)
|
|
|
|
{
|
|
|
|
healthdmg = true;
|
|
|
|
// GLOB instead of GMST because it gets updated during a quest
|
|
|
|
damage *= MWBase::Environment::get().getWorld()->getGlobalFloat("werewolfclawmult");
|
|
|
|
}
|
|
|
|
if(healthdmg)
|
2018-08-29 15:38:12 +00:00
|
|
|
damage *= store.get<ESM::GameSetting>().find("fHandtoHandHealthPer")->mValue.getFloat();
|
2014-12-31 22:13:36 +00:00
|
|
|
|
|
|
|
MWBase::SoundManager *sndMgr = MWBase::Environment::get().getSoundManager();
|
|
|
|
if(isWerewolf)
|
|
|
|
{
|
|
|
|
const ESM::Sound *sound = store.get<ESM::Sound>().searchRandom("WolfHit");
|
|
|
|
if(sound)
|
|
|
|
sndMgr->playSound3D(victim, sound->mId, 1.0f, 1.0f);
|
|
|
|
}
|
2019-07-14 20:56:38 +00:00
|
|
|
else if (!healthdmg)
|
2014-12-31 22:13:36 +00:00
|
|
|
sndMgr->playSound3D(victim, "Hand To Hand Hit", 1.0f, 1.0f);
|
|
|
|
}
|
2015-01-11 01:25:46 +00:00
|
|
|
|
2015-06-26 03:15:07 +00:00
|
|
|
void applyFatigueLoss(const MWWorld::Ptr &attacker, const MWWorld::Ptr &weapon, float attackStrength)
|
2015-01-31 21:28:23 +00:00
|
|
|
{
|
|
|
|
// somewhat of a guess, but using the weapon weight makes sense
|
|
|
|
const MWWorld::Store<ESM::GameSetting>& store = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>();
|
2018-08-29 15:38:12 +00:00
|
|
|
const float fFatigueAttackBase = store.find("fFatigueAttackBase")->mValue.getFloat();
|
|
|
|
const float fFatigueAttackMult = store.find("fFatigueAttackMult")->mValue.getFloat();
|
|
|
|
const float fWeaponFatigueMult = store.find("fWeaponFatigueMult")->mValue.getFloat();
|
2015-01-31 21:28:23 +00:00
|
|
|
CreatureStats& stats = attacker.getClass().getCreatureStats(attacker);
|
|
|
|
MWMechanics::DynamicStat<float> fatigue = stats.getFatigue();
|
|
|
|
const float normalizedEncumbrance = attacker.getClass().getNormalizedEncumbrance(attacker);
|
2017-03-25 18:40:11 +00:00
|
|
|
|
|
|
|
bool godmode = attacker == MWMechanics::getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState();
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
{
|
|
|
|
float fatigueLoss = fFatigueAttackBase + normalizedEncumbrance * fFatigueAttackMult;
|
|
|
|
if (!weapon.isEmpty())
|
|
|
|
fatigueLoss += weapon.getClass().getWeight(weapon) * attackStrength * fWeaponFatigueMult;
|
|
|
|
fatigue.setCurrent(fatigue.getCurrent() - fatigueLoss);
|
|
|
|
stats.setFatigue(fatigue);
|
|
|
|
}
|
2015-01-31 21:28:23 +00:00
|
|
|
}
|
|
|
|
|
2016-11-16 19:15:25 +00:00
|
|
|
float getFightDistanceBias(const MWWorld::Ptr& actor1, const MWWorld::Ptr& actor2)
|
|
|
|
{
|
|
|
|
osg::Vec3f pos1 (actor1.getRefData().getPosition().asVec3());
|
|
|
|
osg::Vec3f pos2 (actor2.getRefData().getPosition().asVec3());
|
|
|
|
|
2020-04-22 13:06:42 +00:00
|
|
|
float d = getAggroDistance(actor1, pos1, pos2);
|
2016-11-16 19:15:25 +00:00
|
|
|
|
|
|
|
static const int iFightDistanceBase = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find(
|
2018-08-29 15:38:12 +00:00
|
|
|
"iFightDistanceBase")->mValue.getInteger();
|
2016-11-16 19:15:25 +00:00
|
|
|
static const float fFightDistanceMultiplier = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find(
|
2018-08-29 15:38:12 +00:00
|
|
|
"fFightDistanceMultiplier")->mValue.getFloat();
|
2016-11-16 19:15:25 +00:00
|
|
|
|
|
|
|
return (iFightDistanceBase - fFightDistanceMultiplier * d);
|
|
|
|
}
|
2019-04-28 12:41:10 +00:00
|
|
|
|
|
|
|
bool isTargetMagicallyHidden(const MWWorld::Ptr& target)
|
|
|
|
{
|
|
|
|
const MagicEffects& magicEffects = target.getClass().getCreatureStats(target).getMagicEffects();
|
|
|
|
return (magicEffects.get(ESM::MagicEffect::Invisibility).getMagnitude() > 0)
|
|
|
|
|| (magicEffects.get(ESM::MagicEffect::Chameleon).getMagnitude() > 75);
|
|
|
|
}
|
2019-10-19 10:40:34 +00:00
|
|
|
|
|
|
|
float getAggroDistance(const MWWorld::Ptr& actor, const osg::Vec3f& lhs, const osg::Vec3f& rhs)
|
|
|
|
{
|
|
|
|
if (canActorMoveByZAxis(actor))
|
|
|
|
return distanceIgnoreZ(lhs, rhs);
|
|
|
|
return distance(lhs, rhs);
|
|
|
|
}
|
2014-01-21 00:01:21 +00:00
|
|
|
}
|