tbftss/src/galaxy/mission.c

552 lines
12 KiB
C
Raw Normal View History

2015-10-20 13:51:49 +02:00
/*
Copyright (C) 2015 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 "mission.h"
static void loadObjectives(cJSON *node);
2015-10-26 09:10:13 +01:00
static void loadTriggers(cJSON *node);
2015-10-20 13:51:49 +02:00
static void loadPlayer(cJSON *node);
static void loadFighters(cJSON *node);
static void loadFighterGroups(cJSON *node);
2015-10-26 18:27:43 +01:00
static void loadEntities(cJSON *node);
static void loadEntityGroups(cJSON *node);
2015-10-20 13:51:49 +02:00
static unsigned long hashcode(const char *str);
static char **toFighterTypeArray(char *types, int *numTypes);
static void loadEpicData(cJSON *node);
2015-10-20 13:51:49 +02:00
void loadMission(char *filename)
{
cJSON *root;
char *text, music[MAX_NAME_LENGTH];
startSectionTransition();
stopMusic();
text = readFile(filename);
srand(hashcode(filename));
root = cJSON_Parse(text);
battle.background = getTexture(cJSON_GetObjectItem(root, "background")->valuestring);
battle.planetTexture = getTexture(cJSON_GetObjectItem(root, "planet")->valuestring);
battle.planet.x = rand() % SCREEN_WIDTH - rand() % SCREEN_WIDTH;
battle.planet.y = rand() % SCREEN_HEIGHT - rand() % SCREEN_HEIGHT;
loadObjectives(cJSON_GetObjectItem(root, "objectives"));
2015-10-26 09:10:13 +01:00
loadTriggers(cJSON_GetObjectItem(root, "triggers"));
2015-10-20 13:51:49 +02:00
loadPlayer(cJSON_GetObjectItem(root, "player"));
loadFighters(cJSON_GetObjectItem(root, "fighters"));
loadFighterGroups(cJSON_GetObjectItem(root, "fighterGroups"));
2015-10-26 18:27:43 +01:00
loadEntities(cJSON_GetObjectItem(root, "entities"));
loadEntityGroups(cJSON_GetObjectItem(root, "entityGroups"));
2015-10-20 13:51:49 +02:00
STRNCPY(music, cJSON_GetObjectItem(root, "music")->valuestring, MAX_NAME_LENGTH);
if (cJSON_GetObjectItem(root, "epic"))
{
loadEpicData(cJSON_GetObjectItem(root, "epic"));
}
2015-10-20 13:51:49 +02:00
cJSON_Delete(root);
free(text);
srand(time(NULL));
endSectionTransition();
/* only increment num missions started if there are objectives (Free Flight excluded, for example) */
if (battle.objectiveHead.next)
{
game.stats[STAT_MISSIONS_STARTED]++;
}
else
2015-10-20 13:51:49 +02:00
{
battle.status = MS_IN_PROGRESS;
}
activateNextWaypoint();
initPlayer();
2015-10-20 13:51:49 +02:00
playMusic(music);
}
2015-10-26 09:10:13 +01:00
void completeMission(void)
{
if (battle.status == MS_IN_PROGRESS)
{
battle.status = MS_COMPLETE;
battle.missionFinishedTimer = FPS;
selectWidget("continue", "battleWon");
game.stats[STAT_MISSIONS_COMPLETED]++;
fleeAllEnemies();
}
2015-10-26 09:10:13 +01:00
}
void failMission(void)
{
if (battle.status == MS_IN_PROGRESS)
{
battle.status = MS_FAILED;
battle.missionFinishedTimer = FPS;
selectWidget("retry", "battleLost");
failIncompleteObjectives();
}
2015-10-26 09:10:13 +01:00
}
2015-10-20 13:51:49 +02:00
static void loadObjectives(cJSON *node)
{
Objective *o;
if (node)
{
node = node->child;
while (node)
{
o = malloc(sizeof(Objective));
memset(o, 0, sizeof(Objective));
2015-10-26 09:10:13 +01:00
battle.objectiveTail->next = o;
battle.objectiveTail = o;
2015-10-20 13:51:49 +02:00
2015-10-29 12:08:47 +01:00
o->active = 1;
2015-10-20 13:51:49 +02:00
STRNCPY(o->description, cJSON_GetObjectItem(node, "description")->valuestring, MAX_DESCRIPTION_LENGTH);
STRNCPY(o->targetName, cJSON_GetObjectItem(node, "targetName")->valuestring, MAX_NAME_LENGTH);
o->targetValue = cJSON_GetObjectItem(node, "targetValue")->valueint;
o->targetType = lookup(cJSON_GetObjectItem(node, "targetType")->valuestring);
2015-10-29 12:08:47 +01:00
if (cJSON_GetObjectItem(node, "active"))
{
o->active = cJSON_GetObjectItem(node, "active")->valueint;
}
if (cJSON_GetObjectItem(node, "isCondition"))
{
o->isCondition = cJSON_GetObjectItem(node, "isCondition")->valueint;
}
if (cJSON_GetObjectItem(node, "isOptional"))
{
o->isOptional = cJSON_GetObjectItem(node, "isOptional")->valueint;
}
2015-10-20 13:51:49 +02:00
2015-10-26 09:10:13 +01:00
node = node->next;
}
}
}
static void loadTriggers(cJSON *node)
{
Trigger *t;
if (node)
{
node = node->child;
while (node)
{
t = malloc(sizeof(Trigger));
memset(t, 0, sizeof(Trigger));
battle.triggerTail->next = t;
battle.triggerTail = t;
t->type = lookup(cJSON_GetObjectItem(node, "type")->valuestring);
STRNCPY(t->targetName, cJSON_GetObjectItem(node, "targetName")->valuestring, MAX_NAME_LENGTH);
t->targetValue = cJSON_GetObjectItem(node, "targetValue")->valueint;
t->action = lookup(cJSON_GetObjectItem(node, "action")->valuestring);
2015-10-20 13:51:49 +02:00
2015-10-29 12:08:47 +01:00
if (cJSON_GetObjectItem(node, "actionValue"))
{
STRNCPY(t->actionValue, cJSON_GetObjectItem(node, "actionValue")->valuestring, MAX_NAME_LENGTH);
}
2015-10-20 13:51:49 +02:00
node = node->next;
}
}
}
static 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 = (GRID_SIZE * GRID_CELL_WIDTH) / 2;
player->y = (GRID_SIZE * GRID_CELL_HEIGHT) / 2;
2015-10-20 13:51:49 +02:00
}
static void loadFighters(cJSON *node)
{
Entity *f;
2015-10-20 13:51:49 +02:00
char *type;
int side;
float x, y;
2015-10-20 13:51:49 +02:00
if (node)
{
node = node->child;
while (node)
{
type = cJSON_GetObjectItem(node, "type")->valuestring;
side = lookup(cJSON_GetObjectItem(node, "side")->valuestring);
x = cJSON_GetObjectItem(node, "x")->valuedouble * GRID_CELL_WIDTH;
y = cJSON_GetObjectItem(node, "y")->valuedouble * GRID_CELL_HEIGHT;
2015-10-20 13:51:49 +02:00
f = spawnFighter(type, x, y, side);
STRNCPY(f->name, cJSON_GetObjectItem(node, "name")->valuestring, MAX_NAME_LENGTH);
if (cJSON_GetObjectItem(node, "flags"))
{
f->flags = flagsToLong(cJSON_GetObjectItem(node, "flags")->valuestring);
}
2015-10-29 12:08:47 +01:00
if (cJSON_GetObjectItem(node, "active"))
{
f->active = cJSON_GetObjectItem(node, "active")->valueint;
}
2015-10-20 13:51:49 +02:00
node = node->next;
}
}
}
static void loadFighterGroups(cJSON *node)
{
Entity *f;
char **types, *name, *groupName, *type;
int side, scatter, number, active;
int i, numTypes;
2015-11-15 12:31:43 +01:00
long flags;
float x, y;
2015-10-20 13:51:49 +02:00
scatter = 1;
2015-10-29 12:08:47 +01:00
active = 1;
2015-10-20 13:51:49 +02:00
if (node)
{
node = node->child;
while (node)
{
groupName = NULL;
2015-11-15 12:31:43 +01:00
flags = -1;
types = toFighterTypeArray(cJSON_GetObjectItem(node, "types")->valuestring, &numTypes);
2015-10-20 13:51:49 +02:00
side = lookup(cJSON_GetObjectItem(node, "side")->valuestring);
number = cJSON_GetObjectItem(node, "number")->valueint;
name = cJSON_GetObjectItem(node, "name")->valuestring;
x = cJSON_GetObjectItem(node, "x")->valuedouble * GRID_CELL_WIDTH;
y = cJSON_GetObjectItem(node, "y")->valuedouble * GRID_CELL_HEIGHT;
2015-10-20 13:51:49 +02:00
if (cJSON_GetObjectItem(node, "groupName"))
{
groupName = cJSON_GetObjectItem(node, "groupName")->valuestring;
}
if (cJSON_GetObjectItem(node, "scatter"))
{
scatter = cJSON_GetObjectItem(node, "scatter")->valueint;
}
2015-10-29 12:08:47 +01:00
if (cJSON_GetObjectItem(node, "active"))
{
active = cJSON_GetObjectItem(node, "active")->valueint;
}
2015-11-15 12:31:43 +01:00
if (cJSON_GetObjectItem(node, "flags"))
{
flags = flagsToLong(cJSON_GetObjectItem(node, "flags")->valuestring);
}
2015-10-20 13:51:49 +02:00
for (i = 0 ; i < number ; i++)
{
type = types[rand() % numTypes];
2015-10-20 13:51:49 +02:00
f = spawnFighter(type, x, y, side);
f->x += (rand() % scatter) - (rand() % scatter);
f->y += (rand() % scatter) - (rand() % scatter);
2015-10-29 12:08:47 +01:00
f->active = active;
2015-11-15 12:31:43 +01:00
if (flags != -1)
{
f->flags = flags;
}
2015-10-20 13:51:49 +02:00
STRNCPY(f->name, name, MAX_NAME_LENGTH);
if (groupName)
{
STRNCPY(f->groupName, groupName, MAX_NAME_LENGTH);
}
2015-10-20 13:51:49 +02:00
}
node = node->next;
free(types);
2015-10-20 13:51:49 +02:00
}
}
}
2015-10-26 18:27:43 +01:00
static void loadEntities(cJSON *node)
{
Entity *e;
int type;
if (node)
{
node = node->child;
while (node)
{
type = lookup(cJSON_GetObjectItem(node, "type")->valuestring);
switch (type)
{
case ET_WAYPOINT:
e = spawnWaypoint();
break;
2015-11-09 23:46:57 +01:00
case ET_EXTRACTION_POINT:
e = spawnExtractionPoint();
break;
2015-10-26 18:27:43 +01:00
}
if (cJSON_GetObjectItem(node, "name"))
{
STRNCPY(e->name, cJSON_GetObjectItem(node, "name")->valuestring, MAX_NAME_LENGTH);
}
e->x = cJSON_GetObjectItem(node, "x")->valuedouble * GRID_CELL_WIDTH;
e->y = cJSON_GetObjectItem(node, "y")->valuedouble * GRID_CELL_HEIGHT;
2015-10-26 18:27:43 +01:00
if (cJSON_GetObjectItem(node, "flags"))
{
e->flags = flagsToLong(cJSON_GetObjectItem(node, "flags")->valuestring);
}
SDL_QueryTexture(e->texture, NULL, NULL, &e->w, &e->h);
2015-10-26 18:27:43 +01:00
node = node->next;
}
}
}
static void loadEntityGroups(cJSON *node)
{
Entity *e;
char *name, *groupName;
int i, type, scatter, number;
float x, y;
scatter = 1;
if (node)
{
node = node->child;
while (node)
{
type = lookup(cJSON_GetObjectItem(node, "type")->valuestring);
number = cJSON_GetObjectItem(node, "number")->valueint;
x = cJSON_GetObjectItem(node, "x")->valuedouble * GRID_CELL_WIDTH;
y = cJSON_GetObjectItem(node, "y")->valuedouble * GRID_CELL_HEIGHT;
2015-10-29 12:08:47 +01:00
name = NULL;
groupName = NULL;
if (cJSON_GetObjectItem(node, "name"))
{
name = cJSON_GetObjectItem(node, "name")->valuestring;
}
if (cJSON_GetObjectItem(node, "groupName"))
{
groupName = cJSON_GetObjectItem(node, "groupName")->valuestring;
}
if (cJSON_GetObjectItem(node, "scatter"))
{
scatter = cJSON_GetObjectItem(node, "scatter")->valueint;
}
for (i = 0 ; i < number ; i++)
{
switch (type)
{
case ET_WAYPOINT:
e = spawnWaypoint();
break;
}
if (name)
{
STRNCPY(e->name, name, MAX_NAME_LENGTH);
}
if (groupName)
{
STRNCPY(e->groupName, groupName, MAX_NAME_LENGTH);
}
e->id = battle.entId++;
e->x = x;
e->y = y;
e->x += (rand() % scatter) - (rand() % scatter);
e->y += (rand() % scatter) - (rand() % scatter);
SDL_QueryTexture(e->texture, NULL, NULL, &e->w, &e->h);
}
node = node->next;
}
}
}
static char **toFighterTypeArray(char *types, int *numTypes)
{
int i;
char **typeArray, *type;
*numTypes = 1;
for (i = 0 ; i < strlen(types) ; i++)
{
if (types[i] == ';')
{
*numTypes = *numTypes + 1;
}
}
typeArray = malloc(*numTypes * sizeof(char*));
i = 0;
type = strtok(types, ";");
while (type)
{
typeArray[i] = malloc(strlen(type) + 1);
strcpy(typeArray[i], type);
type = strtok(NULL, ";");
i++;
}
return typeArray;
}
static void loadEpicData(cJSON *node)
{
Entity *e;
int numFighters[SIDE_MAX];
memset(numFighters, 0, sizeof(int) * SIDE_MAX);
battle.epic = 1;
battle.epicFighterLimit = cJSON_GetObjectItem(node, "fighterLimit")->valueint;
for (e = battle.entityHead.next ; e != NULL ; e = e->next)
{
if (e->active && e->type == ET_FIGHTER && numFighters[e->side]++ >= battle.epicFighterLimit)
{
e->active = 0;
}
}
}
Mission *getMission(char *filename)
2015-10-20 13:51:49 +02:00
{
StarSystem *starSystem;
2015-10-20 13:51:49 +02:00
Mission *mission;
for (starSystem = game.starSystemHead.next ; starSystem != NULL ; starSystem = starSystem->next)
2015-10-20 13:51:49 +02:00
{
for (mission = starSystem->missionHead.next ; mission != NULL ; mission = mission->next)
2015-10-20 13:51:49 +02:00
{
if (strcmp(mission->filename, filename) == 0)
{
return mission;
}
2015-10-20 13:51:49 +02:00
}
}
return NULL;
}
int isMissionAvailable(Mission *mission, Mission *prev)
{
2015-11-14 14:03:40 +01:00
#if !DEBUG
Mission *reqMission;
if (mission->requires)
{
if (strcmp(mission->requires, "PREVIOUS") == 0)
{
return prev->completed;
}
else
{
reqMission = getMission(mission->requires);
if (reqMission != NULL)
{
return reqMission->completed;
}
}
}
#endif
return 1;
}
2015-10-20 13:51:49 +02:00
static unsigned long hashcode(const char *str)
{
unsigned long hash = 5381;
int c;
c = *str;
while (c)
{
hash = ((hash << 5) + hash) + c;
c = *str++;
}
return abs(hash);
}