Implement basic spellcasting AI (Fixes #961)
Select a weapon to attack with in AiCombat and equip it (Fixes #1609, Fixes #1772)deque
parent
c2a91148f3
commit
0fe9612afb
@ -0,0 +1,397 @@
|
||||
#include "aicombataction.hpp"
|
||||
|
||||
#include "../mwbase/environment.hpp"
|
||||
#include "../mwbase/world.hpp"
|
||||
|
||||
#include "../mwworld/class.hpp"
|
||||
#include "../mwworld/esmstore.hpp"
|
||||
#include "../mwworld/inventorystore.hpp"
|
||||
#include "../mwworld/actionequip.hpp"
|
||||
|
||||
#include "../mwmechanics/creaturestats.hpp"
|
||||
|
||||
#include <components/esm/loadench.hpp>
|
||||
#include <components/esm/loadmgef.hpp>
|
||||
|
||||
namespace
|
||||
{
|
||||
|
||||
// RangeTypes using bitflags to allow multiple range types, as can be the case with spells having multiple effects.
|
||||
enum RangeTypes
|
||||
{
|
||||
Self = 0x1,
|
||||
Touch = 0x10,
|
||||
Target = 0x100
|
||||
};
|
||||
|
||||
int getRangeTypes (const ESM::EffectList& effects)
|
||||
{
|
||||
int types = 0;
|
||||
for (std::vector<ESM::ENAMstruct>::const_iterator it = effects.mList.begin(); it != effects.mList.end(); ++it)
|
||||
{
|
||||
if (it->mRange == ESM::RT_Self)
|
||||
types |= Self;
|
||||
else if (it->mRange == ESM::RT_Touch)
|
||||
types |= Touch;
|
||||
else if (it->mRange == ESM::RT_Target)
|
||||
types |= Target;
|
||||
}
|
||||
return types;
|
||||
}
|
||||
|
||||
void suggestCombatRange(int rangeTypes, float& rangeAttack, float& rangeFollow)
|
||||
{
|
||||
if (rangeTypes & Touch)
|
||||
{
|
||||
rangeAttack = 100.f;
|
||||
rangeFollow = 300.f;
|
||||
}
|
||||
else if (rangeTypes & Target)
|
||||
{
|
||||
rangeAttack = 1000.f;
|
||||
rangeFollow = 0.f;
|
||||
}
|
||||
else
|
||||
{
|
||||
// For Self spells, distance doesn't matter, so back away slightly to avoid enemy hits
|
||||
rangeAttack = 600.f;
|
||||
rangeFollow = 0.f;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
namespace MWMechanics
|
||||
{
|
||||
|
||||
float ratePotion (const MWWorld::Ptr &item, const MWWorld::Ptr& actor)
|
||||
{
|
||||
if (item.getTypeName() != typeid(ESM::Potion).name())
|
||||
return 0.f;
|
||||
|
||||
const ESM::Potion* potion = item.get<ESM::Potion>()->mBase;
|
||||
return rateEffects(potion->mEffects, actor, MWWorld::Ptr());
|
||||
}
|
||||
|
||||
float rateWeapon (const MWWorld::Ptr &item, const MWWorld::Ptr& actor, const MWWorld::Ptr& target)
|
||||
{
|
||||
if (item.getTypeName() != typeid(ESM::Weapon).name())
|
||||
return 0.f;
|
||||
|
||||
const ESM::Weapon* weapon = item.get<ESM::Weapon>()->mBase;
|
||||
|
||||
// TODO: Check that we have ammunition if needed
|
||||
|
||||
float rating=0.f;
|
||||
|
||||
if (weapon->mData.mType >= ESM::Weapon::MarksmanBow)
|
||||
{
|
||||
rating = (weapon->mData.mChop[0] + weapon->mData.mChop[1]) / 2.f;
|
||||
}
|
||||
else
|
||||
{
|
||||
for (int i=0; i<2; ++i)
|
||||
{
|
||||
rating += weapon->mData.mSlash[i];
|
||||
rating += weapon->mData.mThrust[i];
|
||||
rating += weapon->mData.mChop[i];
|
||||
}
|
||||
rating /= 6.f;
|
||||
}
|
||||
|
||||
if (item.getClass().hasItemHealth(item))
|
||||
rating *= item.getClass().getItemHealth(item) / float(item.getClass().getItemMaxHealth(item));
|
||||
|
||||
rating *= actor.getClass().getSkill(actor, item.getClass().getEquipmentSkill(item)) / 100.f;
|
||||
|
||||
if (!weapon->mEnchant.empty())
|
||||
{
|
||||
const ESM::Enchantment* enchantment = MWBase::Environment::get().getWorld()->getStore().get<ESM::Enchantment>().find(weapon->mEnchant);
|
||||
if (enchantment->mData.mType == ESM::Enchantment::WhenStrikes
|
||||
&& (item.getCellRef().getEnchantmentCharge() == -1
|
||||
|| item.getCellRef().getEnchantmentCharge() >= enchantment->mData.mCost))
|
||||
rating += rateEffects(enchantment->mEffects, actor, target);
|
||||
}
|
||||
return rating;
|
||||
}
|
||||
|
||||
float rateSpell(const ESM::Spell *spell, const MWWorld::Ptr &actor, const MWWorld::Ptr& target)
|
||||
{
|
||||
const CreatureStats& stats = actor.getClass().getCreatureStats(actor);
|
||||
|
||||
// Never casting racial spells (ST_Power and F_Always)
|
||||
if (spell->mData.mType != ESM::Spell::ST_Spell || spell->mData.mFlags & ESM::Spell::F_Always)
|
||||
return 0.f;
|
||||
|
||||
if (spell->mData.mCost > stats.getMagicka().getCurrent())
|
||||
return 0.f;
|
||||
|
||||
// Spells don't stack, so early out if the spell is still active on the target
|
||||
int types = getRangeTypes(spell->mEffects);
|
||||
if ((types & Self) && stats.getActiveSpells().isSpellActive(spell->mId))
|
||||
return 0.f;
|
||||
if ( ((types & Touch) || (types & Target)) && target.getClass().getCreatureStats(target).getActiveSpells().isSpellActive(spell->mId))
|
||||
return 0.f;
|
||||
|
||||
return rateEffects(spell->mEffects, actor, target);
|
||||
}
|
||||
|
||||
float rateMagicItem(const MWWorld::Ptr &ptr, const MWWorld::Ptr &actor, const MWWorld::Ptr& target)
|
||||
{
|
||||
if (ptr.getClass().getEnchantment(ptr).empty())
|
||||
return 0.f;
|
||||
|
||||
const ESM::Enchantment* enchantment = MWBase::Environment::get().getWorld()->getStore().get<ESM::Enchantment>().find(ptr.getClass().getEnchantment(ptr));
|
||||
if (enchantment->mData.mType == ESM::Enchantment::CastOnce)
|
||||
{
|
||||
return rateEffects(enchantment->mEffects, actor, target);
|
||||
}
|
||||
else
|
||||
return 0.f;
|
||||
}
|
||||
|
||||
float rateEffect(const ESM::ENAMstruct &effect, const MWWorld::Ptr &actor, const MWWorld::Ptr &target)
|
||||
{
|
||||
// NOTE: target may be empty
|
||||
|
||||
float baseRating = 1;
|
||||
switch (effect.mEffectID)
|
||||
{
|
||||
case ESM::MagicEffect::Soultrap:
|
||||
case ESM::MagicEffect::AlmsiviIntervention:
|
||||
case ESM::MagicEffect::DivineIntervention:
|
||||
case ESM::MagicEffect::CalmCreature:
|
||||
case ESM::MagicEffect::CalmHumanoid:
|
||||
case ESM::MagicEffect::Charm:
|
||||
case ESM::MagicEffect::DetectAnimal:
|
||||
case ESM::MagicEffect::DetectEnchantment:
|
||||
case ESM::MagicEffect::DetectKey:
|
||||
case ESM::MagicEffect::FrenzyCreature:
|
||||
case ESM::MagicEffect::FrenzyHumanoid:
|
||||
case ESM::MagicEffect::Telekinesis:
|
||||
case ESM::MagicEffect::Mark:
|
||||
case ESM::MagicEffect::Recall:
|
||||
case ESM::MagicEffect::Jump:
|
||||
case ESM::MagicEffect::WaterBreathing:
|
||||
case ESM::MagicEffect::SwiftSwim:
|
||||
case ESM::MagicEffect::WaterWalking:
|
||||
case ESM::MagicEffect::SlowFall:
|
||||
case ESM::MagicEffect::Light:
|
||||
case ESM::MagicEffect::Lock:
|
||||
case ESM::MagicEffect::Open:
|
||||
case ESM::MagicEffect::TurnUndead:
|
||||
return 0.f;
|
||||
case ESM::MagicEffect::Feather:
|
||||
return 0.f; // TODO: check if target is overencumbered
|
||||
case ESM::MagicEffect::Levitate:
|
||||
return 0.f; // AI isn't designed to take advantage of this, and could be perceived as unfair anyway
|
||||
// TODO: check if Beast race (can't wear boots or helm)
|
||||
/*
|
||||
case ESM::MagicEffect::BoundBoots:
|
||||
case ESM::MagicEffect::BoundHelm:
|
||||
*/
|
||||
case ESM::MagicEffect::RestoreHealth:
|
||||
case ESM::MagicEffect::RestoreMagicka:
|
||||
case ESM::MagicEffect::RestoreFatigue:
|
||||
if (effect.mRange == ESM::RT_Self)
|
||||
{
|
||||
const DynamicStat<float>& current = actor.getClass().getCreatureStats(actor).
|
||||
getDynamic(effect.mEffectID - ESM::MagicEffect::RestoreHealth);
|
||||
float toHeal = (effect.mMagnMin + effect.mMagnMax)/2.f * effect.mDuration;
|
||||
// Effect doesn't heal more than we need, *or* we are below 1/2 health
|
||||
if (current.getModified() - current.getCurrent() > toHeal
|
||||
|| current.getCurrent() < current.getModified()*0.5)
|
||||
return 10000.f;
|
||||
else
|
||||
return -10000.f; // Save for later
|
||||
}
|
||||
break;
|
||||
|
||||
// Give a small boost to all direct damage effects. This is combat, after all!
|
||||
case ESM::MagicEffect::FireDamage:
|
||||
case ESM::MagicEffect::ShockDamage:
|
||||
case ESM::MagicEffect::FrostDamage:
|
||||
case ESM::MagicEffect::Poison:
|
||||
case ESM::MagicEffect::AbsorbHealth:
|
||||
case ESM::MagicEffect::DamageHealth:
|
||||
baseRating *= 4;
|
||||
break;
|
||||
|
||||
case ESM::MagicEffect::Paralyze: // *Evil laughter*
|
||||
baseRating *= 5;
|
||||
break;
|
||||
|
||||
// TODO: rate these effects very high if we are currently suffering from negative effects that could be cured
|
||||
case ESM::MagicEffect::Dispel:
|
||||
case ESM::MagicEffect::CureParalyzation:
|
||||
case ESM::MagicEffect::CurePoison:
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
// TODO: for non-cumulative effects (e.g. paralyze), check if the target is already suffering from them
|
||||
|
||||
// TODO: could take into account target's resistance/weakness against the effect
|
||||
|
||||
const ESM::MagicEffect* magicEffect = MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().find(effect.mEffectID);
|
||||
|
||||
baseRating *= magicEffect->mData.mBaseCost;
|
||||
|
||||
if (magicEffect->mData.mFlags & ESM::MagicEffect::NoMagnitude)
|
||||
return 0.f; // No clue how useful this would be; will need special cases for each effect
|
||||
|
||||
float rating = baseRating * (effect.mMagnMin + effect.mMagnMax)/2.f;
|
||||
if (!(magicEffect->mData.mFlags & ESM::MagicEffect::NoDuration))
|
||||
rating *= effect.mDuration;
|
||||
|
||||
if (magicEffect->mData.mFlags & ESM::MagicEffect::Harmful)
|
||||
rating *= -1.f;
|
||||
|
||||
// Currently treating all "on target" or "on touch" effects to target the enemy actor.
|
||||
// Combat AI is egoistic, so doesn't consider applying positive effects to friendly actors.
|
||||
if (effect.mRange != ESM::RT_Self)
|
||||
rating *= -1.f;
|
||||
return rating;
|
||||
}
|
||||
|
||||
float rateEffects(const ESM::EffectList &list, const MWWorld::Ptr& actor, const MWWorld::Ptr& target)
|
||||
{
|
||||
// NOTE: target may be empty
|
||||
float rating = 0.f;
|
||||
for (std::vector<ESM::ENAMstruct>::const_iterator it = list.mList.begin(); it != list.mList.end(); ++it)
|
||||
{
|
||||
rating += rateEffect(*it, actor, target);
|
||||
}
|
||||
return rating;
|
||||
}
|
||||
|
||||
void ActionSpell::prepare(const MWWorld::Ptr &actor)
|
||||
{
|
||||
actor.getClass().getCreatureStats(actor).getSpells().setSelectedSpell(mSpellId);
|
||||
actor.getClass().getCreatureStats(actor).setDrawState(DrawState_Spell);
|
||||
if (actor.getClass().hasInventoryStore(actor))
|
||||
{
|
||||
MWWorld::InventoryStore& inv = actor.getClass().getInventoryStore(actor);
|
||||
inv.setSelectedEnchantItem(inv.end());
|
||||
}
|
||||
}
|
||||
|
||||
void ActionSpell::getCombatRange(float& rangeAttack, float& rangeFollow)
|
||||
{
|
||||
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(mSpellId);
|
||||
int types = getRangeTypes(spell->mEffects);
|
||||
suggestCombatRange(types, rangeAttack, rangeFollow);
|
||||
}
|
||||
|
||||
void ActionEnchantedItem::prepare(const MWWorld::Ptr &actor)
|
||||
{
|
||||
actor.getClass().getCreatureStats(actor).getSpells().setSelectedSpell(std::string());
|
||||
actor.getClass().getInventoryStore(actor).setSelectedEnchantItem(mItem);
|
||||
actor.getClass().getCreatureStats(actor).setDrawState(DrawState_Spell);
|
||||
}
|
||||
|
||||
void ActionEnchantedItem::getCombatRange(float& rangeAttack, float& rangeFollow)
|
||||
{
|
||||
const ESM::Enchantment* enchantment = MWBase::Environment::get().getWorld()->getStore().get<ESM::Enchantment>().find(mItem->getClass().getEnchantment(*mItem));
|
||||
int types = getRangeTypes(enchantment->mEffects);
|
||||
suggestCombatRange(types, rangeAttack, rangeFollow);
|
||||
}
|
||||
|
||||
void ActionPotion::getCombatRange(float& rangeAttack, float& rangeFollow)
|
||||
{
|
||||
// distance doesn't matter, so back away slightly to avoid enemy hits
|
||||
rangeAttack = 600.f;
|
||||
rangeFollow = 0.f;
|
||||
}
|
||||
|
||||
void ActionPotion::prepare(const MWWorld::Ptr &actor)
|
||||
{
|
||||
actor.getClass().apply(actor, mPotion.getCellRef().getRefId(), actor);
|
||||
actor.getClass().getContainerStore(actor).remove(mPotion, 1, actor);
|
||||
}
|
||||
|
||||
void ActionWeapon::prepare(const MWWorld::Ptr &actor)
|
||||
{
|
||||
if (actor.getClass().hasInventoryStore(actor))
|
||||
{
|
||||
if (mWeapon.isEmpty())
|
||||
actor.getClass().getInventoryStore(actor).unequipSlot(MWWorld::InventoryStore::Slot_CarriedRight, actor);
|
||||
else
|
||||
{
|
||||
MWWorld::ActionEquip equip(mWeapon);
|
||||
equip.execute(actor);
|
||||
}
|
||||
// TODO: equip ammunition and shield where needed
|
||||
}
|
||||
actor.getClass().getCreatureStats(actor).setDrawState(DrawState_Weapon);
|
||||
}
|
||||
|
||||
void ActionWeapon::getCombatRange(float& rangeAttack, float& rangeFollow)
|
||||
{
|
||||
// Already done in AiCombat itself
|
||||
}
|
||||
|
||||
boost::shared_ptr<Action> prepareNextAction(const MWWorld::Ptr &actor, const MWWorld::Ptr &target)
|
||||
{
|
||||
Spells& spells = actor.getClass().getCreatureStats(actor).getSpells();
|
||||
|
||||
float bestActionRating = 0.f;
|
||||
// Default to hand-to-hand combat
|
||||
boost::shared_ptr<Action> bestAction (new ActionWeapon(MWWorld::Ptr()));
|
||||
|
||||
if (actor.getClass().hasInventoryStore(actor))
|
||||
{
|
||||
MWWorld::InventoryStore& store = actor.getClass().getInventoryStore(actor);
|
||||
|
||||
for (MWWorld::ContainerStoreIterator it = store.begin(); it != store.end(); ++it)
|
||||
{
|
||||
float rating = ratePotion(*it, actor);
|
||||
if (rating > bestActionRating)
|
||||
{
|
||||
bestActionRating = rating;
|
||||
bestAction.reset(new ActionPotion(*it));
|
||||
}
|
||||
}
|
||||
|
||||
for (MWWorld::ContainerStoreIterator it = store.begin(); it != store.end(); ++it)
|
||||
{
|
||||
float rating = rateMagicItem(*it, actor, target);
|
||||
if (rating > bestActionRating)
|
||||
{
|
||||
bestActionRating = rating;
|
||||
bestAction.reset(new ActionEnchantedItem(it));
|
||||
}
|
||||
}
|
||||
|
||||
for (MWWorld::ContainerStoreIterator it = store.begin(); it != store.end(); ++it)
|
||||
{
|
||||
float rating = rateWeapon(*it, actor, target);
|
||||
if (rating > bestActionRating)
|
||||
{
|
||||
bestActionRating = rating;
|
||||
bestAction.reset(new ActionWeapon(*it));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (Spells::TIterator it = spells.begin(); it != spells.end(); ++it)
|
||||
{
|
||||
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(it->first);
|
||||
|
||||
float rating = rateSpell(spell, actor, target);
|
||||
if (rating > bestActionRating)
|
||||
{
|
||||
bestActionRating = rating;
|
||||
bestAction.reset(new ActionSpell(spell->mId));
|
||||
}
|
||||
}
|
||||
|
||||
if (bestAction.get())
|
||||
bestAction->prepare(actor);
|
||||
|
||||
return bestAction;
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,83 @@
|
||||
#ifndef OPENMW_AICOMBAT_ACTION_H
|
||||
#define OPENMW_AICOMBAT_ACTION_H
|
||||
|
||||
#include <boost/shared_ptr.hpp>
|
||||
|
||||
#include "../mwworld/ptr.hpp"
|
||||
#include "../mwworld/containerstore.hpp"
|
||||
|
||||
#include <components/esm/loadspel.hpp>
|
||||
|
||||
namespace MWMechanics
|
||||
{
|
||||
|
||||
class Action
|
||||
{
|
||||
public:
|
||||
virtual void prepare(const MWWorld::Ptr& actor) = 0;
|
||||
virtual void getCombatRange (float& rangeAttack, float& rangeFollow) = 0;
|
||||
virtual float getActionCooldown() { return 0.f; }
|
||||
};
|
||||
|
||||
class ActionSpell : public Action
|
||||
{
|
||||
public:
|
||||
ActionSpell(const std::string& spellId) : mSpellId(spellId) {}
|
||||
std::string mSpellId;
|
||||
/// Sets the given spell as selected on the actor's spell list.
|
||||
virtual void prepare(const MWWorld::Ptr& actor);
|
||||
|
||||
virtual void getCombatRange (float& rangeAttack, float& rangeFollow);
|
||||
};
|
||||
|
||||
class ActionEnchantedItem : public Action
|
||||
{
|
||||
public:
|
||||
ActionEnchantedItem(const MWWorld::ContainerStoreIterator& item) : mItem(item) {}
|
||||
MWWorld::ContainerStoreIterator mItem;
|
||||
/// Sets the given item as selected enchanted item in the actor's InventoryStore.
|
||||
virtual void prepare(const MWWorld::Ptr& actor);
|
||||
virtual void getCombatRange (float& rangeAttack, float& rangeFollow);
|
||||
|
||||
/// Since this action has no animation, apply a small cool down for using it
|
||||
virtual float getActionCooldown() { return 1.f; }
|
||||
};
|
||||
|
||||
class ActionPotion : public Action
|
||||
{
|
||||
public:
|
||||
ActionPotion(const MWWorld::Ptr& potion) : mPotion(potion) {}
|
||||
MWWorld::Ptr mPotion;
|
||||
/// Drinks the given potion.
|
||||
virtual void prepare(const MWWorld::Ptr& actor);
|
||||
virtual void getCombatRange (float& rangeAttack, float& rangeFollow);
|
||||
|
||||
/// Since this action has no animation, apply a small cool down for using it
|
||||
virtual float getActionCooldown() { return 1.f; }
|
||||
};
|
||||
|
||||
class ActionWeapon : public Action
|
||||
{
|
||||
public:
|
||||
/// \a weapon may be empty for hand-to-hand combat
|
||||
ActionWeapon(const MWWorld::Ptr& weapon) : mWeapon(weapon) {}
|
||||
MWWorld::Ptr mWeapon;
|
||||
/// Equips the given weapon.
|
||||
virtual void prepare(const MWWorld::Ptr& actor);
|
||||
virtual void getCombatRange (float& rangeAttack, float& rangeFollow);
|
||||
};
|
||||
|
||||
float rateSpell (const ESM::Spell* spell, const MWWorld::Ptr& actor, const MWWorld::Ptr& target);
|
||||
float rateMagicItem (const MWWorld::Ptr& ptr, const MWWorld::Ptr& actor, const MWWorld::Ptr &target);
|
||||
float ratePotion (const MWWorld::Ptr& item, const MWWorld::Ptr &actor);
|
||||
float rateWeapon (const MWWorld::Ptr& item, const MWWorld::Ptr& actor);
|
||||
|
||||
/// @note target may be empty
|
||||
float rateEffect (const ESM::ENAMstruct& effect, const MWWorld::Ptr& actor, const MWWorld::Ptr& target);
|
||||
/// @note target may be empty
|
||||
float rateEffects (const ESM::EffectList& list, const MWWorld::Ptr& actor, const MWWorld::Ptr& target);
|
||||
|
||||
boost::shared_ptr<Action> prepareNextAction (const MWWorld::Ptr& actor, const MWWorld::Ptr& target);
|
||||
}
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue