726 lines
14 KiB
C
726 lines
14 KiB
C
/*
|
|
Copyright (C) 2015-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 "entities.h"
|
|
|
|
static void drawEntity(Entity *e);
|
|
static void doEntity(void);
|
|
static void alignComponents(void);
|
|
static void activateEpicFighters(int side);
|
|
static void restrictToBattleArea(Entity *e);
|
|
static void drawTargetRects(Entity *e);
|
|
static void drawHealthBar(Entity *e);
|
|
static int drawComparator(const void *a, const void *b);
|
|
static void notifyNewArrivals(void);
|
|
static int isComponent(Entity *e);
|
|
|
|
static Entity deadHead;
|
|
static Entity *deadTail;
|
|
static int disabledGlow;
|
|
static int disabledGlowDir;
|
|
|
|
void initEntities(void)
|
|
{
|
|
memset(&deadHead, 0, sizeof(Entity));
|
|
|
|
deadTail = &deadHead;
|
|
|
|
disabledGlow = DISABLED_GLOW_MAX;
|
|
disabledGlowDir = -DISABLED_GLOW_SPEED;
|
|
}
|
|
|
|
Entity *spawnEntity(void)
|
|
{
|
|
Entity *e = malloc(sizeof(Entity));
|
|
memset(e, 0, sizeof(Entity));
|
|
e->active = 1;
|
|
|
|
battle.entityTail->next = e;
|
|
battle.entityTail = e;
|
|
|
|
return e;
|
|
}
|
|
|
|
void doEntities(void)
|
|
{
|
|
int numAllies, numEnemies;
|
|
int numActiveAllies, numActiveEnemies;
|
|
int numSpawnedEnemies;
|
|
Entity *e, *prev;
|
|
|
|
prev = &battle.entityHead;
|
|
|
|
battle.hasThreats = numAllies = numEnemies = numActiveAllies = numActiveEnemies = numSpawnedEnemies = 0;
|
|
|
|
if (dev.playerImmortal)
|
|
{
|
|
player->health = player->maxHealth;
|
|
player->shield = player->maxShield;
|
|
}
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
removeFromQuadtree(e, &battle.quadtree);
|
|
|
|
if (dev.allImmortal)
|
|
{
|
|
e->health = e->maxHealth;
|
|
e->shield = e->maxShield;
|
|
}
|
|
|
|
if (e->active)
|
|
{
|
|
self = e;
|
|
|
|
e->reload = MAX(e->reload - 1, 0);
|
|
|
|
if (e->shieldRechargeRate)
|
|
{
|
|
if (e->shield >= 0)
|
|
{
|
|
if (--e->shieldRecharge <= 0)
|
|
{
|
|
e->shield = MIN(e->shield + 1, e->maxShield);
|
|
|
|
e->shieldRecharge = e->shieldRechargeRate;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
e->shield++;
|
|
}
|
|
}
|
|
|
|
e->armourHit = MAX(e->armourHit - 25, 0);
|
|
e->shieldHit = MAX(e->shieldHit - 5, 0);
|
|
e->systemHit = MAX(e->systemHit - 25, 0);
|
|
|
|
e->aiDamageTimer = MAX(e->aiDamageTimer - 1, 0);
|
|
if (!e->aiDamageTimer)
|
|
{
|
|
e->aiDamagePerSec = 0;
|
|
e->aiFlags &= ~AIF_EVADE;
|
|
}
|
|
|
|
switch (e->type)
|
|
{
|
|
case ET_FIGHTER:
|
|
doFighter();
|
|
break;
|
|
|
|
case ET_CAPITAL_SHIP:
|
|
doCapitalShip();
|
|
break;
|
|
|
|
default:
|
|
doEntity();
|
|
break;
|
|
}
|
|
|
|
if (e->alive == ALIVE_ALIVE || e->alive == ALIVE_DYING)
|
|
{
|
|
if (e->action != NULL)
|
|
{
|
|
if (dev.noEntityActions)
|
|
{
|
|
e->thinkTime = 2;
|
|
}
|
|
|
|
if (--e->thinkTime <= 0)
|
|
{
|
|
e->thinkTime = 0;
|
|
e->action();
|
|
}
|
|
}
|
|
|
|
doRope(e);
|
|
|
|
restrictToBattleArea(e);
|
|
|
|
if (!e->speed)
|
|
{
|
|
e->dx = e->dy = 0;
|
|
}
|
|
|
|
e->x += e->dx;
|
|
e->y += e->dy;
|
|
|
|
addToQuadtree(e, &battle.quadtree);
|
|
}
|
|
else
|
|
{
|
|
if (e == battle.entityTail)
|
|
{
|
|
battle.entityTail = prev;
|
|
}
|
|
|
|
if (e == battle.missionTarget)
|
|
{
|
|
battle.missionTarget = NULL;
|
|
}
|
|
|
|
if (e->killedBy == player && battle.hasSuspicionLevel)
|
|
{
|
|
if (e->aiFlags & (AIF_AVOIDS_COMBAT|AIF_DEFENSIVE))
|
|
{
|
|
battle.suspicionLevel -= (MAX_SUSPICION_LEVEL * 0.5);
|
|
}
|
|
else
|
|
{
|
|
battle.suspicionLevel -= (MAX_SUSPICION_LEVEL * 0.12);
|
|
}
|
|
}
|
|
|
|
if (e == player)
|
|
{
|
|
battle.playerSelect = battle.isEpic;
|
|
}
|
|
|
|
cutRope(e);
|
|
|
|
prev->next = e->next;
|
|
|
|
e->next = NULL;
|
|
|
|
deadTail->next = e;
|
|
deadTail = e;
|
|
|
|
/* actually just creates another fighter in this one's place */
|
|
if (e->type == ET_FIGHTER && battle.isEpic && e->side != player->side && battle.unlimitedEnemies)
|
|
{
|
|
resetFighter(e);
|
|
}
|
|
|
|
e = prev;
|
|
}
|
|
}
|
|
|
|
if (e->type == ET_FIGHTER || e->type == ET_CAPITAL_SHIP)
|
|
{
|
|
if (e->side == player->side)
|
|
{
|
|
numAllies++;
|
|
|
|
if (e->health > 0 && e->active)
|
|
{
|
|
numActiveAllies++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
numEnemies++;
|
|
|
|
if (e->health > 0 && e->active)
|
|
{
|
|
numActiveEnemies++;
|
|
|
|
if (e->spawned)
|
|
{
|
|
numSpawnedEnemies++;
|
|
}
|
|
}
|
|
|
|
if (!(e->flags & EF_DISABLED) || battle.isEpic)
|
|
{
|
|
battle.hasThreats = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
prev = e;
|
|
}
|
|
|
|
battle.numAllies = (battle.isEpic) ? numAllies : numActiveAllies;
|
|
battle.numEnemies = (battle.isEpic) ? numEnemies : numActiveEnemies;
|
|
|
|
if (battle.status == MS_IN_PROGRESS && battle.stats[STAT_TIME] % (FPS * 30) == 0)
|
|
{
|
|
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_DEBUG, "numEnemies=%d, numActiveEnemies=%d, hasThreats=%d", numEnemies, numActiveEnemies, battle.hasThreats);
|
|
}
|
|
|
|
if (battle.isEpic && battle.stats[STAT_TIME] % FPS == 0)
|
|
{
|
|
if (numActiveAllies < battle.epicFighterLimit)
|
|
{
|
|
activateEpicFighters(SIDE_ALLIES);
|
|
}
|
|
|
|
numActiveEnemies -= numSpawnedEnemies;
|
|
|
|
if (numActiveEnemies < battle.epicFighterLimit)
|
|
{
|
|
activateEpicFighters(SIDE_NONE);
|
|
}
|
|
}
|
|
|
|
alignComponents();
|
|
|
|
disabledGlow = MAX(DISABLED_GLOW_MIN, MIN(disabledGlow + disabledGlowDir, DISABLED_GLOW_MAX));
|
|
|
|
if (disabledGlow <= DISABLED_GLOW_MIN)
|
|
{
|
|
disabledGlowDir = DISABLED_GLOW_SPEED;
|
|
}
|
|
else if (disabledGlow >= DISABLED_GLOW_MAX)
|
|
{
|
|
disabledGlowDir = -DISABLED_GLOW_SPEED;
|
|
}
|
|
}
|
|
|
|
static void restrictToBattleArea(Entity *e)
|
|
{
|
|
float force;
|
|
|
|
if (e->x <= BATTLE_AREA_EDGE)
|
|
{
|
|
force = BATTLE_AREA_EDGE - e->x;
|
|
e->dx += force * 0.001;
|
|
e->dx *= 0.95;
|
|
}
|
|
|
|
if (e->y <= BATTLE_AREA_EDGE)
|
|
{
|
|
force = BATTLE_AREA_EDGE - e->y;
|
|
e->dy += force * 0.001;
|
|
e->dy *= 0.95;
|
|
}
|
|
|
|
if (e->x >= BATTLE_AREA_WIDTH - BATTLE_AREA_EDGE)
|
|
{
|
|
force = e->x - (BATTLE_AREA_WIDTH - BATTLE_AREA_EDGE);
|
|
e->dx -= force * 0.001;
|
|
e->dx *= 0.95;
|
|
}
|
|
|
|
if (e->y >= BATTLE_AREA_HEIGHT - BATTLE_AREA_EDGE)
|
|
{
|
|
force = e->y - (BATTLE_AREA_HEIGHT - BATTLE_AREA_EDGE);
|
|
e->dy -= force * 0.001;
|
|
e->dy *= 0.95;
|
|
}
|
|
}
|
|
|
|
static void doEntity(void)
|
|
{
|
|
if (self->die)
|
|
{
|
|
if (self->health <= 0 && self->alive == ALIVE_ALIVE)
|
|
{
|
|
self->health = 0;
|
|
self->alive = ALIVE_DYING;
|
|
self->die();
|
|
|
|
if (self == battle.missionTarget)
|
|
{
|
|
battle.missionTarget = NULL;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (self->alive == ALIVE_DYING)
|
|
{
|
|
self->alive = ALIVE_DEAD;
|
|
}
|
|
else if (self->health <= 0)
|
|
{
|
|
self->alive = ALIVE_DYING;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void alignComponents(void)
|
|
{
|
|
Entity *e;
|
|
float x, y;
|
|
float c, s;
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (isComponent(e))
|
|
{
|
|
removeFromQuadtree(e, &battle.quadtree);
|
|
|
|
s = sin(TO_RAIDANS(e->owner->angle));
|
|
c = cos(TO_RAIDANS(e->owner->angle));
|
|
|
|
x = (e->offsetX * c) - (e->offsetY * s);
|
|
y = (e->offsetX * s) + (e->offsetY * c);
|
|
|
|
x += e->owner->x;
|
|
y += e->owner->y;
|
|
|
|
e->x = x;
|
|
e->y = y;
|
|
|
|
if (e->flags & EF_STATIC)
|
|
{
|
|
e->angle = e->owner->angle;
|
|
}
|
|
|
|
addToQuadtree(e, &battle.quadtree);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int isComponent(Entity *e)
|
|
{
|
|
return (e->type == ET_COMPONENT || e->type == ET_COMPONENT_GUN || e->type == ET_COMPONENT_ENGINE);
|
|
}
|
|
|
|
void drawEntities(void)
|
|
{
|
|
int i;
|
|
Entity *e, **candidates;
|
|
|
|
candidates = getAllEntsWithin(battle.camera.x, battle.camera.y, SCREEN_WIDTH, SCREEN_HEIGHT, NULL);
|
|
|
|
/* counting entities to draw */
|
|
for (i = 0, e = candidates[i] ; e != NULL ; e = candidates[++i]) {};
|
|
|
|
qsort(candidates, i, sizeof(Entity*), drawComparator);
|
|
|
|
for (i = 0, e = candidates[i] ; e != NULL ; e = candidates[++i])
|
|
{
|
|
self = e;
|
|
|
|
if (e->draw)
|
|
{
|
|
e->draw();
|
|
}
|
|
else
|
|
{
|
|
drawEntity(e);
|
|
}
|
|
|
|
drawHealthBar(e);
|
|
|
|
drawTargetRects(e);
|
|
|
|
drawRope(e);
|
|
}
|
|
}
|
|
|
|
static void drawEntity(Entity *e)
|
|
{
|
|
setAtlasColor(255, 255, 255, 255);
|
|
|
|
if (e->armourHit > 0)
|
|
{
|
|
setAtlasColor(255, 255 - e->armourHit, 255 - e->armourHit, 255);
|
|
}
|
|
|
|
if (e->systemHit > 0)
|
|
{
|
|
setAtlasColor(255 - e->systemHit, 255, 255, 255);
|
|
}
|
|
|
|
if (e->flags & EF_DISABLED)
|
|
{
|
|
setAtlasColor(disabledGlow, disabledGlow, 255, 255);
|
|
}
|
|
|
|
blitRotated(e->texture, e->x - battle.camera.x, e->y - battle.camera.y, e->angle);
|
|
|
|
if (e->shieldHit > 0)
|
|
{
|
|
drawShieldHitEffect(e);
|
|
}
|
|
}
|
|
|
|
static void drawHealthBar(Entity *e)
|
|
{
|
|
SDL_Rect r;
|
|
|
|
if (app.gameplay.healthBars && !(e->flags & EF_NO_HEALTH_BAR) && e->health > 0)
|
|
{
|
|
r.x = e->x - (e->w / 2) - battle.camera.x;
|
|
r.y = e->y - e->h - battle.camera.y;
|
|
r.w = 32;
|
|
r.h = 1;
|
|
|
|
if (e->side == player->side || e->flags & EF_FRIENDLY_HEALTH_BAR)
|
|
{
|
|
SDL_SetRenderDrawColor(app.renderer, 0, 128, 0, 255);
|
|
}
|
|
else
|
|
{
|
|
SDL_SetRenderDrawColor(app.renderer, 128, 0, 0, 255);
|
|
}
|
|
|
|
SDL_RenderFillRect(app.renderer, &r);
|
|
|
|
r.w = 32 * (e->health * 1.0f / e->maxHealth);
|
|
|
|
if (e->side == player->side || e->flags & EF_FRIENDLY_HEALTH_BAR)
|
|
{
|
|
SDL_SetRenderDrawColor(app.renderer, 0, 255, 0, 255);
|
|
}
|
|
else
|
|
{
|
|
SDL_SetRenderDrawColor(app.renderer, 255, 0, 0, 255);
|
|
}
|
|
|
|
SDL_RenderFillRect(app.renderer, &r);
|
|
}
|
|
}
|
|
|
|
static void drawTargetRects(Entity *e)
|
|
{
|
|
SDL_Rect r;
|
|
|
|
int size = MAX(e->w, e->h) + 16;
|
|
|
|
if (player->alive == ALIVE_ALIVE && e == player->target)
|
|
{
|
|
r.x = e->x - (size / 2) - battle.camera.x;
|
|
r.y = e->y - (size / 2) - battle.camera.y;
|
|
r.w = size;
|
|
r.h = size;
|
|
|
|
SDL_SetRenderDrawColor(app.renderer, 255, 0, 0, 255);
|
|
SDL_RenderDrawRect(app.renderer, &r);
|
|
}
|
|
|
|
if ((e == battle.missionTarget || e->flags & EF_MISSION_TARGET) && (e->flags & EF_NO_MT_BOX) == 0)
|
|
{
|
|
r.x = e->x - (size / 2) - battle.camera.x - 4;
|
|
r.y = e->y - (size / 2) - battle.camera.y - 4;
|
|
r.w = size + 8;
|
|
r.h = size + 8;
|
|
|
|
SDL_SetRenderDrawColor(app.renderer, 0, 255, 0, 255);
|
|
SDL_RenderDrawRect(app.renderer, &r);
|
|
}
|
|
|
|
if (e == battle.messageSpeaker && e != player && battle.stats[STAT_TIME] % 40 < 20)
|
|
{
|
|
r.x = e->x - (size / 2) - battle.camera.x;
|
|
r.y = e->y - (size / 2) - battle.camera.y;
|
|
r.w = size;
|
|
r.h = size;
|
|
|
|
SDL_SetRenderDrawColor(app.renderer, 255, 255, 255, 255);
|
|
SDL_RenderDrawRect(app.renderer, &r);
|
|
}
|
|
}
|
|
|
|
void activateEntities(char *names)
|
|
{
|
|
Entity *e;
|
|
char *name;
|
|
|
|
name = strtok(names, ";");
|
|
|
|
while (name)
|
|
{
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (strcmp(e->name, name) == 0)
|
|
{
|
|
e->active = 1;
|
|
|
|
if (e->type == ET_CAPITAL_SHIP)
|
|
{
|
|
updateCapitalShipComponentProperties(e, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
name = strtok(NULL, ";");
|
|
}
|
|
|
|
notifyNewArrivals();
|
|
}
|
|
|
|
void activateEntityGroups(char *groupNames)
|
|
{
|
|
Entity *e;
|
|
char *groupName;
|
|
|
|
groupName = strtok(groupNames, ";");
|
|
|
|
while (groupName)
|
|
{
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (strcmp(e->groupName, groupName) == 0)
|
|
{
|
|
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_DEBUG, "Activated %s (%s)", e->name, groupName);
|
|
|
|
e->active = 1;
|
|
|
|
if (e->type == ET_CAPITAL_SHIP)
|
|
{
|
|
updateCapitalShipComponentProperties(e, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
groupName = strtok(NULL, ";");
|
|
}
|
|
|
|
notifyNewArrivals();
|
|
}
|
|
|
|
/*
|
|
* Some craft, such as capital ships, might be performing a long action and won't notice new craft arrive for well over 30 seconds.
|
|
* We'll knock the times down to a max of 1 second, so they can react faster.
|
|
*/
|
|
static void notifyNewArrivals(void)
|
|
{
|
|
Entity *e;
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (e->active && (e->type == ET_FIGHTER || e->type == ET_CAPITAL_SHIP))
|
|
{
|
|
e->aiActionTime = MIN(e->aiActionTime, FPS);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void activateEpicFighters(int side)
|
|
{
|
|
Entity *e;
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (!e->active && e->type == ET_FIGHTER && !(e->flags & EF_NO_EPIC) && ((side == SIDE_ALLIES && e->side == SIDE_ALLIES) || (side != SIDE_ALLIES && e->side != SIDE_ALLIES)))
|
|
{
|
|
e->active = 1;
|
|
|
|
/* don't spring into existence in front of the player */
|
|
if (isOnBattleScreen(e->x, e->y, e->w, e->h))
|
|
{
|
|
e->x = player->x;
|
|
e->y = player->y;
|
|
|
|
e->x += (rand() % 2) ? -SCREEN_WIDTH : SCREEN_WIDTH;
|
|
e->y += (rand() % 2) ? -SCREEN_HEIGHT : SCREEN_HEIGHT;
|
|
}
|
|
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void countNumEnemies(void)
|
|
{
|
|
Entity *e;
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (e->side != SIDE_ALLIES && (e->type == ET_FIGHTER || e->type == ET_CAPITAL_SHIP) && (!(e->flags & EF_NO_THREAT)))
|
|
{
|
|
battle.numInitialEnemies++;
|
|
}
|
|
}
|
|
|
|
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_DEBUG, "battle.numInitialEnemies=%d", battle.numInitialEnemies);
|
|
}
|
|
|
|
void addAllToQuadtree(void)
|
|
{
|
|
Entity *e;
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (e->active)
|
|
{
|
|
addToQuadtree(e, &battle.quadtree);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int drawComparator(const void *a, const void *b)
|
|
{
|
|
Entity *e1 = *((Entity**)a);
|
|
Entity *e2 = *((Entity**)b);
|
|
|
|
return e2->type - e1->type;
|
|
}
|
|
|
|
void killEntity(char *name)
|
|
{
|
|
Entity *e;
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (strcmp(e->name, name) == 0)
|
|
{
|
|
e->health = 0;
|
|
e->deathType = DT_INSTANT;
|
|
|
|
/* prevent objectives and conditions from firing */
|
|
strcpy(e->name, "");
|
|
strcpy(e->groupName, "");
|
|
}
|
|
}
|
|
}
|
|
|
|
void updateEntitySide(char *sideStr, char *name)
|
|
{
|
|
Entity *e;
|
|
int side;
|
|
|
|
side = lookup(sideStr);
|
|
|
|
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (strcmp(e->name, name) == 0)
|
|
{
|
|
e->side = side;
|
|
|
|
if (e->side != player->side)
|
|
{
|
|
e->flags |= EF_MISSION_TARGET;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void awardPandoranCraftTrophy(void)
|
|
{
|
|
Entity *e;
|
|
|
|
for (e = deadHead.next ; e != NULL ; e = e->next)
|
|
{
|
|
if (e->killedBy == player && e->side == SIDE_PANDORAN)
|
|
{
|
|
awardTrophy("PANDORAN");
|
|
}
|
|
}
|
|
}
|
|
|
|
void destroyEntities(void)
|
|
{
|
|
Entity *e;
|
|
|
|
while (deadHead.next)
|
|
{
|
|
e = deadHead.next;
|
|
deadHead.next = e->next;
|
|
free(e);
|
|
}
|
|
|
|
deadTail = &deadHead;
|
|
}
|