tbftss/src/battle/player.c

638 lines
12 KiB
C
Raw Normal View History

2015-10-20 13:51:49 +02:00
/*
2016-02-21 16:50:27 +01:00
Copyright (C) 2015-2016 Parallel Realities
2015-10-20 13:51:49 +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 "player.h"
static void selectTarget(void);
static void switchGuns(void);
static void cycleRadarZoom(void);
static void selectMissionTarget(void);
static void selectNewPlayer(int dir);
static void initPlayerSelect(void);
static void activateBoost(void);
2015-11-16 00:23:03 +01:00
static void deactivateBoost(void);
static void activateECM(void);
2015-11-23 15:52:11 +01:00
static void handleKeyboard(void);
static void faceMouse(void);
static void handleMouse(void);
2015-12-14 12:41:43 +01:00
static void preFireMissile(void);
2016-03-02 23:19:26 +01:00
static void applyRestrictions(void);
static int isPriorityMissionTarget(Entity *e, int dist, int closest);
static int selectedPlayerIndex;
static int availableGuns[BT_MAX];
static Entity *availablePlayerUnits[MAX_SELECTABLE_PLAYERS];
void initPlayer(void)
{
int i, n;
2016-03-03 19:03:07 +01:00
memset(&availableGuns, 0, sizeof(int) * BT_MAX);
2016-03-03 19:03:07 +01:00
battle.numPlayerGuns = 0;
2016-03-03 19:03:07 +01:00
player->selectedGunType = -1;
2016-03-03 19:03:07 +01:00
2015-12-14 15:05:02 +01:00
if (!player->combinedGuns)
{
2015-12-14 15:05:02 +01:00
for (i = 0 ; i < MAX_FIGHTER_GUNS ; i++)
{
2015-12-14 15:05:02 +01:00
n = player->guns[i].type;
2016-03-03 19:03:07 +01:00
2015-12-14 15:05:02 +01:00
if (n)
{
if (!availableGuns[n])
{
battle.numPlayerGuns++;
}
2016-03-03 19:03:07 +01:00
2015-12-14 15:05:02 +01:00
availableGuns[n] = 1;
2016-03-03 19:03:07 +01:00
2015-12-14 15:05:02 +01:00
if (player->selectedGunType == -1)
{
player->selectedGunType = n;
}
}
}
}
2015-12-14 15:05:02 +01:00
else
{
player->selectedGunType = 0;
}
2016-03-03 19:03:07 +01:00
STRNCPY(player->name, "Player", MAX_NAME_LENGTH);
2016-03-03 19:03:07 +01:00
player->action = NULL;
2016-03-03 19:03:07 +01:00
2015-11-15 18:12:04 +01:00
battle.boostTimer = BOOST_RECHARGE_TIME;
battle.ecmTimer = ECM_RECHARGE_TIME;
2016-03-03 19:03:07 +01:00
2015-11-17 08:23:50 +01:00
game.stats[STAT_EPIC_KILL_STREAK] = MAX(game.stats[STAT_EPIC_KILL_STREAK], battle.stats[STAT_EPIC_KILL_STREAK]);
2016-03-03 19:03:07 +01:00
2015-11-17 08:23:50 +01:00
battle.stats[STAT_EPIC_KILL_STREAK] = 0;
}
2015-10-20 13:51:49 +02:00
void doPlayer(void)
{
2015-11-15 18:12:04 +01:00
battle.boostTimer = MIN(battle.boostTimer + 1, BOOST_RECHARGE_TIME);
battle.ecmTimer = MIN(battle.ecmTimer + 1, ECM_RECHARGE_TIME);
2016-03-03 19:03:07 +01:00
2015-10-20 13:51:49 +02:00
if (player != NULL)
{
self = player;
2016-03-02 23:19:26 +01:00
if (game.currentMission->challengeData.isChallenge)
{
applyRestrictions();
}
2015-10-20 13:51:49 +02:00
if (player->alive == ALIVE_ALIVE)
{
2015-11-23 15:52:11 +01:00
handleKeyboard();
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
handleMouse();
2016-03-03 19:03:07 +01:00
2015-12-10 11:16:44 +01:00
if (!player->target || player->target->health <= 0 || player->target->systemPower <= 0)
{
selectTarget();
}
2015-11-23 15:52:11 +01:00
}
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
player->angle = ((int)player->angle) % 360;
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
if (player->health <= 0 && battle.status == MS_IN_PROGRESS)
{
battle.stats[STAT_PLAYER_KILLED]++;
/* the player is known as "Player", so we need to check the craft they were assigned to */
if (strcmp(game.currentMission->craft, "ATAF") == 0)
{
awardTrophy("ATAF_DESTROYED");
}
if (game.currentMission->challengeData.isChallenge)
{
if (!game.currentMission->challengeData.allowPlayerDeath)
{
failMission();
}
}
else if (!battle.isEpic)
2015-10-20 13:51:49 +02:00
{
2015-11-23 15:52:11 +01:00
failMission();
2015-10-20 13:51:49 +02:00
}
2015-11-23 15:52:11 +01:00
else if (player->health == -FPS)
2015-10-20 13:51:49 +02:00
{
2015-11-23 15:52:11 +01:00
initPlayerSelect();
2015-10-20 13:51:49 +02:00
}
2015-11-23 15:52:11 +01:00
}
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
if (battle.status == MS_IN_PROGRESS)
{
selectMissionTarget();
}
2016-03-03 19:03:07 +01:00
2015-12-18 11:12:37 +01:00
if (dev.playerUnlimitedMissiles)
{
player->missiles = 999;
}
2015-11-23 15:52:11 +01:00
}
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
if (battle.boostTimer == (int)BOOST_FINISHED_TIME)
{
deactivateBoost();
}
}
2016-03-02 23:19:26 +01:00
static void applyRestrictions(void)
{
if (game.currentMission->challengeData.noMissiles)
{
player->missiles = 0;
}
if (game.currentMission->challengeData.noBoost)
{
battle.boostTimer = 0;
}
if (game.currentMission->challengeData.noECM)
{
battle.ecmTimer = 0;
}
if (game.currentMission->challengeData.noGuns)
{
player->reload = 1;
}
}
2015-11-23 15:52:11 +01:00
static void handleKeyboard(void)
{
if (battle.status == MS_IN_PROGRESS)
{
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_BOOST))
2015-11-23 15:52:11 +01:00
{
if (battle.boostTimer == BOOST_RECHARGE_TIME)
{
playSound(SND_BOOST);
2016-03-03 19:03:07 +01:00
activateBoost();
}
else
{
playSound(SND_GUI_DENIED);
}
2016-03-03 19:03:07 +01:00
clearControl(CONTROL_BOOST);
2015-11-23 15:52:11 +01:00
}
2016-03-03 19:03:07 +01:00
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_TARGET))
2015-11-23 15:52:11 +01:00
{
2015-11-26 10:53:49 +01:00
selectTarget();
2016-03-03 19:03:07 +01:00
clearControl(CONTROL_TARGET);
2015-11-26 10:53:49 +01:00
}
2016-03-03 19:03:07 +01:00
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_ECM))
2015-11-26 10:53:49 +01:00
{
if (battle.ecmTimer == ECM_RECHARGE_TIME)
{
playSound(SND_ECM);
2016-03-03 19:03:07 +01:00
activateECM();
}
else
{
playSound(SND_GUI_DENIED);
}
2016-03-03 19:03:07 +01:00
clearControl(CONTROL_ECM);
2015-11-23 15:52:11 +01:00
}
2016-03-03 19:03:07 +01:00
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_BRAKE))
2015-11-23 15:52:11 +01:00
{
2015-11-26 10:53:49 +01:00
applyFighterBrakes();
2015-11-23 15:52:11 +01:00
}
2016-03-03 19:03:07 +01:00
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_GUNS))
2015-12-14 12:41:43 +01:00
{
switchGuns();
2016-03-03 19:03:07 +01:00
clearControl(CONTROL_GUNS);
2015-12-14 12:41:43 +01:00
}
2016-03-03 19:03:07 +01:00
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_RADAR))
2015-12-14 12:41:43 +01:00
{
cycleRadarZoom();
2016-03-03 19:03:07 +01:00
clearControl(CONTROL_RADAR);
2015-12-14 12:41:43 +01:00
}
2016-03-03 19:03:07 +01:00
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_MISSILE))
2015-12-14 12:41:43 +01:00
{
preFireMissile();
2016-03-03 19:03:07 +01:00
clearControl(CONTROL_MISSILE);
2015-12-14 12:41:43 +01:00
}
2015-11-23 15:52:11 +01:00
}
else
{
player->dx *= 0.99;
player->dy *= 0.99;
}
2015-11-23 15:52:11 +01:00
}
static void handleMouse(void)
{
faceMouse();
if (battle.status == MS_IN_PROGRESS)
{
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_FIRE) && !player->reload && player->guns[0].type)
2015-11-23 15:52:11 +01:00
{
if (player->selectedGunType != BT_ROCKET)
{
fireGuns(player);
}
else
{
fireRocket(player);
}
2015-11-23 15:52:11 +01:00
}
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_ACCELERATE))
2015-11-23 15:52:11 +01:00
{
2016-03-02 23:19:26 +01:00
if (battle.boostTimer > BOOST_FINISHED_TIME || game.currentMission->challengeData.noBoost)
2015-10-20 13:51:49 +02:00
{
2015-11-23 15:52:11 +01:00
applyFighterThrust();
}
2015-10-20 13:51:49 +02:00
}
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_MISSILE))
2015-10-20 13:51:49 +02:00
{
2015-12-14 12:41:43 +01:00
preFireMissile();
2015-11-23 15:52:11 +01:00
app.mouse.button[SDL_BUTTON_MIDDLE] = 0;
2015-10-20 13:51:49 +02:00
}
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_GUNS))
{
2016-02-21 08:48:22 +01:00
switchGuns();
app.mouse.button[SDL_BUTTON_X1] = 0;
}
2016-03-05 14:21:17 +01:00
if (isControl(CONTROL_RADAR))
{
2016-02-21 08:48:22 +01:00
cycleRadarZoom();
app.mouse.button[SDL_BUTTON_X2] = 0;
}
2015-10-20 13:51:49 +02:00
}
2015-11-23 15:52:11 +01:00
}
static void faceMouse(void)
{
int dir;
int x, y, wantedAngle;
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
x = player->x - battle.camera.x;
y = player->y - battle.camera.y;
wantedAngle = getAngle(x, y, app.mouse.x, app.mouse.y);
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
wantedAngle %= 360;
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
if (fabs(wantedAngle - player->angle) > 2)
2015-11-16 00:23:03 +01:00
{
2015-11-23 15:52:11 +01:00
dir = ((int)(wantedAngle - player->angle + 360)) % 360 > 180 ? -1 : 1;
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
player->angle += dir * 4;
2016-03-03 19:03:07 +01:00
2015-11-23 15:52:11 +01:00
player->angle = mod(player->angle, 360);
2015-11-16 00:23:03 +01:00
}
}
2015-12-14 12:41:43 +01:00
static void preFireMissile(void)
{
if (player->missiles && player->target)
{
if (getDistance(player->x, player->y, player->target->x, player->target->y) <= SCREEN_WIDTH)
{
fireMissile(player);
}
else
{
2016-02-28 14:45:17 +01:00
addHudMessage(colors.white, _("Target not in range"));
2015-12-14 12:41:43 +01:00
}
}
}
void initPlayerSelect(void)
{
Entity *e;
2016-03-03 19:03:07 +01:00
memset(&availablePlayerUnits, 0, sizeof(Entity*) * MAX_SELECTABLE_PLAYERS);
2016-03-03 19:03:07 +01:00
selectedPlayerIndex = 0;
2016-03-03 19:03:07 +01:00
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
{
2015-11-19 13:44:56 +01:00
if (e->active && e->type == ET_FIGHTER && e->health > 0 && e->side == SIDE_ALLIES && selectedPlayerIndex < MAX_SELECTABLE_PLAYERS)
{
availablePlayerUnits[selectedPlayerIndex++] = e;
}
}
2016-03-03 19:03:07 +01:00
if (selectedPlayerIndex > 0)
{
battle.playerSelect = 1;
2016-03-03 19:03:07 +01:00
selectedPlayerIndex = 0;
memset(&app.keyboard, 0, sizeof(int) * MAX_KEYBOARD_KEYS);
}
else
{
2016-02-28 14:02:57 +01:00
battle.isEpic = 0;
2016-03-03 19:03:07 +01:00
2015-11-14 09:41:31 +01:00
failMission();
}
}
void doPlayerSelect(void)
{
if (isControl(CONTROL_PREV_FIGHTER))
{
selectNewPlayer(-1);
clearControl(CONTROL_PREV_FIGHTER);
}
if (isControl(CONTROL_NEXT_FIGHTER))
{
selectNewPlayer(1);
clearControl(CONTROL_NEXT_FIGHTER);
}
if (player->health > 0 && isAcceptControl())
{
battle.playerSelect = 0;
initPlayer();
resetAcceptControls();
}
}
static void selectNewPlayer(int dir)
{
do
{
selectedPlayerIndex += dir;
2016-03-03 19:03:07 +01:00
selectedPlayerIndex = mod(selectedPlayerIndex, MAX_SELECTABLE_PLAYERS);
2016-03-03 19:03:07 +01:00
player = availablePlayerUnits[selectedPlayerIndex];
}
while (player == NULL);
2016-03-03 19:03:07 +01:00
battle.camera.x = player->x - (SCREEN_WIDTH / 2);
battle.camera.y = player->y - (SCREEN_HEIGHT / 2);
2015-10-20 13:51:49 +02:00
}
static void activateBoost(void)
{
self->dx += sin(TO_RAIDANS(self->angle)) * 10;
self->dy += -cos(TO_RAIDANS(self->angle)) * 10;
self->thrust = sqrt((self->dx * self->dx) + (self->dy * self->dy));
2016-03-03 19:03:07 +01:00
2015-11-15 18:12:04 +01:00
battle.boostTimer = 0;
2016-03-03 19:03:07 +01:00
2015-11-17 08:23:50 +01:00
battle.stats[STAT_BOOST]++;
}
2015-11-16 00:23:03 +01:00
static void deactivateBoost(void)
{
float v, thrust;
2016-03-03 19:03:07 +01:00
2015-11-16 00:23:03 +01:00
thrust = -1;
2016-03-03 19:03:07 +01:00
2015-11-16 00:23:03 +01:00
while (thrust != self->thrust)
{
thrust = self->thrust;
2016-03-03 19:03:07 +01:00
2015-11-16 00:23:03 +01:00
v = (self->speed / sqrt(self->thrust));
self->dx = v * self->dx;
self->dy = v * self->dy;
self->thrust = sqrt((self->dx * self->dx) + (self->dy * self->dy));
}
}
static void activateECM(void)
{
2015-11-15 18:12:04 +01:00
battle.ecmTimer = 0;
2016-03-03 19:03:07 +01:00
2015-11-29 09:34:25 +01:00
addECMEffect(player);
2016-03-03 19:03:07 +01:00
2015-11-17 08:23:50 +01:00
battle.stats[STAT_ECM]++;
}
static void switchGuns(void)
{
int i;
2016-03-03 19:03:07 +01:00
i = player->selectedGunType;
2016-03-03 19:03:07 +01:00
2015-12-14 15:05:02 +01:00
if (!player->combinedGuns)
{
2015-12-14 15:05:02 +01:00
do
{
i = (i + 1) % BT_MAX;
}
while (!availableGuns[i]);
}
2016-03-03 19:03:07 +01:00
2016-02-20 16:04:15 +01:00
if (player->selectedGunType != i)
{
playSound(SND_SELECT_WEAPON);
player->selectedGunType = i;
}
}
2015-10-20 13:51:49 +02:00
static void selectTarget(void)
{
2015-11-01 11:46:24 +01:00
unsigned int closest = MAX_TARGET_RANGE;
unsigned int dist = MAX_TARGET_RANGE;
2015-11-23 11:14:28 +01:00
int i, total;
Entity *e, *near;
Entity *targets[MAX_SELECTABLE_TARGETS];
2016-03-03 19:03:07 +01:00
2015-11-23 11:14:28 +01:00
i = 0;
2015-11-23 15:52:11 +01:00
near = NULL;
2015-11-23 11:14:28 +01:00
memset(targets, 0, sizeof(Entity*) * MAX_SELECTABLE_TARGETS);
2015-10-20 13:51:49 +02:00
if (player->target && (!player->target->health || !player->target->systemPower))
{
player->target = NULL;
}
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
2015-10-20 13:51:49 +02:00
{
2016-04-01 11:49:41 +02:00
if (e->active && e != player && (e->flags & EF_TAKES_DAMAGE) && (!(e->flags & EF_NO_TARGET)) && e->side != player->side && e->alive == ALIVE_ALIVE && e->systemPower > 0 && i < MAX_SELECTABLE_TARGETS)
2015-10-20 13:51:49 +02:00
{
dist = getDistance(self->x, self->y, e->x, e->y);
2015-10-20 13:51:49 +02:00
if (dist < closest)
{
2015-11-23 11:14:28 +01:00
near = e;
2015-10-20 13:51:49 +02:00
closest = dist;
}
2016-03-03 19:03:07 +01:00
2015-11-23 11:14:28 +01:00
if (collision(0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, e->x - battle.camera.x - (e->w / 2), e->y - battle.camera.y - (e->h / 2), e->w, e->h))
{
targets[i++] = e;
}
else if (e == player->target)
{
player->target = NULL;
}
2015-10-20 13:51:49 +02:00
}
}
2016-03-03 19:03:07 +01:00
2015-11-23 11:14:28 +01:00
total = i;
2016-03-03 19:03:07 +01:00
2015-11-23 11:14:28 +01:00
for (i = 0 ; i < total ; i++)
{
if (targets[i] == player->target)
{
if (i + 1 < MAX_SELECTABLE_TARGETS && targets[i + 1])
{
player->target = targets[i + 1];
return;
}
else
{
player->target = targets[0];
}
}
}
2016-03-03 19:03:07 +01:00
2015-12-10 11:16:44 +01:00
if (!player->target || !targets[0])
2015-11-23 11:14:28 +01:00
{
player->target = near;
}
2015-10-20 13:51:49 +02:00
}
static void selectMissionTarget(void)
{
2015-11-01 11:46:24 +01:00
unsigned int closest = MAX_TARGET_RANGE;
unsigned int dist = MAX_TARGET_RANGE;
Entity *e;
2016-03-03 19:03:07 +01:00
battle.missionTarget = NULL;
2016-03-03 19:03:07 +01:00
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
{
if (e->active && e->flags & EF_MISSION_TARGET && e->alive == ALIVE_ALIVE)
{
2015-11-22 12:42:04 +01:00
dist = getDistance(self->x, self->y, e->x, e->y);
2016-03-03 19:03:07 +01:00
if (battle.missionTarget == NULL)
{
battle.missionTarget = e;
2015-11-22 12:42:04 +01:00
closest = dist;
}
else if (battle.missionTarget->type == ET_WAYPOINT && e->type != ET_WAYPOINT)
{
battle.missionTarget = e;
}
else if (battle.missionTarget->type != ET_WAYPOINT)
{
if (isPriorityMissionTarget(e, dist, closest))
{
battle.missionTarget = e;
closest = dist;
}
}
else if (battle.missionTarget->type == ET_WAYPOINT && e->type == ET_WAYPOINT && e->id == battle.missionTarget->id)
{
battle.missionTarget = e;
}
}
}
}
static int isPriorityMissionTarget(Entity *e, int dist, int closest)
{
/* battle.missionTarget is secondary, e is not */
if ((battle.missionTarget->flags & EF_SECONDARY_TARGET) > (e->flags & EF_SECONDARY_TARGET))
{
return 1;
}
/* battle.missionTarget is not secondary, e is */
if ((battle.missionTarget->flags & EF_SECONDARY_TARGET) < (e->flags & EF_SECONDARY_TARGET))
{
return 0;
}
/* normal distance check */
return dist < closest;
}
static void cycleRadarZoom(void)
{
2015-11-26 11:54:40 +01:00
battle.radarRange++;
battle.radarRange %= 3;
}
2016-02-20 16:04:15 +01:00
int playerHasGun(int type)
{
return availableGuns[type];
}
void loadPlayer(cJSON *node)
{
char *type;
int side;
type = cJSON_GetObjectItem(node, "type")->valuestring;
side = lookup(cJSON_GetObjectItem(node, "side")->valuestring);
player = spawnFighter(type, 0, 0, side);
player->x = BATTLE_AREA_WIDTH / 2;
player->y = BATTLE_AREA_HEIGHT / 2;
if (cJSON_GetObjectItem(node, "x"))
{
player->x = (cJSON_GetObjectItem(node, "x")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_WIDTH;
player->y = (cJSON_GetObjectItem(node, "y")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_HEIGHT;
}
if (strcmp(type, "Tug") == 0)
{
battle.stats[STAT_TUG]++;
}
if (strcmp(type, "Shuttle") == 0)
{
battle.stats[STAT_SHUTTLE]++;
}
}