raze/source/games/sw/src/ai.cpp

1723 lines
44 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
Copyright (C) 1997, 2005 - 3D Realms Entertainment
This file is part of Shadow Warrior version 1.2
Shadow Warrior is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
Original Source: 1997 - Frank Maddin and Jim Norwood
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
*/
//-------------------------------------------------------------------------
#include "ns.h"
#include "build.h"
#include "names2.h"
#include "game.h"
#include "tags.h"
#include "sector.h"
#include "sprite.h"
// temp - should be moved
#include "ai.h"
#include "network.h"
BEGIN_SW_NS
ANIMATOR InitActorRunToward;
2021-11-21 20:45:12 +00:00
bool DropAhead(DSWActor* actor, int min_height);
2021-12-31 22:09:34 +00:00
ANIMATOR* ChooseAction(DECISION decision[]);
#define CHOOSE2(value) (RANDOM_P2(1024) < (value))
2021-10-30 10:50:20 +00:00
int Distance(int x1, int y1, int x2, int y2)
{
int min;
if ((x2 = x2 - x1) < 0)
x2 = -x2;
if ((y2 = y2 - y1) < 0)
y2 = -y2;
if (x2 > y2)
min = y2;
else
min = x2;
2021-12-27 17:19:30 +00:00
return x2 + y2 - (min >> 1);
}
2021-10-29 22:04:34 +00:00
bool ActorMoveHitReact(DSWActor* actor)
{
// Should only return true if there is a reaction to what was hit that
// would cause the calling function to abort
2021-12-23 22:41:00 +00:00
auto coll = actor->user.coll;
if (coll.type == kHitSprite)
{
auto hitActor = coll.actor();
2021-12-23 22:41:00 +00:00
if (hitActor->hasU() && hitActor->user.PlayerP)
{
// if you ran into a player - call close range functions
2021-10-30 11:05:07 +00:00
DoActorPickClosePlayer(actor);
2021-12-23 22:41:00 +00:00
auto action = ChooseAction(actor->user.Personality->TouchTarget);
if (action)
{
(*action)(actor);
return true;
}
}
}
return false;
}
2021-10-30 10:09:34 +00:00
bool ActorFlaming(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
auto flame = actor->user.flameActor;
if (flame != nullptr)
{
int size;
size = ActorSizeZ(actor) - (ActorSizeZ(actor) >> 2);
if (ActorSizeZ(flame) > size)
return true;
}
return false;
}
2021-10-30 10:09:34 +00:00
void DoActorSetSpeed(DSWActor* actor, uint8_t speed)
{
2021-12-27 18:34:06 +00:00
if (actor->spr.cstat & (CSTAT_SPRITE_RESTORE))
return;
2021-12-23 22:41:00 +00:00
ASSERT(actor->user.Attrib);
2021-12-23 22:41:00 +00:00
actor->user.speed = speed;
2021-10-30 10:09:34 +00:00
if (ActorFlaming(actor))
2021-12-27 17:19:30 +00:00
actor->spr.xvel = actor->user.Attrib->Speed[speed] + (actor->user.Attrib->Speed[speed] >> 1);
else
2021-12-23 22:41:00 +00:00
actor->spr.xvel = actor->user.Attrib->Speed[speed];
}
/*
!AIC - Does a table lookup based on a random value from 0 to 1023.
These tables are defined at the top of all actor files such as ninja.c,
goro.c etc.
*/
2021-12-31 22:09:34 +00:00
ANIMATOR* ChooseAction(DECISION decision[])
{
// !JIM! Here is an opportunity for some AI, instead of randomness!
2021-11-21 20:45:12 +00:00
int random_value = RANDOM_P2(1024<<5)>>5;
2021-11-21 20:45:12 +00:00
for (int i = 0; true; i++)
{
ASSERT(i < 10);
if (random_value <= decision[i].range)
{
return decision[i].action;
}
}
}
/*
!AIC - Sometimes just want the offset of the action
*/
2021-11-21 20:45:12 +00:00
int ChooseActionNumber(int16_t decision[])
{
2021-11-21 20:45:12 +00:00
int random_value = RANDOM_P2(1024<<5)>>5;
2021-11-21 20:45:12 +00:00
for (int i = 0; true; i++)
{
if (random_value <= decision[i])
{
return i;
}
}
}
2021-12-31 22:09:34 +00:00
int DoActorNoise(ANIMATOR* Action, DSWActor* actor)
{
if (Action == InitActorAmbientNoise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_ambient, v3df_follow);
}
else if (Action == InitActorAlertNoise)
{
2021-12-23 22:41:00 +00:00
if (actor->hasU() && !actor->user.DidAlert) // This only allowed once
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_alert, v3df_follow);
}
else if (Action == InitActorAttackNoise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_attack, v3df_follow);
}
else if (Action == InitActorPainNoise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_pain, v3df_follow);
}
else if (Action == InitActorDieNoise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_die, v3df_none);
}
else if (Action == InitActorExtra1Noise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_extra1, v3df_follow);
}
else if (Action == InitActorExtra2Noise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_extra2, v3df_follow);
}
else if (Action == InitActorExtra3Noise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_extra3, v3df_follow);
}
else if (Action == InitActorExtra4Noise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_extra4, v3df_follow);
}
else if (Action == InitActorExtra5Noise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_extra5, v3df_follow);
}
else if (Action == InitActorExtra6Noise)
{
2021-10-31 09:54:52 +00:00
PlaySpriteSound(actor, attr_extra6, v3df_follow);
}
return 0;
}
2021-10-30 10:52:38 +00:00
bool CanSeePlayer(DSWActor* actor)
{
// if actor can still see the player
int look_height = ActorZOfTop(actor);
if (actor->user.targetActor && FAFcansee(actor->int_pos().X, actor->int_pos().Y, look_height, actor->sector(), actor->user.targetActor->int_pos().X, actor->user.targetActor->int_pos().Y, ActorUpperZ(actor->user.targetActor), actor->user.targetActor->sector()))
return true;
else
return false;
}
2021-10-30 11:05:07 +00:00
int CanHitPlayer(DSWActor* actor)
{
HitInfo hit{};
int xvect,yvect,zvect;
2021-11-21 20:45:12 +00:00
int ang;
// if actor can still see the player
int zhs, zhh;
zhs = actor->int_pos().Z - (ActorSizeZ(actor) >> 1);
2021-10-30 12:03:02 +00:00
2021-12-23 22:41:00 +00:00
auto targ = actor->user.targetActor;
// get angle to target
ang = getangle(targ->int_pos().X - actor->int_pos().X, targ->int_pos().Y - actor->int_pos().Y);
// get x,yvect
xvect = bcos(ang);
yvect = bsin(ang);
// get zvect
zhh = targ->int_pos().Z - (ActorSizeZ(targ) >> 1);
if (targ->int_pos().X - actor->int_pos().X != 0)
zvect = xvect * ((zhh - zhs) / (targ->int_pos().X - actor->int_pos().X));
else if (targ->int_pos().Y - actor->int_pos().Y != 0)
zvect = yvect * ((zhh - zhs) / (targ->int_pos().Y - actor->int_pos().Y));
else
return false;
FAFhitscan(actor->int_pos().X, actor->int_pos().Y, zhs, actor->sector(),
xvect,
yvect,
zvect,
hit, CLIPMASK_MISSILE);
if (hit.hitSector == nullptr)
return false;
2021-12-23 22:41:00 +00:00
if (hit.actor() == actor->user.targetActor)
return true;
return false;
}
/*
!AIC - Pick a nearby player to be the actors target
*/
2021-10-30 11:05:07 +00:00
int DoActorPickClosePlayer(DSWActor* actor)
{
int dist, near_dist = MAX_ACTIVE_RANGE, a,b,c;
2021-11-21 20:45:12 +00:00
int pnum;
2021-12-31 14:59:11 +00:00
PLAYER* pp;
// if actor can still see the player
int look_height = ActorZOfTop(actor);
2020-09-09 18:32:24 +00:00
bool found = false;
2020-10-15 15:45:07 +00:00
int i;
2021-12-23 22:41:00 +00:00
if (actor->user.ID == ZOMBIE_RUN_R0 && gNet.MultiGameType == MULTI_GAME_COOPERATIVE)
goto TARGETACTOR;
// Set initial target to Player 0
actor->user.targetActor = Player->actor;
2021-12-27 18:34:06 +00:00
if (actor->user.Flags2 & (SPR2_DONT_TARGET_OWNER))
{
TRAVERSE_CONNECT(pnum)
{
pp = &Player[pnum];
if (GetOwner(actor) == pp->actor)
continue;
2021-12-23 22:41:00 +00:00
actor->user.targetActor = pp->actor;
break;
}
}
// Set initial target to the closest player
near_dist = MAX_ACTIVE_RANGE;
TRAVERSE_CONNECT(pnum)
{
pp = &Player[pnum];
// Zombies don't target their masters!
2021-12-27 18:34:06 +00:00
if (actor->user.Flags2 & (SPR2_DONT_TARGET_OWNER))
{
if (GetOwner(actor) == pp->actor)
continue;
2021-10-31 10:36:58 +00:00
if (!PlayerTakeDamage(pp, actor))
continue;
// if co-op don't hurt teammate
2021-12-23 22:41:00 +00:00
// if (gNet.MultiGameType == MULTI_GAME_COOPERATIVE && !gNet.HurtTeammate && actor->user.spal == pp->actor->spr.spal)
// continue;
}
2022-08-20 10:11:15 +00:00
DISTANCE(actor->int_pos().X, actor->int_pos().Y, pp->int_ppos().X, pp->int_ppos().Y, dist, a, b, c);
if (dist < near_dist)
{
near_dist = dist;
2021-12-23 22:41:00 +00:00
actor->user.targetActor = pp->actor;
}
}
// see if you can find someone close that you can SEE
near_dist = MAX_ACTIVE_RANGE;
found = false;
TRAVERSE_CONNECT(pnum)
{
pp = &Player[pnum];
// Zombies don't target their masters!
2021-12-27 18:34:06 +00:00
if (actor->user.Flags2 & (SPR2_DONT_TARGET_OWNER))
{
if (GetOwner(actor) == pp->actor)
continue;
2021-10-31 10:36:58 +00:00
if (!PlayerTakeDamage(pp, actor))
continue;
}
2022-08-20 10:11:15 +00:00
DISTANCE(actor->int_pos().X, actor->int_pos().Y, pp->int_ppos().X, pp->int_ppos().Y, dist, a, b, c);
DSWActor* plActor = pp->actor;
if (dist < near_dist && FAFcansee(actor->int_pos().X, actor->int_pos().Y, look_height, actor->sector(), plActor->int_pos().X, plActor->int_pos().Y, ActorUpperZ(plActor), plActor->sector()))
{
near_dist = dist;
2021-12-23 22:41:00 +00:00
actor->user.targetActor = pp->actor;
found = true;
}
}
TARGETACTOR:
// this is only for Zombies right now
// zombie target other actors
if (!found && (actor->user.Flags2 & SPR2_DONT_TARGET_OWNER))
{
near_dist = MAX_ACTIVE_RANGE;
2021-10-30 13:28:26 +00:00
SWStatIterator it(STAT_ENEMY);
while (auto itActor = it.Next())
{
2021-10-30 13:28:26 +00:00
if (itActor == actor || !itActor->hasU())
continue;
if ((itActor->user.Flags & (SPR_SUICIDE | SPR_DEAD)))
continue;
DISTANCE(actor->int_pos().X, actor->int_pos().Y, itActor->int_pos().X, itActor->int_pos().Y, dist, a, b, c);
if (dist < near_dist && FAFcansee(actor->int_pos().X, actor->int_pos().Y, look_height, actor->sector(), itActor->int_pos().X, itActor->int_pos().Y, ActorUpperZ(itActor), itActor->sector()))
{
near_dist = dist;
2021-12-23 22:41:00 +00:00
actor->user.targetActor = itActor;
}
}
}
return 0;
}
2021-10-30 12:09:09 +00:00
DSWActor* GetPlayerSpriteNum(DSWActor* actor)
{
2021-11-21 20:45:12 +00:00
int pnum;
2021-12-31 14:59:11 +00:00
PLAYER* pp;
TRAVERSE_CONNECT(pnum)
{
pp = &Player[pnum];
2021-12-23 22:41:00 +00:00
if (pp->actor == actor->user.targetActor)
{
return pp->actor;
}
}
2021-10-30 12:09:09 +00:00
return nullptr;
}
2021-10-30 12:11:47 +00:00
int CloseRangeDist(DSWActor* actor1, DSWActor* actor2)
{
int clip1 = actor1->spr.clipdist;
int clip2 = actor2->spr.clipdist;
// add clip boxes and a fudge factor
2021-10-30 12:11:47 +00:00
const int DIST_CLOSE_RANGE = 400;
2021-10-30 12:11:47 +00:00
return (clip1 << 2) + (clip2 << 2) + DIST_CLOSE_RANGE;
}
2021-10-30 12:11:47 +00:00
int DoActorOperate(DSWActor* actor)
{
2021-11-26 18:30:32 +00:00
HitInfo near{};
int z[2];
unsigned int i;
2021-12-23 22:41:00 +00:00
if (actor->user.ID == HORNET_RUN_R0 || actor->user.ID == EEL_RUN_R0 || actor->user.ID == BUNNY_RUN_R0)
return false;
2021-12-23 22:41:00 +00:00
if (actor->user.Rot == actor->user.ActorActionSet->Sit || actor->user.Rot == actor->user.ActorActionSet->Stand)
return false;
2021-12-23 22:41:00 +00:00
if ((actor->user.WaitTics -= ACTORMOVETICS) > 0)
return false;
z[0] = actor->int_pos().Z - ActorSizeZ(actor) + Z(5);
z[1] = actor->int_pos().Z - (ActorSizeZ(actor) >> 1);
for (i = 0; i < SIZ(z); i++)
{
2022-08-16 21:17:01 +00:00
neartag({ actor->int_pos().X, actor->int_pos().Y, z[i] }, actor->sector(), actor->int_ang(), near, 1024, NTAG_SEARCH_LO_HI);
}
if (near.hitSector != nullptr && near.int_hitpos().X < 1024)
{
2021-11-26 18:30:32 +00:00
if (OperateSector(near.hitSector, false))
{
2021-12-23 22:41:00 +00:00
actor->user.WaitTics = 2 * 120;
2021-12-23 22:41:00 +00:00
NewStateGroup(actor, actor->user.ActorActionSet->Sit);
}
}
return true;
}
DECISION GenericFlaming[] =
{
{30, InitActorAttack},
{512, InitActorRunToward},
{1024, InitActorRunAway},
};
/*
!AIC KEY - This routine decides what the actor will do next. It is not called
every time through the loop. This would be too slow. It is only called when
the actor needs to know what to do next such as running into something or being
targeted. It makes decisions based on the distance and viewablity of its target
2021-12-23 22:41:00 +00:00
(actor->user.targetActor). When it figures out the situatation with its target it calls
ChooseAction which does a random table lookup to decide what action to initialize.
Once this action is initialized it will be called until it can't figure out what to
do anymore and then this routine is called again.
*/
2021-12-31 22:09:34 +00:00
ANIMATOR* DoActorActionDecide(DSWActor* actor)
{
int dist;
2021-12-31 22:09:34 +00:00
ANIMATOR* action;
2020-09-09 18:32:24 +00:00
bool ICanSee=false;
// REMINDER: This function is not even called if SpriteControl doesn't let
// it get called
2021-12-23 22:41:00 +00:00
ASSERT(actor->user.Personality);
2021-12-23 22:41:00 +00:00
actor->user.Dist = 0;
action = InitActorDecide;
// target is gone.
2021-12-23 22:41:00 +00:00
if (actor->user.targetActor == nullptr)
{
return action;
}
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_JUMPING | SPR_FALLING))
{
//CON_Message("Jumping or falling");
return action;
}
// everybody on fire acts like this
2021-10-30 10:09:34 +00:00
if (ActorFlaming(actor))
{
action = ChooseAction(&GenericFlaming[0]);
//CON_Message("On Fire");
return action;
}
2021-10-30 10:52:38 +00:00
ICanSee = CanSeePlayer(actor); // Only need to call once
// But need the result multiple times
// !AIC KEY - If aware of player - var is changed in SpriteControl
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_ACTIVE))
{
// Try to operate stuff
2021-10-30 12:11:47 +00:00
DoActorOperate(actor);
// if far enough away and cannot see the player
2022-08-17 16:59:33 +00:00
dist = DistanceI(actor->spr.pos, actor->user.targetActor->spr.pos);
if (dist > 30000 && !ICanSee)
{
// Enemy goes inactive - he is still allowed to roam about for about
// 5 seconds trying to find another player before his active_range is
// bumped down
2021-12-27 17:58:15 +00:00
actor->user.Flags &= ~(SPR_ACTIVE);
// You've lost the player - now decide what to do
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->LostTarget);
//CON_Message("LostTarget");
return action;
}
2021-10-30 12:09:09 +00:00
auto pActor = GetPlayerSpriteNum(actor);
// check for short range attack possibility
2021-12-23 22:41:00 +00:00
if ((dist < CloseRangeDist(actor, actor->user.targetActor) && ICanSee) ||
2021-12-23 22:45:39 +00:00
(pActor && pActor->hasU() && pActor->user.WeaponNum == WPN_FIST && actor->user.ID != RIPPER2_RUN_R0 && actor->user.ID != RIPPER_RUN_R0))
{
if ((actor->user.ID == COOLG_RUN_R0 && (actor->spr.cstat & CSTAT_SPRITE_TRANSLUCENT)) || (actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Evasive);
else
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->CloseRange);
//CON_Message("CloseRange");
return action;
}
// if player is facing me and I'm being attacked
if (Facing(actor, actor->user.targetActor) && (actor->user.Flags & SPR_ATTACKED) && ICanSee)
{
// if I'm a target - at least one missile comming at me
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_TARGETED))
{
// not going to evade, reset the target bit
2021-12-27 17:58:15 +00:00
actor->user.Flags &= ~(SPR_TARGETED); // as far as actor
// knows, its not a
// target any more
2021-12-23 22:41:00 +00:00
if (actor->user.ActorActionSet->Duck && RANDOM_P2(1024<<8)>>8 < 100)
action = InitActorDuck;
else
{
if ((actor->user.ID == COOLG_RUN_R0 && (actor->spr.cstat & CSTAT_SPRITE_TRANSLUCENT)) || (actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Evasive);
else
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Battle);
}
//CON_Message("Battle 1");
return action;
}
// if NOT a target - don't bother with evasive action and start
// fighting
else
{
if ((actor->user.ID == COOLG_RUN_R0 && (actor->spr.cstat & CSTAT_SPRITE_TRANSLUCENT)) || (actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Evasive);
else
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Battle);
//CON_Message("Battle 2");
return action;
}
}
// if player is NOT facing me he is running or unaware of actor
else if (ICanSee)
{
if ((actor->user.ID == COOLG_RUN_R0 && (actor->spr.cstat & CSTAT_SPRITE_TRANSLUCENT)) || (actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Evasive);
else
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Offense);
//CON_Message("Offense");
return action;
}
else
{
// You've lost the player - now decide what to do
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->LostTarget);
//CON_Message("Close but cant see, LostTarget");
return action;
}
}
// Not active - not aware of player and cannot see him
else
{
// try and find another player
// pick a closeby player as the (new) target
if (actor->spr.hitag != TAG_SWARMSPOT)
2021-10-30 11:05:07 +00:00
DoActorPickClosePlayer(actor);
// if close by
2022-08-17 16:59:33 +00:00
dist = DistanceI(actor->spr.pos, actor->user.targetActor->spr.pos);
if (dist < 15000 || ICanSee)
{
2021-12-23 22:41:00 +00:00
if ((Facing(actor, actor->user.targetActor) && dist < 10000) || ICanSee)
{
2021-10-30 12:11:47 +00:00
DoActorOperate(actor);
// Don't let player completely sneek up behind you
2021-12-23 22:41:00 +00:00
action = ChooseAction(actor->user.Personality->Surprised);
//CON_Message("Surprised");
2021-12-23 22:41:00 +00:00
if (!actor->user.DidAlert && ICanSee)
{
2021-10-30 10:50:20 +00:00
DoActorNoise(InitActorAlertNoise, actor);
2021-12-23 22:41:00 +00:00
actor->user.DidAlert = true;
}
return action;
}
else
{
// Player has not seen actor, to be fair let him know actor
// are there
2021-12-23 22:41:00 +00:00
DoActorNoise(ChooseAction(actor->user.Personality->Broadcast),actor);
//CON_Message("Actor Noise");
return action;
}
}
}
//CON_Message("Couldn't resolve decide, InitActorDecide");
return action;
}
/*
!AIC - Setup to do the decision
*/
int InitActorDecide(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
2021-12-23 22:37:31 +00:00
return DoActorDecide(actor);
}
int DoActorDecide(DSWActor* actor)
{
2021-12-31 22:09:34 +00:00
ANIMATOR* actor_action;
// See what to do next
actor_action = DoActorActionDecide(actor);
// Fix for the GenericFlaming bug for actors that don't have attack states
2021-12-23 22:41:00 +00:00
if (actor_action == InitActorAttack && actor->user.WeaponNum == 0)
return 0; // Just let the actor do as it was doing before in this case
// Target is gone.
2021-12-23 22:41:00 +00:00
if (actor->user.targetActor == nullptr)
return 0;
// zombie is attacking a player
2021-12-23 22:45:39 +00:00
if (actor_action == InitActorAttack && actor->user.ID == ZOMBIE_RUN_R0 && actor->user.targetActor->user.PlayerP)
{
// Don't let zombies shoot at master
2021-12-23 22:41:00 +00:00
if (GetOwner(actor) == actor->user.targetActor)
return 0;
// if this player cannot take damage from this zombie(weapon) return out
2021-12-23 22:45:39 +00:00
if (!PlayerTakeDamage(actor->user.targetActor->user.PlayerP, actor))
return 0;
}
ASSERT(actor_action != nullptr);
if (actor_action != InitActorDecide)
{
// NOT staying put
(*actor_action)(actor);
//CON_Message("DoActorDecide: NOT Staying put");
}
else
{
// Actually staying put
2021-12-23 22:41:00 +00:00
NewStateGroup(actor, actor->user.ActorActionSet->Stand);
//CON_Message("DoActorDecide: Staying put");
}
return 0;
}
// Important note: The functions below are being checked for as state identifiers.
// But they are all identical content wise which makes MSVC merge them together into one.
// Assigning 'sw_snd_scratch' different values makes them different so that merging does not occur.
int sw_snd_scratch = 0;
int InitActorAlertNoise(DSWActor* actor)
{
sw_snd_scratch = 1;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorAmbientNoise(DSWActor* actor)
{
sw_snd_scratch = 2;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorAttackNoise(DSWActor* actor)
{
sw_snd_scratch = 3;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorPainNoise(DSWActor* actor)
{
sw_snd_scratch = 4;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorDieNoise(DSWActor* actor)
{
sw_snd_scratch = 5;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorExtra1Noise(DSWActor* actor)
{
sw_snd_scratch = 6;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorExtra2Noise(DSWActor* actor)
{
sw_snd_scratch = 7;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorExtra3Noise(DSWActor* actor)
{
sw_snd_scratch = 8;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorExtra4Noise(DSWActor* actor)
{
sw_snd_scratch = 9;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorExtra5Noise(DSWActor* actor)
{
sw_snd_scratch = 10;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorExtra6Noise(DSWActor* actor)
{
sw_snd_scratch = 11;
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
/*
!AIC KEY - Routines handle moving toward the player.
*/
int InitActorMoveCloser(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorMoveCloser;
2021-12-23 22:41:00 +00:00
if (actor->user.Rot != actor->user.ActorActionSet->Run)
NewStateGroup(actor, actor->user.ActorActionSet->Run);
2021-12-23 22:41:00 +00:00
(*actor->user.ActorActionFunc)(actor);
return 0;
}
int DoActorCantMoveCloser(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.track = FindTrackToPlayer(actor);
2021-12-23 22:41:00 +00:00
if (actor->user.track >= 0)
{
2022-08-16 21:23:23 +00:00
actor->set_int_ang(getangle((Track[actor->user.track].TrackPoint + actor->user.point)->x - actor->int_pos().X, (Track[actor->user.track].TrackPoint + actor->user.point)->y - actor->int_pos().Y));
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, MID_SPEED);
actor->user.Flags |= (SPR_FIND_PLAYER);
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Run);
}
else
{
// Try to move closer
InitActorReposition(actor);
}
return 0;
}
int DoActorMoveCloser(DSWActor* actor)
{
int nx, ny;
2022-08-16 21:17:01 +00:00
nx = MulScale(actor->spr.xvel, bcos(actor->int_ang()), 14);
ny = MulScale(actor->spr.xvel, bsin(actor->int_ang()), 14);
// if cannot move the sprite
2021-12-23 22:37:31 +00:00
if (!move_actor(actor, nx, ny, 0))
{
2021-10-29 22:04:34 +00:00
if (ActorMoveHitReact(actor))
return 0;
DoActorCantMoveCloser(actor);
return 0;
}
// Do a noise if ok
2021-12-23 22:41:00 +00:00
DoActorNoise(ChooseAction(actor->user.Personality->Broadcast), actor);
// after moving a ways check to see if player is still in sight
2021-12-23 22:41:00 +00:00
if (actor->user.DistCheck > 550)
{
2021-12-23 22:41:00 +00:00
actor->user.DistCheck = 0;
// If player moved out of sight
2021-10-30 10:52:38 +00:00
if (!CanSeePlayer(actor))
{
// stay put and choose another option
InitActorDecide(actor);
return 0;
}
else
{
// turn to face player
2022-08-16 21:23:23 +00:00
actor->set_int_ang(getangle(actor->user.targetActor->int_pos().X - actor->int_pos().X, actor->user.targetActor->int_pos().Y - actor->int_pos().Y));
}
}
// Should be a random value test
2021-12-23 22:41:00 +00:00
if (actor->user.Dist > 512 * 3)
{
InitActorDecide(actor);
}
return 0;
}
/*
!AIC - Find tracks of different types. Toward target, away from target, etc.
*/
2021-11-21 20:47:37 +00:00
int FindTrackToPlayer(DSWActor* actor)
{
2021-11-21 20:45:12 +00:00
int point, track_dir, track;
int i, size;
const uint16_t* type;
int zdiff;
2021-11-21 20:45:12 +00:00
static const uint16_t PlayerAbove[] =
{
BIT(TT_LADDER),
BIT(TT_STAIRS),
BIT(TT_JUMP_UP),
BIT(TT_TRAVERSE),
BIT(TT_OPERATE),
BIT(TT_SCAN)
};
2021-11-21 20:45:12 +00:00
static const uint16_t PlayerBelow[] =
{
BIT(TT_JUMP_DOWN),
BIT(TT_STAIRS),
BIT(TT_TRAVERSE),
BIT(TT_OPERATE),
BIT(TT_SCAN)
};
2021-11-21 20:45:12 +00:00
static const uint16_t PlayerOnLevel[] =
{
BIT(TT_DUCK_N_SHOOT),
BIT(TT_HIDE_N_SHOOT),
BIT(TT_TRAVERSE),
BIT(TT_EXIT),
BIT(TT_OPERATE),
BIT(TT_SCAN)
};
zdiff = ActorUpperZ(actor->user.targetActor) - (actor->int_pos().Z - ActorSizeZ(actor) + Z(8));
2021-12-23 22:37:31 +00:00
if (abs(zdiff) <= Z(20))
{
type = PlayerOnLevel;
size = SIZ(PlayerOnLevel);
}
else
{
if (zdiff < 0)
{
type = PlayerAbove;
size = SIZ(PlayerAbove);
}
else
{
type = PlayerBelow;
size = SIZ(PlayerBelow);
}
}
for (i = 0; i < size; i++)
{
2021-10-31 10:47:52 +00:00
track = ActorFindTrack(actor, 1, type[i], &point, &track_dir);
if (track >= 0)
{
2021-12-23 22:41:00 +00:00
actor->user.point = point;
actor->user.track_dir = track_dir;
Track[track].flags |= (TF_TRACK_OCCUPIED);
return track;
}
}
return -1;
}
2021-11-21 20:47:37 +00:00
int FindTrackAwayFromPlayer(DSWActor* actor)
{
2021-11-21 20:45:12 +00:00
int point, track_dir, track;
unsigned int i;
2021-11-21 20:45:12 +00:00
static const int16_t RunAwayTracks[] =
{
BIT(TT_EXIT),
BIT(TT_LADDER),
BIT(TT_TRAVERSE),
BIT(TT_STAIRS),
BIT(TT_JUMP_UP),
BIT(TT_JUMP_DOWN),
BIT(TT_DUCK_N_SHOOT),
BIT(TT_HIDE_N_SHOOT),
BIT(TT_OPERATE),
BIT(TT_SCAN)
};
for (i = 0; i < SIZ(RunAwayTracks); i++)
{
2021-10-31 10:47:52 +00:00
track = ActorFindTrack(actor, -1, RunAwayTracks[i], &point, &track_dir);
if (track >= 0)
{
2021-12-23 22:41:00 +00:00
actor->user.point = point;
actor->user.track_dir = track_dir;
Track[track].flags |= (TF_TRACK_OCCUPIED);
return track;
}
}
return -1;
}
2021-11-21 20:47:37 +00:00
int FindWanderTrack(DSWActor* actor)
{
2021-11-21 20:45:12 +00:00
int point, track_dir, track;
unsigned int i;
2021-11-21 20:45:12 +00:00
static const int16_t WanderTracks[] =
{
BIT(TT_DUCK_N_SHOOT),
BIT(TT_HIDE_N_SHOOT),
BIT(TT_WANDER),
BIT(TT_JUMP_DOWN),
BIT(TT_JUMP_UP),
BIT(TT_TRAVERSE),
BIT(TT_STAIRS),
BIT(TT_LADDER),
BIT(TT_EXIT),
BIT(TT_OPERATE)
};
for (i = 0; i < SIZ(WanderTracks); i++)
{
2021-10-31 10:47:52 +00:00
track = ActorFindTrack(actor, -1, WanderTracks[i], &point, &track_dir);
if (track >= 0)
{
2021-12-23 22:41:00 +00:00
actor->user.point = point;
actor->user.track_dir = track_dir;
Track[track].flags |= (TF_TRACK_OCCUPIED);
return track;
}
}
return -1;
}
int InitActorRunAway(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Run);
2021-12-23 22:41:00 +00:00
actor->user.track = FindTrackAwayFromPlayer(actor);
2021-12-23 22:41:00 +00:00
if (actor->user.track >= 0)
{
2022-08-16 21:23:23 +00:00
actor->set_int_ang(NORM_ANGLE(getangle((Track[actor->user.track].TrackPoint + actor->user.point)->x - actor->int_pos().X, (Track[actor->user.track].TrackPoint + actor->user.point)->y - actor->int_pos().Y)));
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, FAST_SPEED);
actor->user.Flags |= (SPR_RUN_AWAY);
}
else
{
actor->user.Flags |= (SPR_RUN_AWAY);
InitActorReposition(actor);
}
return 0;
}
int InitActorRunToward(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Run);
InitActorReposition(actor);
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, FAST_SPEED);
return 0;
}
/*
!AIC - Where actors do their attacks. There is some special case code throughout
these. Both close and long range attacks are handled here by transitioning to
the correct attack state.
*/
int InitActorAttack(DSWActor* actor)
{
// zombie is attacking a player
2021-12-23 22:45:39 +00:00
if (actor->user.ID == ZOMBIE_RUN_R0 && actor->user.targetActor->hasU() && actor->user.targetActor->user.PlayerP)
{
// Don't let zombies shoot at master
2021-12-23 22:41:00 +00:00
if (GetOwner(actor) == actor->user.targetActor)
return 0;
// if this player cannot take damage from this zombie(weapon) return out
2021-12-23 22:45:39 +00:00
if (!PlayerTakeDamage(actor->user.targetActor->user.PlayerP, actor))
return 0;
}
if ((actor->user.targetActor->spr.cstat & CSTAT_SPRITE_TRANSLUCENT))
{
InitActorRunAway(actor);
return 0;
}
2021-12-23 22:45:39 +00:00
if (actor->user.targetActor->hasU() && actor->user.targetActor->user.Health <= 0)
{
2021-10-30 11:05:07 +00:00
DoActorPickClosePlayer(actor);
InitActorReposition(actor);
return 0;
}
2021-10-30 10:59:58 +00:00
if (!CanHitPlayer(actor))
{
InitActorReposition(actor);
return 0;
}
// if the guy you are after is dead, look for another and
// reposition
2021-12-23 22:45:39 +00:00
if (actor->user.targetActor->hasU() && actor->user.targetActor->user.PlayerP &&
(actor->user.targetActor->user.PlayerP->Flags & PF_DEAD))
{
2021-10-30 11:05:07 +00:00
DoActorPickClosePlayer(actor);
InitActorReposition(actor);
return 0;
}
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorAttack;
// move into standing frame
2021-12-23 22:41:00 +00:00
//NewStateGroup(actor, actor->user.ActorActionSet->Stand);
// face player when attacking
2022-08-16 21:23:23 +00:00
actor->set_int_ang(NORM_ANGLE(getangle(actor->user.targetActor->int_pos().X - actor->int_pos().X, actor->user.targetActor->int_pos().Y - actor->int_pos().Y)));
// If it's your own kind, lay off!
2021-12-23 22:45:39 +00:00
if (actor->user.ID == actor->user.targetActor->user.ID && !actor->user.targetActor->user.PlayerP)
{
InitActorRunAway(actor);
return 0;
}
// Hari Kari for Ninja's
2021-12-23 22:41:00 +00:00
if (actor->user.ActorActionSet->Death2)
{
//#define SUICIDE_HEALTH_VALUE 26
#define SUICIDE_HEALTH_VALUE 38
//#define SUICIDE_HEALTH_VALUE 50
2021-12-23 22:41:00 +00:00
if (actor->user.Health < SUICIDE_HEALTH_VALUE)
{
if (CHOOSE2(100))
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Death2);
return 0;
}
}
}
2021-12-23 22:41:00 +00:00
(*actor->user.ActorActionFunc)(actor);
return 0;
}
int DoActorAttack(DSWActor* actor)
{
2021-11-21 20:47:37 +00:00
int rand_num;
int dist,a,b,c;
2021-12-23 22:41:00 +00:00
DoActorNoise(ChooseAction(actor->user.Personality->Broadcast),actor);
DISTANCE(actor->int_pos().X, actor->int_pos().Y, actor->user.targetActor->int_pos().X, actor->user.targetActor->int_pos().Y, dist, a, b, c);
2021-10-30 12:09:09 +00:00
auto pActor = GetPlayerSpriteNum(actor);
2021-12-23 22:41:00 +00:00
if ((actor->user.ActorActionSet->CloseAttack[0] && dist < CloseRangeDist(actor, actor->user.targetActor)) ||
2021-12-23 22:45:39 +00:00
(pActor && pActor->hasU() && pActor->user.WeaponNum == WPN_FIST)) // JBF: added null check
{
2021-12-23 22:41:00 +00:00
rand_num = ChooseActionNumber(actor->user.ActorActionSet->CloseAttackPercent);
2021-12-23 22:41:00 +00:00
NewStateGroup(actor, actor->user.ActorActionSet->CloseAttack[rand_num]);
}
else
{
2021-12-23 22:41:00 +00:00
ASSERT(actor->user.WeaponNum != 0);
2021-12-23 22:41:00 +00:00
rand_num = ChooseActionNumber(actor->user.ActorActionSet->AttackPercent);
2021-12-23 22:41:00 +00:00
ASSERT(rand_num < actor->user.WeaponNum);
2021-12-23 22:41:00 +00:00
NewStateGroup(actor, actor->user.ActorActionSet->Attack[rand_num]);
actor->user.ActorActionFunc = DoActorDecide;
}
2021-12-23 22:41:00 +00:00
//actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
int InitActorEvade(DSWActor* actor)
{
2021-12-23 22:37:31 +00:00
// Evade is same thing as run away except when you get to the end of the track
// you stop and take up the fight again.
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Run);
2021-12-23 22:41:00 +00:00
actor->user.track = FindTrackAwayFromPlayer(actor);
2021-12-23 22:41:00 +00:00
if (actor->user.track >= 0)
{
2022-08-16 21:23:23 +00:00
actor->set_int_ang(NORM_ANGLE(getangle((Track[actor->user.track].TrackPoint + actor->user.point)->x - actor->int_pos().X, (Track[actor->user.track].TrackPoint + actor->user.point)->y - actor->int_pos().Y)));
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, FAST_SPEED);
// NOT doing a RUN_AWAY
2021-12-27 17:58:15 +00:00
actor->user.Flags &= ~(SPR_RUN_AWAY);
}
return 0;
}
int InitActorWanderAround(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Run);
2021-10-30 11:05:07 +00:00
DoActorPickClosePlayer(actor);
2021-12-23 22:41:00 +00:00
actor->user.track = FindWanderTrack(actor);
2021-12-23 22:41:00 +00:00
if (actor->user.track >= 0)
{
2022-08-16 21:23:23 +00:00
actor->set_int_ang(getangle((Track[actor->user.track].TrackPoint + actor->user.point)->x - actor->int_pos().X, (Track[actor->user.track].TrackPoint + actor->user.point)->y - actor->int_pos().Y));
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, NORM_SPEED);
}
return 0;
}
int InitActorFindPlayer(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Run);
2021-12-23 22:41:00 +00:00
actor->user.track = FindTrackToPlayer(actor);
2021-12-23 22:41:00 +00:00
if (actor->user.track >= 0)
{
2022-08-16 21:23:23 +00:00
actor->set_int_ang(getangle((Track[actor->user.track].TrackPoint + actor->user.point)->x - actor->int_pos().X, (Track[actor->user.track].TrackPoint + actor->user.point)->y - actor->int_pos().Y));
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, MID_SPEED);
actor->user.Flags |= (SPR_FIND_PLAYER);
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
NewStateGroup(actor, actor->user.ActorActionSet->Run);
}
else
{
InitActorReposition(actor);
}
return 0;
}
int InitActorDuck(DSWActor* actor)
{
2021-11-21 20:47:37 +00:00
int dist;
2021-12-23 22:41:00 +00:00
if (!actor->user.ActorActionSet->Duck)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
return 0;
}
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDuck;
NewStateGroup(actor, actor->user.ActorActionSet->Duck);
2022-08-17 16:59:33 +00:00
dist = DistanceI(actor->spr.pos, actor->user.targetActor->spr.pos);
if (dist > 8000)
{
2021-12-23 22:41:00 +00:00
actor->user.WaitTics = 190;
}
else
{
2021-12-23 22:41:00 +00:00
//actor->user.WaitTics = 120;
actor->user.WaitTics = 60;
}
2021-12-23 22:41:00 +00:00
(*actor->user.ActorActionFunc)(actor);
return 0;
}
int DoActorDuck(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
if ((actor->user.WaitTics -= ACTORMOVETICS) < 0)
{
2021-12-23 22:41:00 +00:00
NewStateGroup(actor, actor->user.ActorActionSet->Rise);
actor->user.ActorActionFunc = DoActorDecide;
2021-12-27 17:58:15 +00:00
actor->user.Flags &= ~(SPR_TARGETED);
}
return 0;
}
int DoActorMoveJump(DSWActor* actor)
{
int nx, ny;
// Move while jumping
2022-08-16 21:17:01 +00:00
nx = MulScale(actor->spr.xvel, bcos(actor->int_ang()), 14);
ny = MulScale(actor->spr.xvel, bsin(actor->int_ang()), 14);
2021-10-30 21:03:26 +00:00
move_actor(actor, nx, ny, 0L);
if (!(actor->user.Flags & (SPR_JUMPING|SPR_FALLING)))
{
InitActorDecide(actor);
}
return 0;
}
2021-11-25 17:52:10 +00:00
Collision move_scan(DSWActor* actor, int ang, int dist, int *stopx, int *stopy, int *stopz)
{
int nx,ny;
uint32_t cliptype = CLIPMASK_ACTOR;
DSWActor* highActor;
DSWActor* lowActor;
2021-11-25 17:52:10 +00:00
sectortype* lo_sectp,* hi_sectp, *ssp;
// moves out a bit but keeps the sprites original postion/sector.
// save off position info
auto pos = actor->spr.pos;
2022-08-20 20:06:00 +00:00
auto sang = actor->spr.angle;
auto loz = actor->user.loz;
auto hiz = actor->user.hiz;
2021-12-23 22:41:00 +00:00
lowActor = actor->user.lowActor;
highActor = actor->user.highActor;
lo_sectp = actor->user.lo_sectp;
hi_sectp = actor->user.hi_sectp;
ssp = actor->sector();
// do the move
2022-08-16 21:23:23 +00:00
actor->set_int_ang(ang);
2022-08-16 21:17:01 +00:00
nx = MulScale(dist, bcos(actor->int_ang()), 14);
ny = MulScale(dist, bsin(actor->int_ang()), 14);
2021-12-23 22:41:00 +00:00
Collision ret = move_sprite(actor, nx, ny, 0, actor->user.ceiling_dist, actor->user.floor_dist, cliptype, 1);
// move_sprite DOES do a getzrange point?
// should I look down with a FAFgetzrange to see where I am?
// remember where it stopped
*stopx = actor->int_pos().X;
*stopy = actor->int_pos().Y;
*stopz = actor->int_pos().Z;
// reset position information
actor->spr.pos = pos;
2022-08-20 20:06:00 +00:00
actor->spr.angle = sang;
2021-12-23 22:41:00 +00:00
actor->user.loz = loz;
actor->user.hiz = hiz;
actor->user.lowActor = lowActor;
actor->user.highActor = highActor;
actor->user.lo_sectp = lo_sectp;
actor->user.hi_sectp = hi_sectp;
2021-11-25 17:52:10 +00:00
ChangeActorSect(actor, ssp);
return ret;
}
2021-10-30 13:20:07 +00:00
enum
{
TOWARD = 1,
AWAY = -1
};
int FindNewAngle(DSWActor* actor, int dir, int DistToMove)
{
2021-11-21 20:47:37 +00:00
static const int16_t toward_angle_delta[4][9] =
{
{ -160, -384, 160, 384, -256, 256, -512, 512, -99},
{ -384, -160, 384, 160, -256, 256, -512, 512, -99},
{ 160, 384, -160, -384, 256, -256, 512, -512, -99},
{ 384, 160, -384, -160, 256, -256, 512, -512, -99}
};
2021-11-21 20:47:37 +00:00
static const int16_t away_angle_delta[4][8] =
{
{ -768, 768, -640, 640, -896, 896, 1024, -99},
{ 768, -768, 640, -640, -896, 896, 1024, -99},
{ 896, -896, -768, 768, -640, 640, 1024, -99},
{ 896, -896, 768, -768, 640, -640, 1024, -99}
};
2021-11-21 20:47:37 +00:00
const int16_t* adp = nullptr;
2021-11-21 20:47:37 +00:00
int new_ang, oang;
int save_ang = -1;
int set;
int dist, stopx, stopy, stopz;
// start out with mininum distance that will be accepted as a move
int save_dist = 500;
// if on fire, run shorter distances
2021-10-30 10:09:34 +00:00
if (ActorFlaming(actor))
DistToMove = (DistToMove >> 2) + (DistToMove >> 3);
// Find angle to from the player
oang = NORM_ANGLE(getangle(actor->user.targetActor->int_pos().X - actor->int_pos().X, actor->user.targetActor->int_pos().Y - actor->int_pos().Y));
// choose a random angle array
switch (dir)
{
case TOWARD:
set = RANDOM_P2(4<<8)>>8;
adp = &toward_angle_delta[set][0];
break;
case AWAY:
set = RANDOM_P2(4<<8)>>8;
2021-10-30 10:59:58 +00:00
if (CanHitPlayer(actor))
{
adp = &toward_angle_delta[set][0];
}
else
{
adp = &away_angle_delta[set][0];
}
break;
default:
2021-10-30 13:20:07 +00:00
Printf("FindNewAngle called with dir=%d!\n",dir);
return 0;
}
for (; *adp != -99; adp++)
{
new_ang = NORM_ANGLE(oang + *adp);
#if 1
// look directly ahead for a ledge
if (!(actor->user.Flags & (SPR_NO_SCAREDZ | SPR_JUMPING | SPR_FALLING | SPR_SWIMMING | SPR_DEAD)))
{
2022-08-16 21:23:23 +00:00
actor->set_int_ang(new_ang);
2021-12-23 22:41:00 +00:00
if (DropAhead(actor, actor->user.lo_step))
{
2022-08-16 21:23:23 +00:00
actor->set_int_ang(oang);
continue;
}
2022-08-16 21:23:23 +00:00
actor->set_int_ang(oang);
}
#endif
// check to see how far we can move
2021-11-25 17:52:10 +00:00
auto ret = move_scan(actor, new_ang, DistToMove, &stopx, &stopy, &stopz);
2021-10-31 09:54:52 +00:00
if (ret.type == kHitNone)
{
// cleanly moved in new direction without hitting something
actor->user.TargetDist = Distance(actor->int_pos().X, actor->int_pos().Y, stopx, stopy);
return new_ang;
}
else
{
// hit something
dist = Distance(actor->int_pos().X, actor->int_pos().Y, stopx, stopy);
if (dist > save_dist)
{
save_ang = new_ang;
save_dist = dist;
}
}
}
if (save_ang != -1)
{
2021-12-23 22:41:00 +00:00
actor->user.TargetDist = save_dist;
// If actor moved to the TargetDist it would look like he was running
// into things.
// To keep this from happening make the TargetDist is less than the
// point you would hit something
2021-12-23 22:41:00 +00:00
if (actor->user.TargetDist > 4000)
actor->user.TargetDist -= 3500;
2022-08-16 21:23:23 +00:00
actor->set_int_ang(save_ang);
return save_ang;
}
return -1;
}
/*
!AIC KEY - Reposition code is called throughout this file. What this does is
pick a new direction close to the target direction (or away from the target
direction if running away) and a distance to move in and tries to move there
with move_scan(). If it hits something it will try again. No movement is
actually acomplished here. This is just testing for clear paths to move in.
Location variables that are changed are saved and reset. FindNewAngle() and
move_scan() are two routines (above) that go with this. This is definately
not called every time through the loop. It would be majorly slow.
*/
2021-10-30 13:20:07 +00:00
int InitActorReposition(DSWActor* actor)
{
2021-11-21 20:47:37 +00:00
int ang;
int rnum;
int dist;
static const int AwayDist[8] =
{
17000,
20000,
26000,
26000,
26000,
32000,
32000,
42000
};
static const int TowardDist[8] =
{
10000,
15000,
20000,
20000,
25000,
30000,
35000,
40000
};
static const int16_t PlayerDist[8] =
{
2000,
3000,
3000,
5000,
5000,
5000,
9000,
9000
};
2021-12-23 22:41:00 +00:00
actor->user.Dist = 0;
rnum = RANDOM_P2(8<<8)>>8;
2022-08-17 16:59:33 +00:00
dist = DistanceI(actor->spr.pos, actor->user.targetActor->spr.pos);
if (dist < PlayerDist[rnum] || (actor->user.Flags & SPR_RUN_AWAY))
{
rnum = RANDOM_P2(8<<8)>>8;
2021-10-30 13:20:07 +00:00
ang = FindNewAngle(actor, AWAY, AwayDist[rnum]);
if (ang == -1)
{
2021-12-23 22:41:00 +00:00
actor->user.Vis = 8;
InitActorPause(actor);
return 0;
}
2022-08-16 21:23:23 +00:00
actor->set_int_ang(ang);
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, FAST_SPEED);
2021-12-27 17:58:15 +00:00
actor->user.Flags &= ~(SPR_RUN_AWAY);
}
else
{
// try to move toward player
rnum = RANDOM_P2(8<<8)>>8;
2021-10-30 13:20:07 +00:00
ang = FindNewAngle(actor, TOWARD, TowardDist[rnum]);
if (ang == -1)
{
// try to move away from player
rnum = RANDOM_P2(8<<8)>>8;
2021-10-30 13:20:07 +00:00
ang = FindNewAngle(actor, AWAY, AwayDist[rnum]);
if (ang == -1)
{
2021-12-23 22:41:00 +00:00
actor->user.Vis = 8;
InitActorPause(actor);
return 0;
}
}
else
{
// pick random speed to move toward the player
if (RANDOM_P2(1024) < 512)
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, NORM_SPEED);
else
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, MID_SPEED);
}
2022-08-16 21:23:23 +00:00
actor->set_int_ang(ang);
}
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorReposition;
if (!(actor->user.Flags & SPR_SWIMMING))
2021-12-23 22:41:00 +00:00
NewStateGroup(actor, actor->user.ActorActionSet->Run);
2021-12-23 22:41:00 +00:00
(*actor->user.ActorActionFunc)(actor);
return 0;
}
2021-10-30 13:20:07 +00:00
int DoActorReposition(DSWActor* actor)
{
int nx, ny;
2022-08-16 21:17:01 +00:00
nx = MulScale(actor->spr.xvel, bcos(actor->int_ang()), 14);
ny = MulScale(actor->spr.xvel, bsin(actor->int_ang()), 14);
// still might hit something and have to handle it.
2021-10-30 21:03:26 +00:00
if (!move_actor(actor, nx, ny, 0L))
{
2021-10-29 22:04:34 +00:00
if (ActorMoveHitReact(actor))
return 0;
2021-12-23 22:41:00 +00:00
actor->user.Vis = 6;
InitActorPause(actor);
return 0;
}
// if close to target distance do a Decision again
2021-12-23 22:41:00 +00:00
if (actor->user.TargetDist < 50)
{
InitActorDecide(actor);
}
return 0;
}
2021-10-30 13:20:07 +00:00
int InitActorPause(DSWActor* actor)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorPause;
2021-12-23 22:41:00 +00:00
(*actor->user.ActorActionFunc)(actor);
return 0;
}
2021-10-30 13:20:07 +00:00
int DoActorPause(DSWActor* actor)
{
// Using Vis instead of WaitTics, var name sucks, but it's the same type
// WaitTics is used by too much other actor code and causes problems here
2021-12-23 22:41:00 +00:00
if ((actor->user.Vis -= ACTORMOVETICS) < 0)
{
2021-12-23 22:41:00 +00:00
actor->user.ActorActionFunc = DoActorDecide;
2021-12-27 17:58:15 +00:00
actor->user.Flags &= ~(SPR_TARGETED);
}
return 0;
}
#include "saveable.h"
static saveable_code saveable_ai_code[] =
{
SAVE_CODE(InitActorDecide),
SAVE_CODE(DoActorDecide),
SAVE_CODE(InitActorAlertNoise),
SAVE_CODE(InitActorAmbientNoise),
SAVE_CODE(InitActorAttackNoise),
SAVE_CODE(InitActorPainNoise),
SAVE_CODE(InitActorDieNoise),
SAVE_CODE(InitActorExtra1Noise),
SAVE_CODE(InitActorExtra2Noise),
SAVE_CODE(InitActorExtra3Noise),
SAVE_CODE(InitActorExtra4Noise),
SAVE_CODE(InitActorExtra5Noise),
SAVE_CODE(InitActorExtra6Noise),
SAVE_CODE(InitActorMoveCloser),
SAVE_CODE(DoActorMoveCloser),
SAVE_CODE(FindTrackToPlayer),
SAVE_CODE(FindTrackAwayFromPlayer),
SAVE_CODE(FindWanderTrack),
SAVE_CODE(InitActorRunAway),
SAVE_CODE(InitActorRunToward),
SAVE_CODE(InitActorAttack),
SAVE_CODE(DoActorAttack),
SAVE_CODE(InitActorEvade),
SAVE_CODE(InitActorWanderAround),
SAVE_CODE(InitActorFindPlayer),
SAVE_CODE(InitActorDuck),
SAVE_CODE(DoActorDuck),
SAVE_CODE(DoActorMoveJump),
SAVE_CODE(InitActorReposition),
SAVE_CODE(DoActorReposition),
SAVE_CODE(DoActorPause)
};
static saveable_data saveable_ai_data[] =
{
SAVE_DATA(GenericFlaming)
};
saveable_module saveable_ai =
{
// code
saveable_ai_code,
SIZ(saveable_ai_code),
// data
saveable_ai_data,
SIZ(saveable_ai_data)
};
END_SW_NS