|
|
@ -31,8 +31,6 @@ namespace
|
|
|
|
//chooses an attack depending on probability to avoid uniformity
|
|
|
|
//chooses an attack depending on probability to avoid uniformity
|
|
|
|
ESM::Weapon::AttackType chooseBestAttack(const ESM::Weapon* weapon, MWMechanics::Movement &movement);
|
|
|
|
ESM::Weapon::AttackType chooseBestAttack(const ESM::Weapon* weapon, MWMechanics::Movement &movement);
|
|
|
|
|
|
|
|
|
|
|
|
void getMinMaxAttackDuration(const MWWorld::Ptr& actor, float (*fMinMaxDurations)[2]);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
osg::Vec3f AimDirToMovingTarget(const MWWorld::Ptr& actor, const MWWorld::Ptr& target, const osg::Vec3f& vLastTargetPos,
|
|
|
|
osg::Vec3f AimDirToMovingTarget(const MWWorld::Ptr& actor, const MWWorld::Ptr& target, const osg::Vec3f& vLastTargetPos,
|
|
|
|
float duration, int weapType, float strength);
|
|
|
|
float duration, int weapType, float strength);
|
|
|
|
|
|
|
|
|
|
|
@ -83,7 +81,6 @@ namespace MWMechanics
|
|
|
|
/// \brief This class holds the variables AiCombat needs which are deleted if the package becomes inactive.
|
|
|
|
/// \brief This class holds the variables AiCombat needs which are deleted if the package becomes inactive.
|
|
|
|
struct AiCombatStorage : AiTemporaryBase
|
|
|
|
struct AiCombatStorage : AiTemporaryBase
|
|
|
|
{
|
|
|
|
{
|
|
|
|
float mTimerAttack;
|
|
|
|
|
|
|
|
float mTimerReact;
|
|
|
|
float mTimerReact;
|
|
|
|
float mTimerCombatMove;
|
|
|
|
float mTimerCombatMove;
|
|
|
|
bool mReadyToAttack;
|
|
|
|
bool mReadyToAttack;
|
|
|
@ -95,15 +92,12 @@ namespace MWMechanics
|
|
|
|
boost::shared_ptr<Action> mCurrentAction;
|
|
|
|
boost::shared_ptr<Action> mCurrentAction;
|
|
|
|
float mActionCooldown;
|
|
|
|
float mActionCooldown;
|
|
|
|
float mStrength;
|
|
|
|
float mStrength;
|
|
|
|
float mMinMaxAttackDuration[3][2];
|
|
|
|
|
|
|
|
bool mMinMaxAttackDurationInitialised;
|
|
|
|
|
|
|
|
bool mForceNoShortcut;
|
|
|
|
bool mForceNoShortcut;
|
|
|
|
ESM::Position mShortcutFailPos;
|
|
|
|
ESM::Position mShortcutFailPos;
|
|
|
|
osg::Vec3f mLastActorPos;
|
|
|
|
osg::Vec3f mLastActorPos;
|
|
|
|
MWMechanics::Movement mMovement;
|
|
|
|
MWMechanics::Movement mMovement;
|
|
|
|
|
|
|
|
|
|
|
|
AiCombatStorage():
|
|
|
|
AiCombatStorage():
|
|
|
|
mTimerAttack(0),
|
|
|
|
|
|
|
|
mTimerReact(0),
|
|
|
|
mTimerReact(0),
|
|
|
|
mTimerCombatMove(0),
|
|
|
|
mTimerCombatMove(0),
|
|
|
|
mReadyToAttack(false),
|
|
|
|
mReadyToAttack(false),
|
|
|
@ -115,7 +109,6 @@ namespace MWMechanics
|
|
|
|
mCurrentAction(),
|
|
|
|
mCurrentAction(),
|
|
|
|
mActionCooldown(0),
|
|
|
|
mActionCooldown(0),
|
|
|
|
mStrength(),
|
|
|
|
mStrength(),
|
|
|
|
mMinMaxAttackDurationInitialised(false),
|
|
|
|
|
|
|
|
mForceNoShortcut(false),
|
|
|
|
mForceNoShortcut(false),
|
|
|
|
mLastActorPos(0,0,0),
|
|
|
|
mLastActorPos(0,0,0),
|
|
|
|
mMovement(){}
|
|
|
|
mMovement(){}
|
|
|
@ -233,41 +226,12 @@ namespace MWMechanics
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(smoothTurn(actor, osg::DegreesToRadians(movement.mRotation[0]), 0)) movement.mRotation[0] = 0;
|
|
|
|
if(smoothTurn(actor, osg::DegreesToRadians(movement.mRotation[0]), 0)) movement.mRotation[0] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float attacksPeriod = 1.0f;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ESM::Weapon::AttackType attackType;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool& attack = storage.mAttack;
|
|
|
|
bool& attack = storage.mAttack;
|
|
|
|
bool& readyToAttack = storage.mReadyToAttack;
|
|
|
|
bool& readyToAttack = storage.mReadyToAttack;
|
|
|
|
float& timerAttack = storage.mTimerAttack;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool& minMaxAttackDurationInitialised = storage.mMinMaxAttackDurationInitialised;
|
|
|
|
|
|
|
|
float (&minMaxAttackDuration)[3][2] = storage.mMinMaxAttackDuration;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(readyToAttack)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!minMaxAttackDurationInitialised)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// TODO: this must be updated when a different weapon is equipped
|
|
|
|
|
|
|
|
// TODO: it would be much easier to ask the CharacterController about the current % completion of the weapon wind-up animation
|
|
|
|
|
|
|
|
getMinMaxAttackDuration(actor, minMaxAttackDuration);
|
|
|
|
|
|
|
|
minMaxAttackDurationInitialised = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (timerAttack < 0) attack = false;
|
|
|
|
if (attack && (characterController.getAttackStrength() >= storage.mStrength || characterController.readyToPrepareAttack()))
|
|
|
|
|
|
|
|
|
|
|
|
timerAttack -= duration;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
timerAttack = -attacksPeriod;
|
|
|
|
|
|
|
|
attack = false;
|
|
|
|
attack = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
characterController.setAttackingOrSpell(attack);
|
|
|
|
characterController.setAttackingOrSpell(attack);
|
|
|
|
|
|
|
|
|
|
|
@ -372,32 +336,26 @@ namespace MWMechanics
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
float& strength = storage.mStrength;
|
|
|
|
float& strength = storage.mStrength;
|
|
|
|
// start new attack
|
|
|
|
// start new attack
|
|
|
|
if(readyToAttack)
|
|
|
|
if(readyToAttack && characterController.readyToStartAttack())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(timerAttack <= -attacksPeriod)
|
|
|
|
attack = true; // attack starts just now
|
|
|
|
{
|
|
|
|
characterController.setAttackingOrSpell(attack);
|
|
|
|
attack = true; // attack starts just now
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!distantCombat) attackType = chooseBestAttack(weapon, movement);
|
|
|
|
|
|
|
|
else attackType = ESM::Weapon::AT_Chop; // cause it's =0
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
strength = Misc::Rng::rollClosedProbability();
|
|
|
|
if (!distantCombat)
|
|
|
|
|
|
|
|
chooseBestAttack(weapon, movement);
|
|
|
|
|
|
|
|
|
|
|
|
// Note: may be 0 for some animations
|
|
|
|
strength = Misc::Rng::rollClosedProbability();
|
|
|
|
timerAttack = minMaxAttackDuration[attackType][0] +
|
|
|
|
|
|
|
|
(minMaxAttackDuration[attackType][1] - minMaxAttackDuration[attackType][0]) * strength;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//say a provoking combat phrase
|
|
|
|
//say a provoking combat phrase
|
|
|
|
if (actor.getClass().isNpc())
|
|
|
|
if (actor.getClass().isNpc())
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
const MWWorld::ESMStore &store = world->getStore();
|
|
|
|
|
|
|
|
int chance = store.get<ESM::GameSetting>().find("iVoiceAttackOdds")->getInt();
|
|
|
|
|
|
|
|
if (Misc::Rng::roll0to99() < chance)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
const MWWorld::ESMStore &store = world->getStore();
|
|
|
|
MWBase::Environment::get().getDialogueManager()->say(actor, "attack");
|
|
|
|
int chance = store.get<ESM::GameSetting>().find("iVoiceAttackOdds")->getInt();
|
|
|
|
|
|
|
|
if (Misc::Rng::roll0to99() < chance)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
MWBase::Environment::get().getDialogueManager()->say(actor, "attack");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -607,39 +565,6 @@ namespace MWMechanics
|
|
|
|
readyToAttack = false;
|
|
|
|
readyToAttack = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(!isStuck && distToTarget > rangeAttack && !distantCombat)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
//special run attack; it shouldn't affect melee combat tactics
|
|
|
|
|
|
|
|
if(actorClass.getMovementSettings(actor).mPosition[1] == 1)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
/* check if actor can overcome the distance = distToTarget - attackerWeapRange
|
|
|
|
|
|
|
|
less than in time of swinging with weapon (t_swing), then start attacking
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
float speed1 = actorClass.getSpeed(actor);
|
|
|
|
|
|
|
|
float speed2 = target.getClass().getSpeed(target);
|
|
|
|
|
|
|
|
if(target.getClass().getMovementSettings(target).mPosition[0] == 0
|
|
|
|
|
|
|
|
&& target.getClass().getMovementSettings(target).mPosition[1] == 0)
|
|
|
|
|
|
|
|
speed2 = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
float s1 = distToTarget - weapRange;
|
|
|
|
|
|
|
|
float t = s1/speed1;
|
|
|
|
|
|
|
|
float s2 = speed2 * t;
|
|
|
|
|
|
|
|
float t_swing =
|
|
|
|
|
|
|
|
minMaxAttackDuration[ESM::Weapon::AT_Thrust][0] +
|
|
|
|
|
|
|
|
(minMaxAttackDuration[ESM::Weapon::AT_Thrust][1] - minMaxAttackDuration[ESM::Weapon::AT_Thrust][0]) * Misc::Rng::rollClosedProbability();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (t + s2/speed1 <= t_swing)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
readyToAttack = true;
|
|
|
|
|
|
|
|
if(timerAttack <= -attacksPeriod)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
timerAttack = t_swing;
|
|
|
|
|
|
|
|
attack = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -765,70 +690,6 @@ ESM::Weapon::AttackType chooseBestAttack(const ESM::Weapon* weapon, MWMechanics:
|
|
|
|
return attackType;
|
|
|
|
return attackType;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void getMinMaxAttackDuration(const MWWorld::Ptr& actor, float (*fMinMaxDurations)[2])
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (!actor.getClass().hasInventoryStore(actor)) // creatures
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
fMinMaxDurations[0][0] = fMinMaxDurations[0][1] = 0.1f;
|
|
|
|
|
|
|
|
fMinMaxDurations[1][0] = fMinMaxDurations[1][1] = 0.1f;
|
|
|
|
|
|
|
|
fMinMaxDurations[2][0] = fMinMaxDurations[2][1] = 0.1f;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// get weapon information: type and speed
|
|
|
|
|
|
|
|
const ESM::Weapon *weapon = NULL;
|
|
|
|
|
|
|
|
MWMechanics::WeaponType weaptype = MWMechanics::WeapType_None;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
MWWorld::ContainerStoreIterator weaponSlot =
|
|
|
|
|
|
|
|
MWMechanics::getActiveWeapon(actor.getClass().getCreatureStats(actor), actor.getClass().getInventoryStore(actor), &weaptype);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
float weapSpeed;
|
|
|
|
|
|
|
|
if (weaptype != MWMechanics::WeapType_HandToHand
|
|
|
|
|
|
|
|
&& weaptype != MWMechanics::WeapType_Spell
|
|
|
|
|
|
|
|
&& weaptype != MWMechanics::WeapType_None)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
weapon = weaponSlot->get<ESM::Weapon>()->mBase;
|
|
|
|
|
|
|
|
weapSpeed = weapon->mData.mSpeed;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else weapSpeed = 1.0f;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
MWRender::Animation *anim = MWBase::Environment::get().getWorld()->getAnimation(actor);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::string weapGroup;
|
|
|
|
|
|
|
|
MWMechanics::getWeaponGroup(weaptype, weapGroup);
|
|
|
|
|
|
|
|
weapGroup = weapGroup + ": ";
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool bRangedWeap = (weaptype >= MWMechanics::WeapType_BowAndArrow && weaptype <= MWMechanics::WeapType_Thrown);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const char *attackType[] = {"chop ", "slash ", "thrust ", "shoot "};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::string textKey = "start";
|
|
|
|
|
|
|
|
std::string textKey2;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// get durations for each attack type
|
|
|
|
|
|
|
|
for (int i = 0; i < (bRangedWeap ? 1 : 3); i++)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
float start1 = anim->getTextKeyTime(weapGroup + (bRangedWeap ? attackType[3] : attackType[i]) + textKey);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (start1 < 0)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
fMinMaxDurations[i][0] = fMinMaxDurations[i][1] = 0.1f;
|
|
|
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
textKey2 = "min attack";
|
|
|
|
|
|
|
|
float start2 = anim->getTextKeyTime(weapGroup + (bRangedWeap ? attackType[3] : attackType[i]) + textKey2);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fMinMaxDurations[i][0] = (start2 - start1) / weapSpeed;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
textKey2 = "max attack";
|
|
|
|
|
|
|
|
start1 = anim->getTextKeyTime(weapGroup + (bRangedWeap ? attackType[3] : attackType[i]) + textKey2);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fMinMaxDurations[i][1] = fMinMaxDurations[i][0] + (start1 - start2) / weapSpeed;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
osg::Vec3f AimDirToMovingTarget(const MWWorld::Ptr& actor, const MWWorld::Ptr& target, const osg::Vec3f& vLastTargetPos,
|
|
|
|
osg::Vec3f AimDirToMovingTarget(const MWWorld::Ptr& actor, const MWWorld::Ptr& target, const osg::Vec3f& vLastTargetPos,
|
|
|
|
float duration, int weapType, float strength)
|
|
|
|
float duration, int weapType, float strength)
|
|
|
|
{
|
|
|
|
{
|
|
|
|