1 #ifdef REGISTER_MONSTER
4 /* function */ m_brute,
5 /* spawnflags */ MON_FLAG_MELEE | MON_FLAG_RANGED | MON_FLAG_MUTATORBLOCKED,
6 /* mins,maxs */ '-36 -36 -20', '36 36 50',
7 /* model */ "ogre.dpm",
9 /* fullname */ _("Brute")
12 #define BRUTE_SETTINGS(monster) \
13 MON_ADD_CVAR(monster, health) \
14 MON_ADD_CVAR(monster, attack_chainsaw_damage) \
15 MON_ADD_CVAR(monster, attack_uzi_bullets) \
16 MON_ADD_CVAR(monster, attack_uzi_damage) \
17 MON_ADD_CVAR(monster, attack_uzi_force) \
18 MON_ADD_CVAR(monster, attack_uzi_chance) \
19 MON_ADD_CVAR(monster, attack_grenade_damage) \
20 MON_ADD_CVAR(monster, attack_grenade_edgedamage) \
21 MON_ADD_CVAR(monster, attack_grenade_force) \
22 MON_ADD_CVAR(monster, attack_grenade_radius) \
23 MON_ADD_CVAR(monster, attack_grenade_speed) \
24 MON_ADD_CVAR(monster, attack_grenade_speed_up) \
25 MON_ADD_CVAR(monster, speed_stop) \
26 MON_ADD_CVAR(monster, speed_run) \
27 MON_ADD_CVAR(monster, speed_walk)
34 const float brute_anim_idle = 0;
35 const float brute_anim_walk = 1;
36 const float brute_anim_run = 2;
37 const float brute_anim_pain = 3;
38 const float brute_anim_swing = 4;
39 const float brute_anim_die = 5;
45 self.brute_cycles += 1;
46 self.angles_y = self.angles_y + random()* 25;
48 monster_melee(self.enemy, MON_CVAR(brute, attack_chainsaw_damage), brute_anim_swing, self.attack_range, 0, DEATH_MONSTER_BRUTE_BLADE, TRUE);
50 if(self.brute_cycles <= 4)
51 defer(0.2, brute_blade);
56 self.brute_cycles += 1;
58 monster_makevectors(self.enemy);
60 sound(self, CH_WEAPON_A, "weapons/uzi_fire.wav", VOL_BASE, ATTEN_NORM);
61 fireBallisticBullet(CENTER_OR_VIEWOFS(self), v_forward, 0.02, 18000, 5, MON_CVAR(brute, attack_uzi_damage), MON_CVAR(brute, attack_uzi_force), DEATH_MONSTER_BRUTE_UZI, 0, 1, 115);
62 endFireBallisticBullet();
64 if(self.brute_cycles <= MON_CVAR(brute, attack_uzi_bullets))
65 defer(0.1, brute_uzi);
68 void brute_grenade_explode()
70 pointparticles(particleeffectnum("grenade_explode"), self.origin, '0 0 0', 1);
71 sound(self, CH_SHOTS, "weapons/grenade_impact.wav", VOL_BASE, ATTEN_NORM);
73 self.event_damage = func_null;
74 self.takedamage = DAMAGE_NO;
76 if(self.movetype == MOVETYPE_NONE)
77 self.velocity = self.oldvelocity;
79 RadiusDamage (self, self.realowner, MON_CVAR(brute, attack_grenade_damage), MON_CVAR(brute, attack_grenade_edgedamage), MON_CVAR(brute, attack_grenade_radius), world, MON_CVAR(brute, attack_grenade_force), self.projectiledeathtype, other);
84 void brute_grenade_damage(entity inflictor, entity attacker, float damage, float deathtype, vector hitloc, vector force)
89 if (!W_CheckProjectileDamage(inflictor.realowner, self.realowner, deathtype, -1)) // no exceptions
90 return; // g_projectiles_damage says to halt
92 self.health = self.health - damage;
95 W_PrepareExplosionByDamage(attacker, self.use);
98 void brute_grenade_touch()
105 void brute_grenade_think()
107 self.nextthink = time;
111 brute_grenade_explode();
120 monster_makevectors(self.enemy);
122 sound(self, CH_WEAPON_A, "weapons/grenade_fire.wav", VOL_BASE, ATTEN_NORM);
125 gren.owner = gren.realowner = self;
126 gren.classname = "grenade";
127 gren.bot_dodge = TRUE;
128 gren.bot_dodgerating = MON_CVAR(brute, attack_grenade_damage);
129 gren.movetype = MOVETYPE_BOUNCE;
130 PROJECTILE_MAKETRIGGER(gren);
131 gren.projectiledeathtype = DEATH_MONSTER_BRUTE_GRENADE;
132 setorigin(gren, CENTER_OR_VIEWOFS(self));
133 setsize(gren, '-3 -3 -3', '3 3 3');
136 gren.nextthink = time;
137 gren.think = brute_grenade_think;
138 gren.use = brute_grenade_explode;
139 gren.touch = brute_grenade_touch;
141 gren.takedamage = DAMAGE_YES;
143 gren.damageforcescale = 0;
144 gren.event_damage = brute_grenade_damage;
145 gren.damagedbycontents = TRUE;
146 gren.missile_flags = MIF_SPLASH | MIF_ARC;
147 W_SetupProjectileVelocityEx(gren, v_forward, v_up, MON_CVAR(brute, attack_grenade_speed), MON_CVAR(brute, attack_grenade_speed_up), 0, 0, FALSE);
149 gren.angles = vectoangles (gren.velocity);
150 gren.flags = FL_PROJECTILE;
152 CSQCProjectile(gren, TRUE, PROJECTILE_GRENADE, TRUE);
155 float brute_attack(float attack_type)
159 case MONSTER_ATTACK_MELEE:
161 self.brute_cycles = 0;
162 self.attack_finished_single = time + 1.3;
167 case MONSTER_ATTACK_RANGED:
169 self.brute_cycles = 0;
170 if(random() <= MON_CVAR(brute, attack_uzi_chance))
172 self.frame = brute_anim_pain;
173 self.attack_finished_single = time + 0.8;
174 defer(0.1, brute_uzi);
178 monster_makevectors(self.enemy);
180 self.frame = brute_anim_pain;
181 self.attack_finished_single = time + 1.2;
191 void spawnfunc_monster_brute()
193 self.classname = "monster_brute";
195 self.monster_spawnfunc = spawnfunc_monster_brute;
197 if(Monster_CheckAppearFlags(self))
200 if not(monster_initialize(MON_BRUTE, FALSE)) { remove(self); return; }
203 // compatibility with old spawns
204 void spawnfunc_monster_ogre() { spawnfunc_monster_brute(); }
206 float m_brute(float req)
212 monster_move(MON_CVAR(brute, speed_run), MON_CVAR(brute, speed_walk), MON_CVAR(brute, speed_stop), brute_anim_run, brute_anim_walk, brute_anim_idle);
217 self.frame = brute_anim_die;
222 if not(self.health) self.health = MON_CVAR(brute, health);
224 self.monster_loot = spawnfunc_item_bullets;
225 self.monster_attackfunc = brute_attack;
226 self.frame = brute_anim_idle;
227 self.weapon = WEP_GRENADE_LAUNCHER;
233 precache_model ("models/monsters/ogre.dpm");
234 precache_sound ("weapons/uzi_fire.wav");
235 precache_sound ("weapons/grenade_impact.wav");
236 precache_sound ("weapons/grenade_fire.wav");
241 MON_CONFIG_SETTINGS(BRUTE_SETTINGS(brute))
251 float m_brute(float req)
257 precache_model ("models/monsters/ogre.dpm");
266 #endif // REGISTER_MONSTER