tbftss/src/battle/mine.c

268 lines
5.3 KiB
C
Raw Normal View History

2016-03-30 08:22:58 +02:00
/*
Copyright (C) 2015-2019,2022 Parallel Realities
2016-03-30 08:22:58 +02: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 "../common.h"
2022-07-31 11:43:20 +02:00
#include "../battle/effects.h"
#include "../battle/entities.h"
#include "../battle/fighters.h"
#include "../battle/objectives.h"
2022-07-31 11:43:20 +02:00
#include "../battle/quadtree.h"
#include "../battle/script.h"
2022-07-31 11:43:20 +02:00
#include "../game/trophies.h"
#include "../system/atlas.h"
2022-07-31 11:43:20 +02:00
#include "../system/sound.h"
#include "../system/util.h"
#include "mine.h"
2022-07-31 11:43:20 +02:00
#define DAMAGE_RANGE 250
#define SYSTEM_POWER 50
#define TRIGGER_RANGE 150
2022-07-31 11:43:20 +02:00
extern Battle battle;
extern Entity *player;
extern Entity *self;
2016-03-30 08:22:58 +02:00
static void think(void);
2016-03-30 23:41:34 +02:00
static void die(void);
2016-03-30 08:22:58 +02:00
static void lookForFighters(void);
2016-04-14 12:56:42 +02:00
static void lookForPlayer(void);
2016-04-01 11:49:41 +02:00
static void doSplashDamage(void);
2018-12-06 09:37:19 +01:00
static AtlasImage *mineWarning = NULL;
static AtlasImage *mineNormal = NULL;
static AtlasImage *shadowMine = NULL;
2016-03-30 08:22:58 +02:00
2016-04-14 12:56:42 +02:00
Entity *spawnMine(int type)
2016-03-30 08:22:58 +02:00
{
Entity *mine = spawnEntity();
2019-11-07 09:13:57 +01:00
2016-04-14 12:56:42 +02:00
if (!mineWarning)
2016-04-01 15:19:55 +02:00
{
2018-12-06 09:37:19 +01:00
shadowMine = getAtlasImage("gfx/entities/shadowMine.png");
mineNormal = getAtlasImage("gfx/entities/mine.png");
mineWarning = getAtlasImage("gfx/entities/mineWarning.png");
2016-04-01 15:19:55 +02:00
}
2016-03-30 08:22:58 +02:00
2016-04-14 12:56:42 +02:00
STRNCPY(mine->name, "Mine", MAX_NAME_LENGTH);
mine->type = type;
2016-03-30 08:22:58 +02:00
mine->health = mine->maxHealth = 1;
2016-04-01 18:02:36 +02:00
mine->speed = 1;
2016-04-01 11:49:41 +02:00
mine->systemPower = SYSTEM_POWER;
2016-04-14 12:56:42 +02:00
mine->texture = (type == ET_MINE) ? mineNormal : shadowMine;
2016-03-30 08:22:58 +02:00
mine->action = think;
2016-03-30 23:41:34 +02:00
mine->die = die;
2022-07-31 11:43:20 +02:00
mine->flags = EF_TAKES_DAMAGE + EF_NO_PLAYER_TARGET + EF_SHORT_RADAR_RANGE + EF_NON_SOLID + EF_NO_HEALTH_BAR;
2019-11-07 09:13:57 +01:00
2016-04-14 12:56:42 +02:00
if (type == ET_SHADOW_MINE)
{
mine->flags &= ~EF_NO_PLAYER_TARGET;
mine->speed = 100 + rand() % 100;
mine->speed *= 0.01;
}
2019-11-07 09:13:57 +01:00
2018-12-06 09:37:19 +01:00
mine->w = mine->texture->rect.w;
mine->h = mine->texture->rect.h;
2016-03-30 08:22:58 +02:00
return mine;
}
static void think(void)
{
2016-04-14 12:56:42 +02:00
self->texture = (self->type == ET_MINE) ? mineNormal : shadowMine;
2019-11-07 09:13:57 +01:00
2016-03-30 08:22:58 +02:00
self->angle += 0.1;
2019-11-07 09:13:57 +01:00
2016-03-30 08:22:58 +02:00
if (self->angle >= 360)
{
self->angle -= 360;
}
2019-11-07 09:13:57 +01:00
2016-04-01 18:02:36 +02:00
self->dx *= 0.99;
self->dy *= 0.99;
2019-11-07 09:13:57 +01:00
2016-04-14 12:56:42 +02:00
if (self->type == ET_MINE)
{
lookForFighters();
}
else
{
lookForPlayer();
}
2019-11-07 09:13:57 +01:00
2016-04-01 15:19:55 +02:00
if (self->systemPower < SYSTEM_POWER && battle.stats[STAT_TIME] % 8 < 4)
2016-04-01 11:49:41 +02:00
{
playBattleSound(SND_MINE_WARNING, self->x, self->y);
2019-11-07 09:13:57 +01:00
2016-04-01 11:49:41 +02:00
self->texture = mineWarning;
}
2016-03-30 08:22:58 +02:00
}
2016-04-01 11:49:41 +02:00
static void lookForFighters(void)
{
Entity *e, **candidates;
2022-07-31 11:43:20 +02:00
int i;
2016-04-01 11:49:41 +02:00
2016-05-15 09:19:26 +02:00
candidates = getAllEntsInRadius(self->x, self->y, DAMAGE_RANGE, self);
2016-04-01 11:49:41 +02:00
2022-07-31 11:43:20 +02:00
for (i = 0, e = candidates[i]; e != NULL; e = candidates[++i])
2016-04-01 11:49:41 +02:00
{
2016-04-01 15:19:55 +02:00
if (e->side != self->side && e->health > 0 && e->type == ET_FIGHTER && getDistance(self->x, self->y, e->x, e->y) <= TRIGGER_RANGE)
2016-04-01 11:49:41 +02:00
{
self->systemPower--;
2019-11-07 09:13:57 +01:00
2016-04-01 11:49:41 +02:00
if (self->systemPower <= 0)
{
self->health = 0;
}
2019-11-07 09:13:57 +01:00
2016-04-01 11:49:41 +02:00
return;
}
}
2019-11-07 09:13:57 +01:00
2016-04-01 11:49:41 +02:00
self->systemPower = SYSTEM_POWER;
}
2016-04-14 12:56:42 +02:00
static void lookForPlayer(void)
{
float dx, dy, norm;
2022-07-31 11:43:20 +02:00
int distance;
2019-11-07 09:13:57 +01:00
2016-05-15 09:19:26 +02:00
if (player->alive == ALIVE_ALIVE)
2016-04-14 12:56:42 +02:00
{
distance = getDistance(self->x, self->y, player->x, player->y);
2019-11-07 09:13:57 +01:00
if (distance < SCREEN_WIDTH * 2)
2016-04-14 12:56:42 +02:00
{
dx = player->x - self->x;
dy = player->y - self->y;
}
else
{
if (--self->aiActionTime > 0)
{
return;
}
2019-11-07 09:13:57 +01:00
dx = rand() % 1000;
dx -= rand() % 1000;
2019-11-07 09:13:57 +01:00
dy = rand() % 1000;
dy -= rand() % 1000;
2019-11-07 09:13:57 +01:00
self->aiActionTime = FPS * (5 + (rand() % 15));
}
2019-11-07 09:13:57 +01:00
norm = sqrt(dx * dx + dy * dy);
2019-11-07 09:13:57 +01:00
dx /= norm;
dy /= norm;
2019-11-07 09:13:57 +01:00
self->dx = dx * self->speed;
self->dy = dy * self->speed;
2019-11-07 09:13:57 +01:00
if (distance <= TRIGGER_RANGE)
{
self->systemPower--;
2019-11-07 09:13:57 +01:00
if (self->systemPower <= 0)
2016-04-14 12:56:42 +02:00
{
self->health = 0;
2016-04-14 12:56:42 +02:00
}
2019-11-07 09:13:57 +01:00
return;
2016-04-14 12:56:42 +02:00
}
}
2019-11-07 09:13:57 +01:00
2016-04-14 12:56:42 +02:00
self->systemPower = SYSTEM_POWER;
}
2016-04-01 11:49:41 +02:00
2016-03-30 23:41:34 +02:00
static void die(void)
{
if (self->killedBy == player)
{
battle.stats[STAT_MINES_DESTROYED]++;
}
2019-11-07 09:13:57 +01:00
2016-03-30 23:41:34 +02:00
addMineExplosion();
2019-11-07 09:13:57 +01:00
2016-04-01 11:49:41 +02:00
doSplashDamage();
2019-11-07 09:13:57 +01:00
2016-04-01 11:49:41 +02:00
playBattleSound(SND_EXPLOSION_5, self->x, self->y);
2019-11-07 09:13:57 +01:00
2016-03-30 23:41:34 +02:00
self->alive = ALIVE_DEAD;
2019-11-07 09:13:57 +01:00
2016-04-14 12:56:42 +02:00
updateObjective(self->name, TT_DESTROY);
2019-11-07 09:13:57 +01:00
runScriptFunction("MINES_DESTROYED %d", battle.stats[STAT_MINES_DESTROYED]);
2016-03-30 23:41:34 +02:00
}
2016-04-01 11:49:41 +02:00
static void doSplashDamage(void)
2016-03-30 08:22:58 +02:00
{
Entity *e, **candidates;
2022-07-31 11:43:20 +02:00
int i, dist, kills;
float damage, percent;
2016-03-30 08:22:58 +02:00
2016-05-15 09:19:26 +02:00
candidates = getAllEntsInRadius(self->x, self->y, DAMAGE_RANGE, self);
2019-11-07 09:13:57 +01:00
2016-05-15 09:19:26 +02:00
kills = 0;
2016-03-30 08:22:58 +02:00
2022-07-31 11:43:20 +02:00
for (i = 0, e = candidates[i]; e != NULL; e = candidates[++i])
2016-03-30 08:22:58 +02:00
{
2016-04-01 18:02:36 +02:00
if (e->health > 0 && (e->type == ET_FIGHTER || e->type == ET_MINE) && !(e->flags & EF_IMMORTAL))
2016-03-30 08:22:58 +02:00
{
2016-04-01 11:49:41 +02:00
dist = getDistance(self->x, self->y, e->x, e->y);
2019-11-07 09:13:57 +01:00
2016-04-01 11:49:41 +02:00
if (dist <= DAMAGE_RANGE)
{
percent = dist;
percent /= DAMAGE_RANGE;
percent = 1 - percent;
2019-11-07 09:13:57 +01:00
2016-05-15 09:19:26 +02:00
damage = DAMAGE_RANGE;
2016-04-01 11:49:41 +02:00
damage *= percent;
2019-11-07 09:13:57 +01:00
2016-04-01 18:02:36 +02:00
if (e->type == ET_FIGHTER)
{
damageFighter(e, damage, 0);
2019-11-07 09:13:57 +01:00
2016-05-15 09:19:26 +02:00
if (self->killedBy == player && e != player && e->health <= 0)
{
kills++;
}
2016-04-01 18:02:36 +02:00
}
else if (e->type == ET_MINE)
{
e->dx = e->x - self->x;
e->dy = e->y - self->y;
2019-11-07 09:13:57 +01:00
2016-04-01 18:02:36 +02:00
e->dx *= 0.01;
e->dy *= 0.01;
}
2016-04-01 11:49:41 +02:00
}
2016-03-30 08:22:58 +02:00
}
}
2019-11-07 09:13:57 +01:00
2016-05-15 09:19:26 +02:00
if (kills >= 2)
{
awardTrophy("2_BIRDS");
}
2016-03-30 08:22:58 +02:00
}