mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-11 19:40:46 +00:00
5448f6ec2c
* DObjects may not be memset to 0. * There was still code trying to retrieve the player index with pointer artithmetic. With an array of pointers this does not work.
657 lines
16 KiB
C++
657 lines
16 KiB
C++
//-------------------------------------------------------------------------
|
|
/*
|
|
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"
|
|
#include "misc.h"
|
|
#include "sprite.h"
|
|
|
|
BEGIN_SW_NS
|
|
|
|
DECISION HornetBattle[] =
|
|
{
|
|
{50, &AF(InitHornetCircle ) },
|
|
{798, &AF(InitActorMoveCloser) },
|
|
{800, &AF(InitActorSetDecide) },
|
|
{1024, &AF(InitActorRunAway ) }
|
|
};
|
|
|
|
DECISION HornetOffense[] =
|
|
{
|
|
{1022, &AF(InitActorMoveCloser) },
|
|
{1024, &AF(InitActorSetDecide) }
|
|
};
|
|
|
|
DECISIONB HornetBroadcast[] =
|
|
{
|
|
{3, attr_alert },
|
|
{6, attr_ambient },
|
|
{1024, 0 }
|
|
};
|
|
|
|
DECISION HornetSurprised[] =
|
|
{
|
|
{100, &AF(InitHornetCircle ) },
|
|
{701, &AF(InitActorMoveCloser) },
|
|
{1024, &AF(InitActorDecide ) }
|
|
};
|
|
|
|
DECISION HornetEvasive[] =
|
|
{
|
|
{20, &AF(InitHornetCircle) },
|
|
{1024, nullptr },
|
|
};
|
|
|
|
DECISION HornetLostTarget[] =
|
|
{
|
|
{900, &AF(InitActorFindPlayer) },
|
|
{1024, &AF(InitActorWanderAround) }
|
|
};
|
|
|
|
DECISION HornetCloseRange[] =
|
|
{
|
|
{900, &AF(InitActorMoveCloser) },
|
|
{1024, &AF(InitActorReposition) }
|
|
};
|
|
|
|
DECISION HornetTouchTarget[] =
|
|
{
|
|
{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] =
|
|
{
|
|
{
|
|
{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]},
|
|
},
|
|
{
|
|
{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]},
|
|
},
|
|
{
|
|
{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]},
|
|
},
|
|
{
|
|
{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]},
|
|
},
|
|
{
|
|
{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]},
|
|
}
|
|
};
|
|
|
|
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] =
|
|
{
|
|
{
|
|
{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]}
|
|
},
|
|
{
|
|
{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]}
|
|
},
|
|
{
|
|
{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]}
|
|
},
|
|
{
|
|
{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]}
|
|
},
|
|
{
|
|
{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]}
|
|
}
|
|
};
|
|
|
|
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[] =
|
|
{
|
|
{HORNET_DIE + 0, HORNET_DIE_RATE, &AF(DoHornetDeath), &s_HornetDie[0]},
|
|
};
|
|
|
|
STATE* sg_HornetDie[] =
|
|
{
|
|
s_HornetDie
|
|
};
|
|
|
|
STATE s_HornetDead[] =
|
|
{
|
|
{HORNET_DEAD, HORNET_DIE_RATE, &AF(DoActorDebris), &s_HornetDead[0]},
|
|
};
|
|
|
|
STATE* sg_HornetDead[] =
|
|
{
|
|
s_HornetDead
|
|
};
|
|
|
|
/*
|
|
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
|
|
};
|
|
|
|
int DoHornetMatchPlayerZ(DSWActor* actor);
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
int SetupHornet(DSWActor* actor)
|
|
{
|
|
if (!(actor->spr.cstat & CSTAT_SPRITE_RESTORE))
|
|
{
|
|
SpawnUser(actor,HORNET_RUN_R0,s_HornetRun[0]);
|
|
actor->user.Health = HEALTH_HORNET;
|
|
}
|
|
|
|
ChangeState(actor, s_HornetRun[0]);
|
|
actor->user.__legacyState.Attrib = &HornetAttrib;
|
|
DoActorSetSpeed(actor, NORM_SPEED);
|
|
actor->user.__legacyState.StateEnd = s_HornetDie;
|
|
actor->user.__legacyState.Rot = sg_HornetRun;
|
|
|
|
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;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
int NullHornet(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_SLIDING))
|
|
DoActorSlide(actor);
|
|
|
|
DoHornetMatchPlayerZ(actor);
|
|
DoActorSectorDamage(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
static const int HORNET_BOB_AMT = 16;
|
|
|
|
int DoHornetMatchPlayerZ(DSWActor* actor)
|
|
{
|
|
|
|
// actor does a sine wave about actor->user.sz - this is the z mid point
|
|
|
|
double zdiff = (ActorZOfMiddle(actor->user.targetActor)) - actor->user.pos.Z;
|
|
|
|
// check z diff of the player and the sprite
|
|
double zdist = 20 + RandomRange(200); // put a random amount
|
|
if (abs(zdiff) > zdist)
|
|
{
|
|
if (zdiff > 0)
|
|
// manipulate the z midpoint
|
|
//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
|
|
double loz = actor->user.loz;
|
|
double hiz = actor->user.hiz;
|
|
|
|
// adjust loz/hiz for water depth
|
|
if (actor->user.lo_sectp && actor->user.lo_sectp->hasU() && FixedToInt(actor->user.lo_sectp->depth_fixed))
|
|
loz -= FixedToInt(actor->user.lo_sectp->depth_fixed) - 8;
|
|
|
|
double bound;
|
|
// lower bound
|
|
if (actor->user.lowActor)
|
|
bound = loz - actor->user.floor_dist;
|
|
else
|
|
bound = loz - actor->user.floor_dist - HORNET_BOB_AMT;
|
|
|
|
if (actor->user.pos.Z > bound)
|
|
{
|
|
actor->user.pos.Z = bound;
|
|
}
|
|
|
|
// upper bound
|
|
if (actor->user.highActor)
|
|
bound = hiz + actor->user.ceiling_dist;
|
|
else
|
|
bound = hiz + actor->user.ceiling_dist + HORNET_BOB_AMT;
|
|
|
|
if (actor->user.pos.Z < bound)
|
|
{
|
|
actor->user.pos.Z = bound;
|
|
}
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
bound = actor->user.hiz + actor->user.ceiling_dist + HORNET_BOB_AMT;
|
|
if (actor->spr.pos.Z < bound)
|
|
{
|
|
// bumped something
|
|
actor->spr.pos.Z = bound + HORNET_BOB_AMT;
|
|
actor->user.pos.Z = actor->spr.pos.Z;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
int InitHornetCircle(DSWActor* actor)
|
|
{
|
|
actor->user.ActorActionFunc = AF(DoHornetCircle);
|
|
|
|
actor->setStateGroup(NAME_Run);
|
|
|
|
// set it close
|
|
DoActorSetSpeed(actor, FAST_SPEED);
|
|
|
|
// set to really fast
|
|
actor->vel.X = 25;
|
|
// angle adjuster
|
|
actor->user.Counter2 = 400 / 3;
|
|
// random angle direction
|
|
if (RANDOM_P2(1024) < 512)
|
|
actor->user.Counter2 = -actor->user.Counter2;
|
|
|
|
// z velocity
|
|
actor->user.jump_speed = 200 + RANDOM_P2(128);
|
|
if (abs(actor->user.pos.Z - actor->user.hiz) < abs(actor->user.pos.Z - actor->user.loz))
|
|
actor->user.jump_speed = -actor->user.jump_speed;
|
|
|
|
actor->user.WaitTics = (RandomRange(3)+1) * 60;
|
|
|
|
actor->callAction();
|
|
|
|
return 0;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
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)))
|
|
{
|
|
//ActorMoveHitReact(actor);
|
|
|
|
// try moving in the opposite direction
|
|
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
|
|
actor->user.pos.Z -= actor->user.jump_speed * ACTORMOVETICS * JUMP_FACTOR;
|
|
|
|
bound = actor->user.hiz + actor->user.ceiling_dist + HORNET_BOB_AMT;
|
|
if (actor->user.pos.Z < bound)
|
|
{
|
|
// bumped something
|
|
actor->user.pos.Z = bound;
|
|
InitActorReposition(actor);
|
|
return 0;
|
|
}
|
|
|
|
// time out
|
|
if ((actor->user.WaitTics -= ACTORMOVETICS) < 0)
|
|
{
|
|
InitActorReposition(actor);
|
|
actor->user.WaitTics = 0;
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
int DoHornetDeath(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_FALLING))
|
|
{
|
|
actor->user.loz = actor->user.zclip;
|
|
DoFall(actor);
|
|
}
|
|
else
|
|
{
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_YCENTER);
|
|
actor->user.jump_speed = 0;
|
|
actor->user.floor_dist = 0;
|
|
DoBeginFall(actor);
|
|
DoFindGroundPoint(actor);
|
|
actor->user.zclip = actor->user.loz;
|
|
}
|
|
|
|
if (actor->user.Flags & (SPR_SLIDING))
|
|
DoActorSlide(actor);
|
|
|
|
// slide while falling
|
|
auto vec = actor->spr.Angles.Yaw.ToVector() * actor->vel.X;
|
|
|
|
actor->user.coll = move_sprite(actor, DVector3(vec, 0), actor->user.ceiling_dist, actor->user.floor_dist, 1, ACTORMOVETICS);
|
|
|
|
// on the ground
|
|
if (actor->spr.pos.Z >= actor->user.loz)
|
|
{
|
|
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;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
// Hornets can swarm around other hornets or whatever is tagged as swarm target
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
int DoCheckSwarm(DSWActor* actor)
|
|
{
|
|
double dist, pdist;
|
|
DSWPlayer* pp;
|
|
|
|
if (!MoveSkip8) return 0; // Don't over check
|
|
|
|
if (!actor->user.targetActor) return 0;
|
|
|
|
// Who's the closest meat!?
|
|
DoActorPickClosePlayer(actor);
|
|
|
|
if (actor->user.targetActor->user.PlayerP)
|
|
{
|
|
pp = actor->user.targetActor->user.PlayerP;
|
|
pdist = (actor->spr.pos.XY() - pp->GetActor()->spr.pos.XY()).LengthSquared();
|
|
}
|
|
else
|
|
return 0;
|
|
|
|
// all enemys
|
|
SWStatIterator it(STAT_ENEMY);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (!itActor->hasU()) continue;
|
|
|
|
if (itActor->spr.hitag != TAG_SWARMSPOT || itActor->spr.lotag != 2) continue;
|
|
|
|
dist = (actor->spr.pos.XY() - itActor->spr.pos.XY()).LengthSquared();
|
|
|
|
if (dist < pdist && actor->user.ID == itActor->user.ID) // Only flock to your own kind
|
|
{
|
|
actor->user.targetActor = itActor; // Set target to swarm center
|
|
}
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
int DoHornetMove(DSWActor* actor)
|
|
{
|
|
// Check for swarming
|
|
// lotag of 1 = Swarm around lotags of 2
|
|
// lotag of 0 is normal
|
|
if (actor->spr.hitag == TAG_SWARMSPOT && actor->spr.lotag == 1)
|
|
DoCheckSwarm(actor);
|
|
|
|
if (actor->user.Flags & (SPR_SLIDING))
|
|
DoActorSlide(actor);
|
|
|
|
if (actor->user.track >= 0)
|
|
ActorFollowTrack(actor, ACTORMOVETICS);
|
|
else
|
|
actor->callAction();
|
|
|
|
DoHornetMatchPlayerZ(actor);
|
|
|
|
DoActorSectorDamage(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
#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
|