123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663 |
- // Copyright (c) ZeniMax Media Inc.
- // Licensed under the GNU General Public License 2.0.
- /*
- ==============================================================================
- hover
- ==============================================================================
- */
- #include "g_local.h"
- #include "m_hover.h"
- #include "m_flash.h"
- static cached_soundindex sound_pain1;
- static cached_soundindex sound_pain2;
- static cached_soundindex sound_death1;
- static cached_soundindex sound_death2;
- static cached_soundindex sound_sight;
- static cached_soundindex sound_search1;
- static cached_soundindex sound_search2;
- // ROGUE
- // daedalus sounds
- static cached_soundindex daed_sound_pain1;
- static cached_soundindex daed_sound_pain2;
- static cached_soundindex daed_sound_death1;
- static cached_soundindex daed_sound_death2;
- static cached_soundindex daed_sound_sight;
- static cached_soundindex daed_sound_search1;
- static cached_soundindex daed_sound_search2;
- // ROGUE
- MONSTERINFO_SIGHT(hover_sight) (edict_t *self, edict_t *other) -> void
- {
- // PMM - daedalus sounds
- if (self->mass < 225)
- gi.sound(self, CHAN_VOICE, sound_sight, 1, ATTN_NORM, 0);
- else
- gi.sound(self, CHAN_VOICE, daed_sound_sight, 1, ATTN_NORM, 0);
- }
- MONSTERINFO_SEARCH(hover_search) (edict_t *self) -> void
- {
- // PMM - daedalus sounds
- if (self->mass < 225)
- {
- if (frandom() < 0.5f)
- gi.sound(self, CHAN_VOICE, sound_search1, 1, ATTN_NORM, 0);
- else
- gi.sound(self, CHAN_VOICE, sound_search2, 1, ATTN_NORM, 0);
- }
- else
- {
- if (frandom() < 0.5f)
- gi.sound(self, CHAN_VOICE, daed_sound_search1, 1, ATTN_NORM, 0);
- else
- gi.sound(self, CHAN_VOICE, daed_sound_search2, 1, ATTN_NORM, 0);
- }
- }
- void hover_run(edict_t *self);
- void hover_dead(edict_t *self);
- void hover_attack(edict_t *self);
- void hover_reattack(edict_t *self);
- void hover_fire_blaster(edict_t *self);
- mframe_t hover_frames_stand[] = {
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand },
- { ai_stand }
- };
- MMOVE_T(hover_move_stand) = { FRAME_stand01, FRAME_stand30, hover_frames_stand, nullptr };
- mframe_t hover_frames_pain3[] = {
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move }
- };
- MMOVE_T(hover_move_pain3) = { FRAME_pain301, FRAME_pain309, hover_frames_pain3, hover_run };
- mframe_t hover_frames_pain2[] = {
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move }
- };
- MMOVE_T(hover_move_pain2) = { FRAME_pain201, FRAME_pain212, hover_frames_pain2, hover_run };
- mframe_t hover_frames_pain1[] = {
- { ai_move },
- { ai_move },
- { ai_move, 2 },
- { ai_move, -8 },
- { ai_move, -4 },
- { ai_move, -6 },
- { ai_move, -4 },
- { ai_move, -3 },
- { ai_move, 1 },
- { ai_move },
- { ai_move },
- { ai_move },
- { ai_move, 3 },
- { ai_move, 1 },
- { ai_move },
- { ai_move, 2 },
- { ai_move, 3 },
- { ai_move, 2 },
- { ai_move, 7 },
- { ai_move, 1 },
- { ai_move },
- { ai_move },
- { ai_move, 2 },
- { ai_move },
- { ai_move },
- { ai_move, 5 },
- { ai_move, 3 },
- { ai_move, 4 }
- };
- MMOVE_T(hover_move_pain1) = { FRAME_pain101, FRAME_pain128, hover_frames_pain1, hover_run };
- mframe_t hover_frames_walk[] = {
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 },
- { ai_walk, 4 }
- };
- MMOVE_T(hover_move_walk) = { FRAME_forwrd01, FRAME_forwrd35, hover_frames_walk, nullptr };
- mframe_t hover_frames_run[] = {
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 },
- { ai_run, 10 }
- };
- MMOVE_T(hover_move_run) = { FRAME_forwrd01, FRAME_forwrd35, hover_frames_run, nullptr };
- static void hover_gib(edict_t *self)
- {
- gi.WriteByte(svc_temp_entity);
- gi.WriteByte(TE_EXPLOSION1);
- gi.WritePosition(self->s.origin);
- gi.multicast(self->s.origin, MULTICAST_PHS, false);
- self->s.skinnum /= 2;
- ThrowGibs(self, 150, {
- { 2, "models/objects/gibs/sm_meat/tris.md2" },
- { 2, "models/objects/gibs/sm_metal/tris.md2", GIB_METALLIC },
- { "models/monsters/hover/gibs/chest.md2", GIB_SKINNED },
- { 2, "models/monsters/hover/gibs/ring.md2", GIB_SKINNED | GIB_METALLIC },
- { 2, "models/monsters/hover/gibs/foot.md2", GIB_SKINNED },
- { "models/monsters/hover/gibs/head.md2", GIB_SKINNED | GIB_HEAD },
- });
- }
- THINK(hover_deadthink) (edict_t *self) -> void
- {
- if (!self->groundentity && level.time < self->timestamp)
- {
- self->nextthink = level.time + FRAME_TIME_S;
- return;
- }
- hover_gib(self);
- }
- void hover_dying(edict_t *self)
- {
- if (self->groundentity)
- {
- hover_deadthink(self);
- return;
- }
- if (brandom())
- return;
- gi.WriteByte(svc_temp_entity);
- gi.WriteByte(TE_PLAIN_EXPLOSION);
- gi.WritePosition(self->s.origin);
- gi.multicast(self->s.origin, MULTICAST_PHS, false);
- if (brandom())
- ThrowGibs(self, 120, {
- { "models/objects/gibs/sm_meat/tris.md2" }
- });
- else
- ThrowGibs(self, 120, {
- { "models/objects/gibs/sm_metal/tris.md2", GIB_METALLIC }
- });
- }
- mframe_t hover_frames_death1[] = {
- { ai_move },
- { ai_move, 0.f, hover_dying },
- { ai_move },
- { ai_move, 0.f, hover_dying },
- { ai_move },
- { ai_move, 0.f, hover_dying },
- { ai_move, -10, hover_dying },
- { ai_move, 3 },
- { ai_move, 5, hover_dying },
- { ai_move, 4, hover_dying },
- { ai_move, 7 }
- };
- MMOVE_T(hover_move_death1) = { FRAME_death101, FRAME_death111, hover_frames_death1, hover_dead };
- mframe_t hover_frames_start_attack[] = {
- { ai_charge, 1 },
- { ai_charge, 1 },
- { ai_charge, 1 }
- };
- MMOVE_T(hover_move_start_attack) = { FRAME_attak101, FRAME_attak103, hover_frames_start_attack, hover_attack };
- mframe_t hover_frames_attack1[] = {
- { ai_charge, -10, hover_fire_blaster },
- { ai_charge, -10, hover_fire_blaster },
- { ai_charge, 0, hover_reattack },
- };
- MMOVE_T(hover_move_attack1) = { FRAME_attak104, FRAME_attak106, hover_frames_attack1, nullptr };
- mframe_t hover_frames_end_attack[] = {
- { ai_charge, 1 },
- { ai_charge, 1 }
- };
- MMOVE_T(hover_move_end_attack) = { FRAME_attak107, FRAME_attak108, hover_frames_end_attack, hover_run };
- /* PMM - circle strafing code */
- #if 0
- mframe_t hover_frames_start_attack2[] = {
- { ai_charge, 15 },
- { ai_charge, 15 },
- { ai_charge, 15 }
- };
- MMOVE_T(hover_move_start_attack2) = { FRAME_attak101, FRAME_attak103, hover_frames_start_attack2, hover_attack };
- #endif
- mframe_t hover_frames_attack2[] = {
- { ai_charge, 10, hover_fire_blaster },
- { ai_charge, 10, hover_fire_blaster },
- { ai_charge, 10, hover_reattack },
- };
- MMOVE_T(hover_move_attack2) = { FRAME_attak104, FRAME_attak106, hover_frames_attack2, nullptr };
- #if 0
- mframe_t hover_frames_end_attack2[] = {
- { ai_charge, 15 },
- { ai_charge, 15 }
- };
- MMOVE_T(hover_move_end_attack2) = { FRAME_attak107, FRAME_attak108, hover_frames_end_attack2, hover_run };
- #endif
- // end of circle strafe
- void hover_reattack(edict_t *self)
- {
- if (self->enemy->health > 0)
- if (visible(self, self->enemy))
- if (frandom() <= 0.6f)
- {
- if (self->monsterinfo.attack_state == AS_STRAIGHT)
- {
- M_SetAnimation(self, &hover_move_attack1);
- return;
- }
- else if (self->monsterinfo.attack_state == AS_SLIDING)
- {
- M_SetAnimation(self, &hover_move_attack2);
- return;
- }
- else
- gi.Com_PrintFmt("hover_reattack: unexpected state {}\n", (int32_t) self->monsterinfo.attack_state);
- }
- M_SetAnimation(self, &hover_move_end_attack);
- }
- void hover_fire_blaster(edict_t *self)
- {
- vec3_t start;
- vec3_t forward, right;
- vec3_t end;
- vec3_t dir;
- if (!self->enemy || !self->enemy->inuse) // PGM
- return; // PGM
- AngleVectors(self->s.angles, forward, right, nullptr);
- vec3_t o = monster_flash_offset[(self->s.frame & 1) ? MZ2_HOVER_BLASTER_2 : MZ2_HOVER_BLASTER_1];
- start = M_ProjectFlashSource(self, o, forward, right);
- end = self->enemy->s.origin;
- end[2] += self->enemy->viewheight;
- dir = end - start;
- dir.normalize();
- // PGM - daedalus fires blaster2
- if (self->mass < 200)
- monster_fire_blaster(self, start, dir, 1, 1000, (self->s.frame & 1) ? MZ2_HOVER_BLASTER_2 : MZ2_HOVER_BLASTER_1, (self->s.frame % 4) ? EF_NONE : EF_HYPERBLASTER);
- else
- monster_fire_blaster2(self, start, dir, 1, 1000, (self->s.frame & 1) ? MZ2_DAEDALUS_BLASTER_2 : MZ2_DAEDALUS_BLASTER, (self->s.frame % 4) ? EF_NONE : EF_BLASTER);
- // PGM
- }
- MONSTERINFO_STAND(hover_stand) (edict_t *self) -> void
- {
- M_SetAnimation(self, &hover_move_stand);
- }
- MONSTERINFO_RUN(hover_run) (edict_t *self) -> void
- {
- if (self->monsterinfo.aiflags & AI_STAND_GROUND)
- M_SetAnimation(self, &hover_move_stand);
- else
- M_SetAnimation(self, &hover_move_run);
- }
- MONSTERINFO_WALK(hover_walk) (edict_t *self) -> void
- {
- M_SetAnimation(self, &hover_move_walk);
- }
- MONSTERINFO_ATTACK(hover_start_attack) (edict_t *self) -> void
- {
- M_SetAnimation(self, &hover_move_start_attack);
- }
- void hover_attack(edict_t *self)
- {
- float chance = 0.5f;
- if (self->mass > 150) // the daedalus strafes more
- chance += 0.1f;
- if (frandom() > chance)
- {
- M_SetAnimation(self, &hover_move_attack1);
- self->monsterinfo.attack_state = AS_STRAIGHT;
- }
- else // circle strafe
- {
- if (frandom() <= 0.5f) // switch directions
- self->monsterinfo.lefty = !self->monsterinfo.lefty;
- M_SetAnimation(self, &hover_move_attack2);
- self->monsterinfo.attack_state = AS_SLIDING;
- }
- }
- PAIN(hover_pain) (edict_t *self, edict_t *other, float kick, int damage, const mod_t &mod) -> void
- {
- if (level.time < self->pain_debounce_time)
- return;
- self->pain_debounce_time = level.time + 3_sec;
- float r = frandom();
- //====
- if (r < 0.5f)
- {
- // PMM - daedalus sounds
- if (self->mass < 225)
- gi.sound(self, CHAN_VOICE, sound_pain1, 1, ATTN_NORM, 0);
- else
- gi.sound(self, CHAN_VOICE, daed_sound_pain1, 1, ATTN_NORM, 0);
- }
- else
- {
- // PMM - daedalus sounds
- if (self->mass < 225)
- gi.sound(self, CHAN_VOICE, sound_pain2, 1, ATTN_NORM, 0);
- else
- gi.sound(self, CHAN_VOICE, daed_sound_pain2, 1, ATTN_NORM, 0);
- }
- // PGM
- //====
- if (!M_ShouldReactToPain(self, mod))
- return; // no pain anims in nightmare
- r = frandom();
- if (damage <= 25)
- {
- if (r < 0.5f)
- M_SetAnimation(self, &hover_move_pain3);
- else
- M_SetAnimation(self, &hover_move_pain2);
- }
- else
- {
- //====
- // PGM pain sequence is WAY too long
- if (r < 0.3f)
- M_SetAnimation(self, &hover_move_pain1);
- else
- M_SetAnimation(self, &hover_move_pain2);
- // PGM
- //====
- }
- }
- MONSTERINFO_SETSKIN(hover_setskin) (edict_t *self) -> void
- {
- if (self->health < (self->max_health / 2))
- self->s.skinnum |= 1; // PGM support for skins 2 & 3.
- else
- self->s.skinnum &= ~1; // PGM support for skins 2 & 3.
- }
- void hover_dead(edict_t *self)
- {
- self->mins = { -16, -16, -24 };
- self->maxs = { 16, 16, -8 };
- self->movetype = MOVETYPE_TOSS;
- self->think = hover_deadthink;
- self->nextthink = level.time + FRAME_TIME_S;
- self->timestamp = level.time + 15_sec;
- gi.linkentity(self);
- }
- DIE(hover_die) (edict_t *self, edict_t *inflictor, edict_t *attacker, int damage, const vec3_t &point, const mod_t &mod) -> void
- {
- self->s.effects = EF_NONE;
- self->monsterinfo.power_armor_type = IT_NULL;
- if (M_CheckGib(self, mod))
- {
- hover_gib(self);
- return;
- }
- if (self->deadflag)
- return;
- // regular death
- // PMM - daedalus sounds
- if (self->mass < 225)
- {
- if (frandom() < 0.5f)
- gi.sound(self, CHAN_VOICE, sound_death1, 1, ATTN_NORM, 0);
- else
- gi.sound(self, CHAN_VOICE, sound_death2, 1, ATTN_NORM, 0);
- }
- else
- {
- if (frandom() < 0.5f)
- gi.sound(self, CHAN_VOICE, daed_sound_death1, 1, ATTN_NORM, 0);
- else
- gi.sound(self, CHAN_VOICE, daed_sound_death2, 1, ATTN_NORM, 0);
- }
- self->deadflag = true;
- self->takedamage = true;
- M_SetAnimation(self, &hover_move_death1);
- }
- static void hover_set_fly_parameters(edict_t *self)
- {
- self->monsterinfo.fly_thrusters = false;
- self->monsterinfo.fly_acceleration = 20.f;
- self->monsterinfo.fly_speed = 120.f;
- // Icarus prefers to keep its distance, but flies slower than the flyer.
- // he never pins because of this.
- self->monsterinfo.fly_min_distance = 150.f;
- self->monsterinfo.fly_max_distance = 350.f;
- }
- /*QUAKED monster_hover (1 .5 0) (-16 -16 -24) (16 16 32) Ambush Trigger_Spawn Sight
- */
- /*QUAKED monster_daedalus (1 .5 0) (-16 -16 -24) (16 16 32) Ambush Trigger_Spawn Sight
- This is the improved icarus monster.
- */
- void SP_monster_hover(edict_t *self)
- {
- if ( !M_AllowSpawn( self ) ) {
- G_FreeEdict( self );
- return;
- }
- self->movetype = MOVETYPE_STEP;
- self->solid = SOLID_BBOX;
- self->s.modelindex = gi.modelindex("models/monsters/hover/tris.md2");
- gi.modelindex("models/monsters/hover/gibs/chest.md2");
- gi.modelindex("models/monsters/hover/gibs/foot.md2");
- gi.modelindex("models/monsters/hover/gibs/head.md2");
- gi.modelindex("models/monsters/hover/gibs/ring.md2");
- self->mins = { -24, -24, -24 };
- self->maxs = { 24, 24, 32 };
- self->health = 240 * st.health_multiplier;
- self->gib_health = -100;
- self->mass = 150;
- self->pain = hover_pain;
- self->die = hover_die;
- self->monsterinfo.stand = hover_stand;
- self->monsterinfo.walk = hover_walk;
- self->monsterinfo.run = hover_run;
- self->monsterinfo.attack = hover_start_attack;
- self->monsterinfo.sight = hover_sight;
- self->monsterinfo.search = hover_search;
- self->monsterinfo.setskin = hover_setskin;
- // PGM
- if (strcmp(self->classname, "monster_daedalus") == 0)
- {
- self->health = 450 * st.health_multiplier;
- self->mass = 225;
- self->yaw_speed = 23;
- if (!st.was_key_specified("power_armor_type"))
- self->monsterinfo.power_armor_type = IT_ITEM_POWER_SCREEN;
- if (!st.was_key_specified("power_armor_power"))
- self->monsterinfo.power_armor_power = 100;
- // PMM - daedalus sounds
- self->monsterinfo.engine_sound = gi.soundindex("daedalus/daedidle1.wav");
- daed_sound_pain1.assign("daedalus/daedpain1.wav");
- daed_sound_pain2.assign("daedalus/daedpain2.wav");
- daed_sound_death1.assign("daedalus/daeddeth1.wav");
- daed_sound_death2.assign("daedalus/daeddeth2.wav");
- daed_sound_sight.assign("daedalus/daedsght1.wav");
- daed_sound_search1.assign("daedalus/daedsrch1.wav");
- daed_sound_search2.assign("daedalus/daedsrch2.wav");
- gi.soundindex("tank/tnkatck3.wav");
- // pmm
- }
- else
- {
- self->yaw_speed = 18;
- sound_pain1.assign("hover/hovpain1.wav");
- sound_pain2.assign("hover/hovpain2.wav");
- sound_death1.assign("hover/hovdeth1.wav");
- sound_death2.assign("hover/hovdeth2.wav");
- sound_sight.assign("hover/hovsght1.wav");
- sound_search1.assign("hover/hovsrch1.wav");
- sound_search2.assign("hover/hovsrch2.wav");
- gi.soundindex("hover/hovatck1.wav");
- self->monsterinfo.engine_sound = gi.soundindex("hover/hovidle1.wav");
- }
- // PGM
- gi.linkentity(self);
- M_SetAnimation(self, &hover_move_stand);
- self->monsterinfo.scale = MODEL_SCALE;
- flymonster_start(self);
- // PGM
- if (strcmp(self->classname, "monster_daedalus") == 0)
- self->s.skinnum = 2;
- // PGM
- self->monsterinfo.aiflags |= AI_ALTERNATE_FLY;
- hover_set_fly_parameters(self);
- }
|