blobwarsAttrition/src/entities/boss/eyeDroidCommander.c

420 lines
7.6 KiB
C
Raw Normal View History

2018-01-26 09:51:07 +01:00
/*
2019-06-02 17:13:34 +02:00
Copyright (C) 2018-2019 Parallel Realities
2018-01-26 09:51:07 +01:00
This program 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include "eyeDroidCommander.h"
static void activate(int activate);
static void tick(void);
static void walk(void);
2018-02-24 17:53:03 +01:00
static void moveTowardsPlayer(int dir);
static void die(void);
2018-01-26 09:51:07 +01:00
static void die2(void);
static void selectWeapon(void);
static void attackPistol(void);
static void attackPlasma(void);
static void attackMissile(void);
static void preFire(void);
static void attack(void);
static void applyDamage(int amount);
static SDL_Rect *getCurrentSprite(void);
2018-01-26 09:51:07 +01:00
static int brakingTimer;
2018-01-30 00:00:14 +01:00
static Sprite *aimedSprite;
static Sprite *missileSprite[2];
static Sprite *plasmaSprite[2];
2018-01-26 09:51:07 +01:00
2018-02-24 17:53:03 +01:00
Entity *initEyeDroidCommander(void)
2018-01-26 09:51:07 +01:00
{
Boss *b;
2019-06-02 17:13:34 +02:00
2018-02-24 17:53:03 +01:00
b = initBoss();
2019-06-02 17:13:34 +02:00
STRNCPY(b->name, "EyeDroid Commander", MAX_NAME_LENGTH);
2018-01-26 09:51:07 +01:00
2018-01-30 00:00:14 +01:00
b->sprite[FACING_LEFT] = getSprite("DroidCommanderLeft");
b->sprite[FACING_RIGHT] = getSprite("DroidCommanderRight");
b->sprite[FACING_DIE] = getSprite("DroidCommanderDie");
2019-06-02 17:13:34 +02:00
b->flags |= EF_WEIGHTLESS | EF_EXPLODES;
2018-01-26 09:51:07 +01:00
b->health = b->healthMax = 250;
2019-06-02 17:13:34 +02:00
2018-02-24 17:53:03 +01:00
b->action = walk;
b->walk = walk;
b->tick = tick;
b->activate = activate;
b->applyDamage = applyDamage;
2018-02-24 17:53:03 +01:00
b->die = die;
b->getCurrentSprite = getCurrentSprite;
2018-01-26 09:51:07 +01:00
brakingTimer = 0;
2019-06-02 17:13:34 +02:00
2018-01-30 00:00:14 +01:00
aimedSprite = getSprite("AimedShot");
2019-06-02 17:13:34 +02:00
2018-01-30 00:00:14 +01:00
missileSprite[0] = getSprite("MissileRight");
missileSprite[1] = getSprite("MissileLeft");
2019-06-02 17:13:34 +02:00
2018-01-30 00:00:14 +01:00
plasmaSprite[0] = getSprite("PlasmaRight");
plasmaSprite[1] = getSprite("PlasmaLeft");
2019-06-02 17:13:34 +02:00
2018-02-24 17:53:03 +01:00
return (Entity*)b;
2018-01-26 09:51:07 +01:00
}
static void activate(int activate)
{
self->flags &= ~EF_GONE;
world.isBossActive = 1;
addTeleportStars(self);
2018-02-25 08:55:17 +01:00
playMusic(1);
2018-01-26 09:51:07 +01:00
}
static void tick(void)
{
Boss *b;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
if (b->health > 0)
2018-01-26 09:51:07 +01:00
{
b->facing = (world.bob->x < b->x) ? FACING_LEFT : FACING_RIGHT;
2018-01-26 09:51:07 +01:00
b->reload = (int) limit(b->reload - 1, 0, FPS);
2018-01-26 09:51:07 +01:00
brakingTimer = (int) limit(--brakingTimer, 0, FPS);
if (brakingTimer > 0)
{
b->dx *= 0.95;
b->dy *= 0.95;
2018-01-26 09:51:07 +01:00
}
}
else
{
2018-02-27 22:55:15 +01:00
addSmokeParticles(b->x + (b->w / 2), b->y, 0);
2018-01-26 09:51:07 +01:00
}
}
static void lookForPlayer(void)
{
2018-02-24 17:53:03 +01:00
float distance;
2019-06-02 17:13:34 +02:00
2018-01-26 09:51:07 +01:00
self->thinkTime = rrnd(0, FPS / 2);
if (rand() % 100 < 5)
{
brakingTimer = rrnd(60, 120);
}
2019-06-02 17:13:34 +02:00
2018-02-24 17:53:03 +01:00
distance = getDistance(world.bob->x, world.bob->y, self->x, self->y);
2018-01-26 09:51:07 +01:00
2018-12-22 22:53:19 +01:00
if (distance > app.config.winHeight)
2018-01-26 09:51:07 +01:00
{
2018-02-24 17:53:03 +01:00
moveTowardsPlayer(1);
2018-01-26 09:51:07 +01:00
return;
}
2019-06-02 17:13:34 +02:00
2018-02-24 17:53:03 +01:00
if (!enemyCanSeePlayer(self))
2018-01-26 09:51:07 +01:00
{
2018-02-24 17:53:03 +01:00
moveTowardsPlayer(1);
2018-01-26 09:51:07 +01:00
return;
}
if (rand() % 100 < 15)
{
selectWeapon();
}
2019-06-02 17:13:34 +02:00
2018-12-22 22:53:19 +01:00
if (distance < app.config.winHeight / 4)
2018-02-24 17:53:03 +01:00
{
moveTowardsPlayer(-6);
}
else
{
moveTowardsPlayer(1);
}
2018-01-26 09:51:07 +01:00
}
static void selectWeapon(void)
{
Boss *b;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
2018-03-29 09:21:36 +02:00
if (world.bob->isOnGround || fabs(self->y - world.bob->y) > 64)
2018-01-26 09:51:07 +01:00
{
b->weaponType = WPN_AIMED_PISTOL;
b->shotsToFire = rrnd(1, 12);
b->action = preFire;
2018-01-26 09:51:07 +01:00
}
else if (rand() % 4 == 0)
{
b->weaponType = WPN_MISSILE;
b->shotsToFire = rrnd(1, 3);
b->action = preFire;
2018-01-26 09:51:07 +01:00
}
else
{
b->weaponType = WPN_PLASMA;
b->shotsToFire = rrnd(1, 12);
b->action = preFire;
2018-01-26 09:51:07 +01:00
}
}
static void walk(void)
{
2018-02-24 17:53:03 +01:00
self->action = (self->health > 0) ? lookForPlayer : die;
2018-01-26 09:51:07 +01:00
}
2018-02-24 17:53:03 +01:00
static void moveTowardsPlayer(int dir)
2018-01-26 09:51:07 +01:00
{
2018-02-24 17:53:03 +01:00
float vel;
2019-06-02 17:13:34 +02:00
2018-02-24 17:53:03 +01:00
vel = 0.5 * dir;
2019-06-02 17:13:34 +02:00
2018-01-26 09:51:07 +01:00
if (brakingTimer == 0)
{
if (world.bob->x < self->x)
{
2018-02-24 17:53:03 +01:00
self->dx -= vel;
2018-01-26 09:51:07 +01:00
}
if (world.bob->x > self->x)
{
2018-02-24 17:53:03 +01:00
self->dx += vel;
2018-01-26 09:51:07 +01:00
}
if (world.bob->y < self->y)
{
2018-02-24 17:53:03 +01:00
self->dy -= vel;
2018-01-26 09:51:07 +01:00
}
if (world.bob->y > self->y)
{
2018-02-24 17:53:03 +01:00
self->dy += vel;
2018-01-26 09:51:07 +01:00
}
self->dx = limit(self->dx, -7.5, 7.5);
self->dy = limit(self->dy, -7.5, 7.5);
}
}
static void preFire(void)
{
Boss *b;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
if (b->reload > 0)
2018-01-26 09:51:07 +01:00
{
return;
}
attack();
b->shotsToFire--;
2018-01-26 09:51:07 +01:00
if (b->shotsToFire == 0)
2018-01-26 09:51:07 +01:00
{
b->walk();
2018-01-26 09:51:07 +01:00
}
}
static void attack(void)
{
Boss *b;
2019-06-02 17:13:34 +02:00
2018-02-25 18:29:44 +01:00
if (self->facing != FACING_DIE)
2018-01-26 09:51:07 +01:00
{
2018-02-25 18:29:44 +01:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
2018-02-25 18:29:44 +01:00
switch (b->weaponType)
{
case WPN_AIMED_PISTOL:
attackPistol();
break;
case WPN_MISSILE:
attackMissile();
break;
case WPN_PLASMA:
attackPlasma();
break;
default:
break;
}
2018-01-26 09:51:07 +01:00
}
}
static void attackPistol(void)
{
int bx, by;
float dx, dy;
2018-01-27 09:13:50 +01:00
Bullet *bullet;
Boss *b;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
2018-01-26 09:51:07 +01:00
bx = world.bob->x + rrnd(-8, 24);
by = world.bob->y + rrnd(-8, 24);
getSlope(bx, by, self->x, self->y, &dx, &dy);
2018-04-24 20:04:08 +02:00
bullet = createBaseBullet((Unit*)self, aimedSprite->w);
2018-01-26 09:51:07 +01:00
bullet->facing = self->facing;
bullet->damage = 1;
bullet->owner = self;
bullet->health = FPS * 3;
bullet->weaponType = WPN_AIMED_PISTOL;
bullet->dx = dx * 12;
bullet->dy = dy * 12;
bullet->sprite[0] = bullet->sprite[1] = aimedSprite;
b->reload = 4;
2018-01-26 09:51:07 +01:00
2018-02-26 19:56:13 +01:00
playBattleSound(SND_MACHINE_GUN, b->uniqueId % MAX_SND_CHANNELS, self->x, self->y);
2018-01-26 09:51:07 +01:00
}
static void attackPlasma(void)
{
Boss *b;
2018-01-27 09:13:50 +01:00
Bullet *bullet;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
2018-04-24 20:04:08 +02:00
bullet = createBaseBullet((Unit*)self, plasmaSprite[0]->w);
2018-01-26 09:51:07 +01:00
bullet->facing = self->facing;
bullet->damage = 2;
bullet->owner = self;
bullet->health = FPS * 3;
bullet->weaponType = WPN_AIMED_PISTOL;
bullet->dx = self->facing == FACING_RIGHT ? 15 : -15;
bullet->dy = 0;
bullet->sprite[0] = plasmaSprite[0];
bullet->sprite[1] = plasmaSprite[1];
b->reload = 4;
2018-01-26 09:51:07 +01:00
2018-02-26 19:56:13 +01:00
playBattleSound(SND_PLASMA, b->uniqueId % MAX_SND_CHANNELS, self->x, self->y);
2018-01-26 09:51:07 +01:00
}
static void attackMissile(void)
{
Boss *b;
2018-01-27 09:13:50 +01:00
Bullet *missile;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
2018-04-24 20:04:08 +02:00
missile = createBaseBullet((Unit*)self, missileSprite[0]->w);
missile->facing = b->facing;
missile->dx = b->facing == FACING_RIGHT ? 15 : -15;
2018-01-26 09:51:07 +01:00
missile->owner = self;
missile->health = FPS * 3;
missile->sprite[0] = missileSprite[0];
missile->sprite[1] = missileSprite[1];
2019-06-02 17:13:34 +02:00
2018-02-24 17:53:03 +01:00
initMissile(missile);
2018-01-26 09:51:07 +01:00
b->reload = 15;
2018-01-26 09:51:07 +01:00
2018-02-26 19:56:13 +01:00
playBattleSound(SND_MISSILE, b->uniqueId % MAX_SND_CHANNELS, self->x, self->y);
2018-01-26 09:51:07 +01:00
}
static void applyDamage(int amount)
{
self->health -= amount;
if (self->health <= 0)
{
self->health = 0;
addExplosion(self->x + self->w / 2, self->y + self->h / 2, 50, self);
}
}
2018-02-24 17:53:03 +01:00
static void die(void)
2018-01-26 09:51:07 +01:00
{
Boss *b;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
b->dx = (randF() - randF()) * 3;
2018-01-26 09:51:07 +01:00
b->spriteTime = 0;
b->spriteFrame = 0;
2018-01-26 09:51:07 +01:00
b->action = die2;
b->thinkTime = 0;
b->flags &= ~(EF_WEIGHTLESS | EF_HALT_AT_EDGE);
2018-01-26 09:51:07 +01:00
b->dy = JUMP_POWER;
2018-01-26 09:51:07 +01:00
if (rand() % 2)
{
2018-02-26 19:56:13 +01:00
playBattleSound(SND_DROID_DIE_1, b->uniqueId % MAX_SND_CHANNELS, self->x, self->y);
2018-01-26 09:51:07 +01:00
}
else
{
2018-02-26 19:56:13 +01:00
playBattleSound(SND_DROID_DIE_2, b->uniqueId % MAX_SND_CHANNELS, self->x, self->y);
2018-01-26 09:51:07 +01:00
}
}
static void die2()
{
Boss *b;
2019-06-02 17:13:34 +02:00
b = (Boss*)self;
2019-06-02 17:13:34 +02:00
if (b->isOnGround)
2018-01-26 09:51:07 +01:00
{
addTeleportStars(self);
2018-02-26 19:56:13 +01:00
playBattleSound(SND_APPEAR, -1, self->x, self->y);
2018-01-26 09:51:07 +01:00
2018-02-24 16:58:14 +01:00
/* don't die! */
b->flags |= EF_GONE;
2018-01-26 09:51:07 +01:00
updateObjective(b->name);
2018-01-26 09:51:07 +01:00
2018-02-13 22:36:42 +01:00
game.stats[STAT_TARGETS_DEFEATED]++;
2018-01-26 09:51:07 +01:00
2018-02-19 20:13:29 +01:00
awardTrophy("EYEDROID_COMMANDER");
2018-02-13 22:36:42 +01:00
game.stats[STAT_ENEMIES_KILLED]++;
2019-06-02 17:13:34 +02:00
2018-02-24 16:58:14 +01:00
b->action = entityIdle;
2018-01-26 09:51:07 +01:00
}
}
static SDL_Rect *getCurrentSprite(void)
2018-01-26 09:51:07 +01:00
{
2018-02-24 16:58:14 +01:00
Sprite *s;
2019-06-02 17:13:34 +02:00
2018-02-24 16:58:14 +01:00
s = (self->alive == ALIVE_ALIVE) ? self->sprite[self->facing] : self->sprite[FACING_DIE];
2019-06-02 17:13:34 +02:00
2018-02-24 16:58:14 +01:00
if (self->spriteFrame >= s->numFrames)
2018-01-26 09:51:07 +01:00
{
2018-02-25 13:36:34 +01:00
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_WARN, "WARNING: %s (%d) bad sprite frames - %d > %d\n", self->name, self->type, self->spriteFrame, s->numFrames);
2018-02-24 16:58:14 +01:00
self->spriteFrame = 0;
2018-01-26 09:51:07 +01:00
}
2019-06-02 17:13:34 +02:00
2018-02-24 16:58:14 +01:00
return &s->frames[self->spriteFrame]->rect;
2018-01-26 09:51:07 +01:00
}