2018-01-26 19:27:15 +01:00
|
|
|
/*
|
|
|
|
Copyright (C) 2018 Parallel Realities
|
|
|
|
|
|
|
|
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 "tankCommander.h"
|
|
|
|
|
|
|
|
static void activate(int activate);
|
|
|
|
static void tick(void);
|
|
|
|
static void walk(void);
|
|
|
|
static void attackPistol(void);
|
|
|
|
static void attackMissile(void);
|
|
|
|
static void die1(void);
|
|
|
|
static void die2(void);
|
|
|
|
static void attack(void);
|
|
|
|
static void applyDamage(int amount);
|
|
|
|
static SDL_Rect getBounds(void);
|
|
|
|
static void preFire(void);
|
|
|
|
static void selectWeapon(void);
|
|
|
|
static void moveTowardsPlayer(void);
|
|
|
|
|
|
|
|
static Entity *tankTrack;
|
|
|
|
static int brakingTimer;
|
|
|
|
static int missileSprite[2];
|
|
|
|
static int aimedSprite;
|
|
|
|
|
|
|
|
void initTankCommander(Entity *e)
|
|
|
|
{
|
|
|
|
initBoss(e);
|
|
|
|
|
|
|
|
STRNCPY(e->name, "Tank Commander", MAX_NAME_LENGTH);
|
|
|
|
|
|
|
|
e->sprite[FACING_LEFT] = getSpriteIndex("TankCommanderLeft");
|
|
|
|
e->sprite[FACING_RIGHT] = getSpriteIndex("TankCommanderRight");
|
|
|
|
e->sprite[FACING_DIE] = getSpriteIndex("TankCommanderDie");
|
|
|
|
|
|
|
|
e->flags |= EF_EXPLODES;
|
|
|
|
|
|
|
|
e->health = e->healthMax = 400;
|
|
|
|
|
|
|
|
e->activate = activate;
|
|
|
|
e->walk = walk;
|
|
|
|
e->tick = tick;
|
|
|
|
e->die = die1;
|
|
|
|
e->applyDamage = applyDamage;
|
|
|
|
e->getBounds = getBounds;
|
|
|
|
|
|
|
|
brakingTimer = 0;
|
|
|
|
|
|
|
|
world.boss = e;
|
|
|
|
|
|
|
|
aimedSprite = getSpriteIndex("AimedShot");
|
|
|
|
|
|
|
|
missileSprite[0] = getSpriteIndex("MissileRight");
|
|
|
|
missileSprite[1] = getSpriteIndex("MissileLeft");
|
|
|
|
|
|
|
|
initTankTrack(tankTrack);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void activate(int activate)
|
|
|
|
{
|
|
|
|
self->flags &= ~EF_GONE;
|
|
|
|
tankTrack->flags &= ~EF_GONE;
|
|
|
|
|
|
|
|
world.isBossActive = 1;
|
|
|
|
|
|
|
|
addTeleportStars(self);
|
|
|
|
addTeleportStars(tankTrack);
|
|
|
|
|
|
|
|
playMusic("", 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tick(void)
|
|
|
|
{
|
|
|
|
if (self->health > 0)
|
|
|
|
{
|
|
|
|
self->facing = (world.bob->x < self->x) ? FACING_LEFT : FACING_RIGHT;
|
|
|
|
|
|
|
|
self->reload = limit(self->reload - 1, 0, FPS);
|
|
|
|
|
|
|
|
brakingTimer = limit(brakingTimer - 1, 0, FPS);
|
|
|
|
|
|
|
|
if (brakingTimer > 0)
|
|
|
|
{
|
|
|
|
self->dx *= 0.9;
|
|
|
|
self->dy *= 0.9;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void lookForPlayer(void)
|
|
|
|
{
|
|
|
|
self->thinkTime = rrnd(0, FPS / 2);
|
|
|
|
|
|
|
|
if (rand() % 10 == 0)
|
|
|
|
{
|
|
|
|
brakingTimer = rrnd(60, 120);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getDistance(world.bob->x, world.bob->y, self->x, self->y) > 650)
|
|
|
|
{
|
|
|
|
moveTowardsPlayer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!enemyCanSeePlayer(self))
|
|
|
|
{
|
|
|
|
moveTowardsPlayer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rand() % 100 <15)
|
|
|
|
{
|
|
|
|
selectWeapon();
|
|
|
|
}
|
|
|
|
|
|
|
|
moveTowardsPlayer();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void walk(void)
|
|
|
|
{
|
|
|
|
self->action = (self->health > 0) ? lookForPlayer : die1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void moveTowardsPlayer(void)
|
|
|
|
{
|
|
|
|
if (brakingTimer == 0)
|
|
|
|
{
|
|
|
|
if (world.bob->x < self->x)
|
|
|
|
{
|
|
|
|
self->dx -= 0.5;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (world.bob->x > self->x)
|
|
|
|
{
|
|
|
|
self->dx += 0.5;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
self->dx = limit(self->dx, -7.5, 7.5);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void selectWeapon(void)
|
|
|
|
{
|
|
|
|
if (abs(self->y - world.bob->y) > 64)
|
|
|
|
{
|
|
|
|
self->weaponType = WPN_AIMED_PISTOL;
|
|
|
|
self->shotsToFire = rrnd(4, 12);
|
|
|
|
self->action = preFire;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
self->weaponType = WPN_MISSILE;
|
|
|
|
self->shotsToFire = rrnd(1, 3);
|
|
|
|
self->action = preFire;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void preFire(void)
|
|
|
|
{
|
|
|
|
if (self->reload > 0)
|
|
|
|
{
|
|
|
|
moveTowardsPlayer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
attack();
|
|
|
|
|
|
|
|
self->shotsToFire--;
|
|
|
|
|
|
|
|
if (self->shotsToFire == 0)
|
|
|
|
{
|
|
|
|
self->walk();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void attack(void)
|
|
|
|
{
|
|
|
|
switch (self->weaponType)
|
|
|
|
{
|
|
|
|
case WPN_AIMED_PISTOL:
|
|
|
|
attackPistol();
|
|
|
|
break;
|
|
|
|
case WPN_MISSILE:
|
|
|
|
attackMissile();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void attackPistol(void)
|
|
|
|
{
|
|
|
|
int bx, by;
|
|
|
|
float dx, dy;
|
2018-01-27 09:13:50 +01:00
|
|
|
Bullet *bullet;
|
2018-01-26 19:27:15 +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);
|
|
|
|
|
|
|
|
bullet = createBaseBullet(self);
|
|
|
|
bullet->x = (self->x + self->w / 2);
|
|
|
|
bullet->y = self->y + 30;
|
|
|
|
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;
|
|
|
|
|
|
|
|
self->reload = 4;
|
|
|
|
|
|
|
|
playSound(SND_MACHINE_GUN, CH_WEAPON);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void attackMissile(void)
|
|
|
|
{
|
2018-01-27 09:13:50 +01:00
|
|
|
Bullet *missile;
|
2018-01-26 19:27:15 +01:00
|
|
|
|
|
|
|
missile = createBaseBullet(self);
|
|
|
|
missile->x = self->x + self->w / 2;
|
|
|
|
missile->y = self->y + 30;
|
|
|
|
missile->facing = self->facing;
|
|
|
|
missile->dx = self->facing == FACING_RIGHT ? 15 : -15;
|
|
|
|
missile->dy = world.bob->y - missile->y;
|
|
|
|
missile->dy *= 0.01;
|
|
|
|
missile->owner = self;
|
|
|
|
missile->health = FPS * 3;
|
|
|
|
missile->sprite[0] = missileSprite[0];
|
|
|
|
missile->sprite[1] = missileSprite[1];
|
|
|
|
|
|
|
|
self->reload = 15;
|
|
|
|
|
|
|
|
playSound(SND_MISSILE, CH_WEAPON);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void die1(void)
|
|
|
|
{
|
|
|
|
self->flags |= EF_BOUNCES;
|
|
|
|
|
|
|
|
self->action = die2;
|
|
|
|
self->thinkTime = 0;
|
|
|
|
|
|
|
|
self->spriteTime = 0;
|
|
|
|
self->spriteFrame = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void die2(void)
|
|
|
|
{
|
|
|
|
int mx, my;
|
|
|
|
|
|
|
|
self->health--;
|
|
|
|
|
|
|
|
if (self->health % 3 == 0)
|
|
|
|
{
|
|
|
|
mx = (int) ((self->x + (self->w / 2)) / MAP_TILE_SIZE);
|
|
|
|
my = (int) ((self->y + self->h) / MAP_TILE_SIZE);
|
|
|
|
|
|
|
|
addScorchDecal(mx, my);
|
|
|
|
|
|
|
|
addExplosion(self->x + rand() % self->w, self->y + rand() % self->h, 50, self);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self->health <= -100)
|
|
|
|
{
|
|
|
|
addTeleportStars(self);
|
|
|
|
addTeleportStars(tankTrack);
|
|
|
|
|
|
|
|
playSound(SND_APPEAR, CH_ANY);
|
|
|
|
|
|
|
|
self->alive = tankTrack->alive = ALIVE_DEAD;
|
|
|
|
|
|
|
|
updateObjective(self->name);
|
|
|
|
|
|
|
|
addDefeatedTarget(self->name);
|
|
|
|
|
|
|
|
game.enemiesKilled++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static SDL_Rect getBounds(void)
|
|
|
|
{
|
|
|
|
if (self->facing == FACING_LEFT)
|
|
|
|
{
|
|
|
|
self->bounds.x = self->x + 98;
|
|
|
|
self->bounds.y = self->y;
|
|
|
|
self->bounds.w = 140;
|
|
|
|
self->bounds.h = self->h;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
self->bounds.x = self->x;
|
|
|
|
self->bounds.y = self->y;
|
|
|
|
self->bounds.w = 140;
|
|
|
|
self->bounds.h = self->h;
|
|
|
|
}
|
|
|
|
|
|
|
|
return self->bounds;
|
|
|
|
}
|