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

658 lines
16 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 "ai.h"
#include "weapon.h"
2020-08-05 22:18:45 +00:00
#include "misc.h"
#include "sprite.h"
BEGIN_SW_NS
DECISION HornetBattle[] =
{
2023-05-24 17:40:05 +00:00
{50, &AF(InitHornetCircle ) },
{798, &AF(InitActorMoveCloser) },
{800, &AF(InitActorSetDecide) },
{1024, &AF(InitActorRunAway ) }
};
DECISION HornetOffense[] =
{
2023-05-24 17:40:05 +00:00
{1022, &AF(InitActorMoveCloser) },
{1024, &AF(InitActorSetDecide) }
};
DECISIONB HornetBroadcast[] =
{
{3, attr_alert },
{6, attr_ambient },
{1024, 0 }
};
DECISION HornetSurprised[] =
{
2023-05-24 17:40:05 +00:00
{100, &AF(InitHornetCircle ) },
{701, &AF(InitActorMoveCloser) },
{1024, &AF(InitActorDecide ) }
};
DECISION HornetEvasive[] =
{
2023-05-24 17:40:05 +00:00
{20, &AF(InitHornetCircle) },
{1024, nullptr },
};
DECISION HornetLostTarget[] =
{
2023-05-24 17:40:05 +00:00
{900, &AF(InitActorFindPlayer) },
{1024, &AF(InitActorWanderAround) }
};
DECISION HornetCloseRange[] =
{
2023-05-24 17:40:05 +00:00
{900, &AF(InitActorMoveCloser) },
{1024, &AF(InitActorReposition) }
};
DECISION HornetTouchTarget[] =
{
2023-05-24 17:40:05 +00:00
{500, &AF(InitHornetCircle) },
{1024, &AF(InitHornetSting ) }
};
PERSONALITY HornetPersonality =
{
HornetBattle,
HornetOffense,
HornetBroadcast,
HornetSurprised,
HornetEvasive,
HornetLostTarget,
HornetCloseRange,
HornetTouchTarget
};
ATTRIBUTE HornetAttrib =
{
{300, 350, 375, 400}, // Speeds
{0, 0, 0, 0}, // Tic Adjusts
0, //MaxWeapons;
{
0, 0, DIGI_HORNETSTING, DIGI_HORNETSTING, DIGI_HORNETDEATH,
0,0,0,0,0
}
};
//////////////////////
//
// HORNET RUN
//////////////////////
#define HORNET_RUN_RATE 7
STATE s_HornetRun[5][2] =
{
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R0 + 0, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[0][1]},
{HORNET_RUN_R0 + 1, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[0][0]},
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R1 + 0, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[1][1]},
{HORNET_RUN_R1 + 1, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[1][0]},
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R2 + 0, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[2][1]},
{HORNET_RUN_R2 + 1, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[2][0]},
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R3 + 0, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[3][1]},
{HORNET_RUN_R3 + 1, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[3][0]},
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R4 + 0, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[4][1]},
{HORNET_RUN_R4 + 1, HORNET_RUN_RATE, &AF(DoHornetMove), &s_HornetRun[4][0]},
}
};
2021-12-31 15:00:14 +00:00
STATE* sg_HornetRun[] =
{
&s_HornetRun[0][0],
&s_HornetRun[1][0],
&s_HornetRun[2][0],
&s_HornetRun[3][0],
&s_HornetRun[4][0]
};
//////////////////////
//
// HORNET STAND
//
//////////////////////
#define HORNET_STAND_RATE (HORNET_RUN_RATE + 5)
STATE s_HornetStand[5][2] =
{
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R0 + 0, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[0][1]},
{HORNET_RUN_R0 + 1, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[0][0]}
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R1 + 0, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[1][1]},
{HORNET_RUN_R1 + 1, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[1][0]}
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R2 + 0, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[2][1]},
{HORNET_RUN_R2 + 1, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[2][0]}
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R3 + 0, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[3][1]},
{HORNET_RUN_R3 + 1, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[3][0]}
},
{
2023-05-24 17:40:05 +00:00
{HORNET_RUN_R4 + 0, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[4][1]},
{HORNET_RUN_R4 + 1, HORNET_STAND_RATE, &AF(DoHornetMove), &s_HornetStand[4][0]}
}
};
2021-12-31 15:00:14 +00:00
STATE* sg_HornetStand[] =
{
&s_HornetStand[0][0],
&s_HornetStand[1][0],
&s_HornetStand[2][0],
&s_HornetStand[3][0],
&s_HornetStand[4][0]
};
//////////////////////
//
// HORNET DIE
//
//////////////////////
#define HORNET_DIE_RATE 20
STATE s_HornetDie[] =
{
2023-05-24 17:40:05 +00:00
{HORNET_DIE + 0, HORNET_DIE_RATE, &AF(DoHornetDeath), &s_HornetDie[0]},
};
2021-12-31 15:00:14 +00:00
STATE* sg_HornetDie[] =
{
s_HornetDie
};
STATE s_HornetDead[] =
{
2023-05-24 17:40:05 +00:00
{HORNET_DEAD, HORNET_DIE_RATE, &AF(DoActorDebris), &s_HornetDead[0]},
};
2021-12-31 15:00:14 +00:00
STATE* sg_HornetDead[] =
{
s_HornetDead
};
/*
2021-12-31 15:00:14 +00:00
STATE* *Stand[MAX_WEAPONS];
STATE* *Run;
STATE* *Jump;
STATE* *Fall;
STATE* *Crawl;
STATE* *Swim;
STATE* *Fly;
STATE* *Rise;
STATE* *Sit;
STATE* *Look;
STATE* *Climb;
STATE* *Pain;
STATE* *Death1;
STATE* *Death2;
STATE* *Dead;
STATE* *DeathJump;
STATE* *DeathFall;
STATE* *CloseAttack[2];
STATE* *Attack[6];
STATE* *Special[2];
*/
ACTOR_ACTION_SET HornetActionSet =
{
sg_HornetStand,
sg_HornetRun,
nullptr,
nullptr,
nullptr,
nullptr,
nullptr,
nullptr,
nullptr,
nullptr,
nullptr, //climb
nullptr, //pain
sg_HornetDie,
nullptr,
sg_HornetDead,
nullptr,
nullptr,
{nullptr},
{0},
{nullptr},
{0},
{nullptr},
nullptr,
nullptr
};
2021-11-01 07:04:23 +00:00
int DoHornetMatchPlayerZ(DSWActor* actor);
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2021-11-01 07:04:23 +00:00
int SetupHornet(DSWActor* actor)
{
if (!(actor->spr.cstat & CSTAT_SPRITE_RESTORE))
{
2021-12-25 20:12:39 +00:00
SpawnUser(actor,HORNET_RUN_R0,s_HornetRun[0]);
2021-12-25 20:11:18 +00:00
actor->user.Health = HEALTH_HORNET;
}
2021-10-30 18:11:31 +00:00
ChangeState(actor, s_HornetRun[0]);
actor->user.__legacyState.Attrib = &HornetAttrib;
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, NORM_SPEED);
actor->user.__legacyState.StateEnd = s_HornetDie;
actor->user.__legacyState.Rot = sg_HornetRun;
2021-10-31 20:17:31 +00:00
EnemyDefaults(actor, &HornetActionSet, &HornetPersonality);
actor->user.Flags |= (SPR_NO_SCAREDZ|SPR_XFLIP_TOGGLE);
actor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
actor->clipdist = 6.25;
actor->user.floor_dist = (16);
actor->user.ceiling_dist = (16);
actor->user.pos.Z = actor->spr.pos.Z;
actor->spr.scale = DVector2(0.578125, 0.5);
// Special looping buzz sound attached to each hornet spawned
PlaySound(DIGI_HORNETBUZZ, actor, v3df_follow|v3df_init);
return 0;
}
DEFINE_ACTION_FUNCTION(DSWHornet, Initialize)
{
PARAM_SELF_PROLOGUE(DSWActor);
SetupHornet(self);
return 0;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int NullHornet(DSWActor* actor)
{
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_SLIDING))
DoActorSlide(actor);
2021-11-01 07:04:23 +00:00
DoHornetMatchPlayerZ(actor);
DoActorSectorDamage(actor);
return 0;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-08-21 08:12:24 +00:00
static const int HORNET_BOB_AMT = 16;
2021-11-01 07:04:23 +00:00
int DoHornetMatchPlayerZ(DSWActor* actor)
{
2021-12-25 20:11:18 +00:00
// actor does a sine wave about actor->user.sz - this is the z mid point
2022-08-21 08:12:24 +00:00
double zdiff = (ActorZOfMiddle(actor->user.targetActor)) - actor->user.pos.Z;
// check z diff of the player and the sprite
2022-08-21 08:12:24 +00:00
double zdist = 20 + RandomRange(200); // put a random amount
if (abs(zdiff) > zdist)
{
if (zdiff > 0)
// manipulate the z midpoint
2021-12-25 20:11:18 +00:00
//actor->user.sz += 256 * ACTORMOVETICS;
actor->user.pos.Z += 1024 * ACTORMOVETICS * zmaptoworld;
else
actor->user.pos.Z -= 256 * ACTORMOVETICS * zmaptoworld;
}
// save off lo and hi z
2022-08-21 08:12:24 +00:00
double loz = actor->user.loz;
double hiz = actor->user.hiz;
// adjust loz/hiz for water depth
2021-12-25 20:11:18 +00:00
if (actor->user.lo_sectp && actor->user.lo_sectp->hasU() && FixedToInt(actor->user.lo_sectp->depth_fixed))
2022-08-21 08:12:24 +00:00
loz -= FixedToInt(actor->user.lo_sectp->depth_fixed) - 8;
2022-08-21 08:12:24 +00:00
double bound;
// lower bound
2021-12-25 20:11:18 +00:00
if (actor->user.lowActor)
2022-08-21 08:12:24 +00:00
bound = loz - actor->user.floor_dist;
else
2022-08-21 08:12:24 +00:00
bound = loz - actor->user.floor_dist - HORNET_BOB_AMT;
2022-08-21 08:12:24 +00:00
if (actor->user.pos.Z > bound)
{
2022-08-21 08:12:24 +00:00
actor->user.pos.Z = bound;
}
// upper bound
2021-12-25 20:11:18 +00:00
if (actor->user.highActor)
2022-08-21 08:12:24 +00:00
bound = hiz + actor->user.ceiling_dist;
else
2022-08-21 08:12:24 +00:00
bound = hiz + actor->user.ceiling_dist + HORNET_BOB_AMT;
2022-08-21 08:12:24 +00:00
if (actor->user.pos.Z < bound)
{
2022-08-21 08:12:24 +00:00
actor->user.pos.Z = bound;
}
2022-08-21 08:12:24 +00:00
actor->user.pos.Z = min(actor->user.pos.Z, loz - actor->user.floor_dist);
actor->user.pos.Z = max(actor->user.pos.Z, hiz + actor->user.ceiling_dist);
2021-12-25 20:11:18 +00:00
actor->user.Counter = (actor->user.Counter + (ACTORMOVETICS << 3) + (ACTORMOVETICS << 1)) & 2047;
actor->spr.pos.Z = actor->user.pos.Z + HORNET_BOB_AMT * BobVal(actor->user.Counter);
2022-08-21 08:12:24 +00:00
bound = actor->user.hiz + actor->user.ceiling_dist + HORNET_BOB_AMT;
if (actor->spr.pos.Z < bound)
{
// bumped something
2022-08-21 08:12:24 +00:00
actor->spr.pos.Z = bound + HORNET_BOB_AMT;
actor->user.pos.Z = actor->spr.pos.Z;
}
return 0;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int InitHornetCircle(DSWActor* actor)
{
actor->user.ActorActionFunc = AF(DoHornetCircle);
actor->setStateGroup(NAME_Run);
// set it close
2021-10-30 10:09:34 +00:00
DoActorSetSpeed(actor, FAST_SPEED);
// set to really fast
2022-09-04 21:05:10 +00:00
actor->vel.X = 25;
// angle adjuster
2022-09-04 21:05:10 +00:00
actor->user.Counter2 = 400 / 3;
// random angle direction
if (RANDOM_P2(1024) < 512)
2021-12-25 20:11:18 +00:00
actor->user.Counter2 = -actor->user.Counter2;
// z velocity
2021-12-25 20:11:18 +00:00
actor->user.jump_speed = 200 + RANDOM_P2(128);
2022-08-21 08:12:24 +00:00
if (abs(actor->user.pos.Z - actor->user.hiz) < abs(actor->user.pos.Z - actor->user.loz))
2021-12-25 20:11:18 +00:00
actor->user.jump_speed = -actor->user.jump_speed;
2021-12-25 20:11:18 +00:00
actor->user.WaitTics = (RandomRange(3)+1) * 60;
actor->callAction();
return 0;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int DoHornetCircle(DSWActor* actor)
{
double bound;
actor->spr.Angles.Yaw += mapangle(actor->user.Counter2);
if (!move_actor(actor, DVector3(actor->spr.Angles.Yaw.ToVector() * actor->vel.X, 0)))
{
2021-10-29 22:04:34 +00:00
//ActorMoveHitReact(actor);
// try moving in the opposite direction
2021-12-25 20:11:18 +00:00
actor->user.Counter2 = -actor->user.Counter2;
actor->spr.Angles.Yaw += DAngle180;
if (!move_actor(actor, DVector3(actor->spr.Angles.Yaw.ToVector() * actor->vel.X, 0)))
{
InitActorReposition(actor);
return 0;
}
}
// move in the z direction
2022-08-21 12:08:03 +00:00
actor->user.pos.Z -= actor->user.jump_speed * ACTORMOVETICS * JUMP_FACTOR;
2022-08-21 08:12:24 +00:00
bound = actor->user.hiz + actor->user.ceiling_dist + HORNET_BOB_AMT;
if (actor->user.pos.Z < bound)
{
// bumped something
2022-08-21 08:12:24 +00:00
actor->user.pos.Z = bound;
InitActorReposition(actor);
return 0;
}
// time out
2021-12-25 20:11:18 +00:00
if ((actor->user.WaitTics -= ACTORMOVETICS) < 0)
{
InitActorReposition(actor);
2021-12-25 20:11:18 +00:00
actor->user.WaitTics = 0;
return 0;
}
return 0;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int DoHornetDeath(DSWActor* actor)
{
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_FALLING))
{
2022-08-20 22:01:00 +00:00
actor->user.loz = actor->user.zclip;
2021-10-29 21:41:33 +00:00
DoFall(actor);
}
else
{
2021-12-27 17:58:15 +00:00
actor->spr.cstat &= ~(CSTAT_SPRITE_YCENTER);
2021-12-25 20:11:18 +00:00
actor->user.jump_speed = 0;
actor->user.floor_dist = 0;
2021-10-29 21:41:33 +00:00
DoBeginFall(actor);
DoFindGroundPoint(actor);
2022-08-20 22:01:00 +00:00
actor->user.zclip = actor->user.loz;
}
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_SLIDING))
DoActorSlide(actor);
// slide while falling
auto vec = actor->spr.Angles.Yaw.ToVector() * actor->vel.X;
2022-08-30 22:47:02 +00:00
actor->user.coll = move_sprite(actor, DVector3(vec, 0), actor->user.ceiling_dist, actor->user.floor_dist, 1, ACTORMOVETICS);
// on the ground
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z >= actor->user.loz)
{
2021-12-27 17:58:15 +00:00
actor->user.Flags &= ~(SPR_FALLING|SPR_SLIDING);
actor->spr.cstat &= ~(CSTAT_SPRITE_YFLIP); // If upside down, reset it
actor->setStateGroup(NAME_Dead);
DeleteNoSoundOwner(actor);
return 0;
}
return 0;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
// Hornets can swarm around other hornets or whatever is tagged as swarm target
2022-09-04 20:52:03 +00:00
//
//---------------------------------------------------------------------------
int DoCheckSwarm(DSWActor* actor)
{
2022-09-04 21:05:10 +00:00
double dist, pdist;
DSWPlayer* pp;
if (!MoveSkip8) return 0; // Don't over check
2021-12-25 20:11:18 +00:00
if (!actor->user.targetActor) return 0;
// Who's the closest meat!?
2021-10-30 11:05:07 +00:00
DoActorPickClosePlayer(actor);
2021-12-25 20:11:18 +00:00
if (actor->user.targetActor->user.PlayerP)
{
2021-12-25 20:11:18 +00:00
pp = actor->user.targetActor->user.PlayerP;
pdist = (actor->spr.pos.XY() - pp->GetActor()->spr.pos.XY()).LengthSquared();
}
else
return 0;
// all enemys
2021-11-06 08:29:38 +00:00
SWStatIterator it(STAT_ENEMY);
while (auto itActor = it.Next())
{
if (!itActor->hasU()) continue;
if (itActor->spr.hitag != TAG_SWARMSPOT || itActor->spr.lotag != 2) continue;
2022-09-04 21:05:10 +00:00
dist = (actor->spr.pos.XY() - itActor->spr.pos.XY()).LengthSquared();
2021-12-25 20:11:18 +00:00
if (dist < pdist && actor->user.ID == itActor->user.ID) // Only flock to your own kind
{
2021-12-25 20:11:18 +00:00
actor->user.targetActor = itActor; // Set target to swarm center
}
}
return true;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int DoHornetMove(DSWActor* actor)
{
// Check for swarming
// lotag of 1 = Swarm around lotags of 2
// lotag of 0 is normal
2021-12-24 12:21:49 +00:00
if (actor->spr.hitag == TAG_SWARMSPOT && actor->spr.lotag == 1)
DoCheckSwarm(actor);
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_SLIDING))
DoActorSlide(actor);
2021-12-25 20:11:18 +00:00
if (actor->user.track >= 0)
2021-11-01 13:36:46 +00:00
ActorFollowTrack(actor, ACTORMOVETICS);
else
actor->callAction();
2021-11-01 07:04:23 +00:00
DoHornetMatchPlayerZ(actor);
DoActorSectorDamage(actor);
return 0;
}
2022-09-04 20:52:03 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
#include "saveable.h"
static saveable_data saveable_hornet_data[] =
{
SAVE_DATA(HornetPersonality),
SAVE_DATA(HornetAttrib),
SAVE_DATA(s_HornetRun),
SAVE_DATA(sg_HornetRun),
SAVE_DATA(s_HornetStand),
SAVE_DATA(sg_HornetStand),
SAVE_DATA(s_HornetDie),
SAVE_DATA(sg_HornetDie),
SAVE_DATA(s_HornetDead),
SAVE_DATA(sg_HornetDead),
SAVE_DATA(HornetActionSet),
};
saveable_module saveable_hornet =
{
// code
nullptr, 0,
// data
saveable_hornet_data,
SIZ(saveable_hornet_data)
};
END_SW_NS