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);
|
|
|
|
static void loadPlayer(cJSON *node);
|
|
|
|
static void loadFighters(cJSON *node);
|
2015-12-07 20:19:14 +01:00
|
|
|
static void loadCapitalShips(cJSON *node);
|
2015-10-26 18:27:43 +01:00
|
|
|
static void loadEntities(cJSON *node);
|
2015-11-16 13:31:50 +01:00
|
|
|
static void loadItems(cJSON *node);
|
2015-12-22 13:42:04 +01:00
|
|
|
static void loadLocations(cJSON *node);
|
2015-10-20 13:51:49 +02:00
|
|
|
static unsigned long hashcode(const char *str);
|
2015-12-07 20:19:14 +01:00
|
|
|
static char **toTypeArray(char *types, int *numTypes);
|
2015-11-01 10:25:10 +01:00
|
|
|
static void loadEpicData(cJSON *node);
|
2015-10-20 13:51:49 +02:00
|
|
|
|
|
|
|
void loadMission(char *filename)
|
|
|
|
{
|
|
|
|
cJSON *root;
|
2016-01-28 18:03:23 +01:00
|
|
|
char *text, music[MAX_DESCRIPTION_LENGTH];
|
2015-10-20 13:51:49 +02:00
|
|
|
|
|
|
|
startSectionTransition();
|
|
|
|
|
|
|
|
stopMusic();
|
|
|
|
|
2015-12-03 00:20:19 +01:00
|
|
|
text = readFile(getFileLocation(filename));
|
2015-10-20 13:51:49 +02:00
|
|
|
|
|
|
|
srand(hashcode(filename));
|
|
|
|
|
|
|
|
root = cJSON_Parse(text);
|
|
|
|
|
|
|
|
battle.background = getTexture(cJSON_GetObjectItem(root, "background")->valuestring);
|
|
|
|
battle.planetTexture = getTexture(cJSON_GetObjectItem(root, "planet")->valuestring);
|
2016-02-13 17:59:00 +01:00
|
|
|
battle.planet.x = (SCREEN_WIDTH / 2) - (rand() % SCREEN_WIDTH) + (rand() % SCREEN_WIDTH);
|
|
|
|
battle.planet.y = (SCREEN_HEIGHT / 2) - (rand() % SCREEN_HEIGHT) + (rand() % SCREEN_HEIGHT);
|
2015-10-20 13:51:49 +02:00
|
|
|
|
|
|
|
loadObjectives(cJSON_GetObjectItem(root, "objectives"));
|
|
|
|
|
|
|
|
loadPlayer(cJSON_GetObjectItem(root, "player"));
|
|
|
|
|
|
|
|
loadFighters(cJSON_GetObjectItem(root, "fighters"));
|
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
loadCapitalShips(cJSON_GetObjectItem(root, "capitalShips"));
|
|
|
|
|
2015-10-26 18:27:43 +01:00
|
|
|
loadEntities(cJSON_GetObjectItem(root, "entities"));
|
|
|
|
|
2015-11-16 13:31:50 +01:00
|
|
|
loadItems(cJSON_GetObjectItem(root, "items"));
|
|
|
|
|
2015-12-22 13:42:04 +01:00
|
|
|
loadLocations(cJSON_GetObjectItem(root, "locations"));
|
|
|
|
|
2016-01-28 18:03:23 +01:00
|
|
|
STRNCPY(music, cJSON_GetObjectItem(root, "music")->valuestring, MAX_DESCRIPTION_LENGTH);
|
2015-10-20 13:51:49 +02:00
|
|
|
|
2015-11-01 10:25:10 +01:00
|
|
|
if (cJSON_GetObjectItem(root, "epic"))
|
|
|
|
{
|
|
|
|
loadEpicData(cJSON_GetObjectItem(root, "epic"));
|
|
|
|
}
|
|
|
|
|
2016-01-25 16:09:14 +01:00
|
|
|
if (cJSON_GetObjectItem(root, "manualComplete"))
|
|
|
|
{
|
|
|
|
battle.manualComplete = cJSON_GetObjectItem(root, "manualComplete")->valueint;
|
|
|
|
}
|
|
|
|
|
2016-01-25 15:58:07 +01:00
|
|
|
if (cJSON_GetObjectItem(root, "unwinnable"))
|
|
|
|
{
|
|
|
|
battle.unwinnable = cJSON_GetObjectItem(root, "unwinnable")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-11-28 11:11:20 +01:00
|
|
|
initScript(cJSON_GetObjectItem(root, "script"));
|
|
|
|
|
2015-10-20 13:51:49 +02:00
|
|
|
free(text);
|
|
|
|
|
|
|
|
srand(time(NULL));
|
|
|
|
|
|
|
|
endSectionTransition();
|
|
|
|
|
2015-10-25 01:01:46 +02:00
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
2015-10-28 20:12:58 +01:00
|
|
|
activateNextWaypoint();
|
2015-10-23 08:08:18 +02:00
|
|
|
|
2015-12-31 12:25:19 +01:00
|
|
|
countNumEnemies();
|
|
|
|
|
2015-10-23 08:08:18 +02:00
|
|
|
initPlayer();
|
|
|
|
|
2015-10-20 13:51:49 +02:00
|
|
|
playMusic(music);
|
|
|
|
}
|
|
|
|
|
2015-10-26 09:10:13 +01:00
|
|
|
void completeMission(void)
|
|
|
|
{
|
2015-11-02 20:18:26 +01:00
|
|
|
if (battle.status == MS_IN_PROGRESS)
|
|
|
|
{
|
|
|
|
battle.status = MS_COMPLETE;
|
|
|
|
battle.missionFinishedTimer = FPS;
|
|
|
|
selectWidget("continue", "battleWon");
|
|
|
|
|
|
|
|
game.stats[STAT_MISSIONS_COMPLETED]++;
|
2015-11-15 14:26:46 +01:00
|
|
|
|
2015-11-29 17:50:06 +01:00
|
|
|
completeConditions();
|
|
|
|
|
2015-11-21 18:32:39 +01:00
|
|
|
retreatEnemies();
|
2015-11-21 18:36:42 +01:00
|
|
|
|
|
|
|
player->flags |= EF_IMMORTAL;
|
2015-11-02 20:18:26 +01:00
|
|
|
}
|
2015-10-26 09:10:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void failMission(void)
|
|
|
|
{
|
2015-11-02 20:18:26 +01:00
|
|
|
if (battle.status == MS_IN_PROGRESS)
|
|
|
|
{
|
|
|
|
battle.status = MS_FAILED;
|
|
|
|
battle.missionFinishedTimer = FPS;
|
|
|
|
selectWidget("retry", "battleLost");
|
2015-11-14 09:41:07 +01:00
|
|
|
|
|
|
|
failIncompleteObjectives();
|
2015-11-21 18:36:42 +01:00
|
|
|
|
|
|
|
player->flags |= EF_IMMORTAL;
|
2015-11-02 20:18:26 +01:00
|
|
|
}
|
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;
|
2015-10-22 08:08:03 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2015-10-22 08:08:03 +02:00
|
|
|
if (cJSON_GetObjectItem(node, "isCondition"))
|
|
|
|
{
|
|
|
|
o->isCondition = cJSON_GetObjectItem(node, "isCondition")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-12-22 18:58:18 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "isEliminateAll"))
|
2015-10-22 08:08:03 +02:00
|
|
|
{
|
2015-12-22 18:58:18 +01:00
|
|
|
o->isEliminateAll = cJSON_GetObjectItem(node, "isEliminateAll")->valueint;
|
|
|
|
o->targetValue = 1;
|
2015-10-22 08:08:03 +02:00
|
|
|
}
|
2015-10-20 13:51:49 +02:00
|
|
|
|
2015-10-26 09:10:13 +01:00
|
|
|
node = node->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-20 13:51:49 +02:00
|
|
|
static void loadPlayer(cJSON *node)
|
|
|
|
{
|
|
|
|
char *type;
|
|
|
|
int side;
|
|
|
|
|
|
|
|
type = cJSON_GetObjectItem(node, "type")->valuestring;
|
|
|
|
side = lookup(cJSON_GetObjectItem(node, "side")->valuestring);
|
|
|
|
|
2015-11-02 11:49:10 +01:00
|
|
|
player = spawnFighter(type, 0, 0, side);
|
2016-02-21 08:48:22 +01:00
|
|
|
player->x = BATTLE_AREA_WIDTH / 2;
|
|
|
|
player->y = BATTLE_AREA_HEIGHT / 2;
|
2015-11-17 08:23:50 +01:00
|
|
|
|
2015-11-22 12:41:28 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "x"))
|
|
|
|
{
|
2016-02-21 08:48:22 +01:00
|
|
|
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;
|
2015-11-22 12:41:28 +01:00
|
|
|
}
|
|
|
|
|
2015-11-17 08:23:50 +01:00
|
|
|
if (strcmp(type, "Tug") == 0)
|
|
|
|
{
|
|
|
|
battle.stats[STAT_TUG]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(type, "Shuttle") == 0)
|
|
|
|
{
|
|
|
|
battle.stats[STAT_SHUTTLE]++;
|
|
|
|
}
|
2015-10-20 13:51:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void loadFighters(cJSON *node)
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
Entity *e;
|
2015-11-14 09:41:07 +01:00
|
|
|
char **types, *name, *groupName, *type;
|
2015-11-02 11:49:10 +01:00
|
|
|
int side, scatter, number, active;
|
2015-12-12 19:24:23 +01:00
|
|
|
int i, numTypes, addFlags, addAIFlags;
|
2015-12-02 08:24:12 +01:00
|
|
|
long flags, aiFlags;
|
2015-11-02 11:49:10 +01:00
|
|
|
float x, y;
|
2015-10-20 13:51:49 +02:00
|
|
|
|
|
|
|
if (node)
|
|
|
|
{
|
|
|
|
node = node->child;
|
|
|
|
|
|
|
|
while (node)
|
|
|
|
{
|
2015-11-29 13:56:15 +01:00
|
|
|
name = NULL;
|
2015-11-14 09:41:07 +01:00
|
|
|
groupName = NULL;
|
2015-11-15 12:31:43 +01:00
|
|
|
flags = -1;
|
2015-12-02 08:24:12 +01:00
|
|
|
aiFlags = -1;
|
2015-11-26 18:41:05 +01:00
|
|
|
scatter = 1;
|
|
|
|
active = 1;
|
|
|
|
number = 1;
|
2015-11-14 09:41:07 +01:00
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
types = toTypeArray(cJSON_GetObjectItem(node, "types")->valuestring, &numTypes);
|
2015-10-20 13:51:49 +02:00
|
|
|
side = lookup(cJSON_GetObjectItem(node, "side")->valuestring);
|
2016-02-21 08:48:22 +01:00
|
|
|
x = (cJSON_GetObjectItem(node, "x")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_WIDTH;
|
|
|
|
y = (cJSON_GetObjectItem(node, "y")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_HEIGHT;
|
2015-10-20 13:51:49 +02:00
|
|
|
|
2015-11-29 13:56:15 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "name"))
|
|
|
|
{
|
|
|
|
name = cJSON_GetObjectItem(node, "name")->valuestring;
|
|
|
|
}
|
|
|
|
|
2015-11-14 09:41:07 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "groupName"))
|
|
|
|
{
|
|
|
|
groupName = cJSON_GetObjectItem(node, "groupName")->valuestring;
|
|
|
|
}
|
|
|
|
|
2015-11-26 18:41:05 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "number"))
|
|
|
|
{
|
|
|
|
number = cJSON_GetObjectItem(node, "number")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-10-24 09:51:43 +02:00
|
|
|
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"))
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
flags = flagsToLong(cJSON_GetObjectItem(node, "flags")->valuestring, &addFlags);
|
2015-11-15 12:31:43 +01:00
|
|
|
}
|
|
|
|
|
2015-12-02 08:24:12 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "aiFlags"))
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
aiFlags = flagsToLong(cJSON_GetObjectItem(node, "aiFlags")->valuestring, &addAIFlags);
|
2015-12-02 08:24:12 +01:00
|
|
|
}
|
|
|
|
|
2015-10-20 13:51:49 +02:00
|
|
|
for (i = 0 ; i < number ; i++)
|
|
|
|
{
|
2015-10-25 13:56:24 +01:00
|
|
|
type = types[rand() % numTypes];
|
|
|
|
|
2015-12-14 09:15:41 +01:00
|
|
|
e = spawnFighter(type, x, y, side);
|
2015-10-20 13:51:49 +02:00
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
if (scatter > 1)
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
e->x += (rand() % scatter) - (rand() % scatter);
|
|
|
|
e->y += (rand() % scatter) - (rand() % scatter);
|
2015-12-07 20:19:14 +01:00
|
|
|
}
|
2015-10-24 09:51:43 +02:00
|
|
|
|
2015-12-14 09:15:41 +01:00
|
|
|
e->active = active;
|
2015-10-29 12:08:47 +01:00
|
|
|
|
2015-11-15 12:31:43 +01:00
|
|
|
if (flags != -1)
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
if (addFlags)
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
e->flags |= flags;
|
2015-12-12 19:24:23 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
e->flags = flags;
|
2015-12-12 19:24:23 +01:00
|
|
|
|
2015-12-14 09:15:41 +01:00
|
|
|
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_WARN, "Flags for '%s' (%s) replaced", e->name, e->defName);
|
2015-12-12 19:24:23 +01:00
|
|
|
}
|
2015-11-15 12:31:43 +01:00
|
|
|
}
|
|
|
|
|
2015-12-02 08:24:12 +01:00
|
|
|
if (aiFlags != -1)
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
if (addAIFlags)
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
e->aiFlags |= aiFlags;
|
2015-12-12 19:24:23 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
e->aiFlags = aiFlags;
|
2015-12-12 19:24:23 +01:00
|
|
|
|
2015-12-14 09:15:41 +01:00
|
|
|
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_WARN, "AI Flags for '%s' (%s) replaced", e->name, e->defName);
|
2015-12-12 19:24:23 +01:00
|
|
|
}
|
2015-12-02 08:24:12 +01:00
|
|
|
}
|
|
|
|
|
2015-11-29 13:56:15 +01:00
|
|
|
if (name)
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
STRNCPY(e->name, name, MAX_NAME_LENGTH);
|
2015-11-29 13:56:15 +01:00
|
|
|
}
|
2015-11-14 09:41:07 +01:00
|
|
|
|
|
|
|
if (groupName)
|
|
|
|
{
|
2015-12-14 09:15:41 +01:00
|
|
|
STRNCPY(e->groupName, groupName, MAX_NAME_LENGTH);
|
2015-11-14 09:41:07 +01:00
|
|
|
}
|
2015-10-20 13:51:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
node = node->next;
|
2015-10-25 13:56:24 +01:00
|
|
|
|
2015-12-20 12:58:35 +01:00
|
|
|
for (i = 0 ; i < numTypes ; i++)
|
|
|
|
{
|
|
|
|
free(types[i]);
|
|
|
|
}
|
|
|
|
|
2015-10-25 13:56:24 +01:00
|
|
|
free(types);
|
2015-10-20 13:51:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
static void loadCapitalShips(cJSON *node)
|
|
|
|
{
|
|
|
|
Entity *e;
|
|
|
|
char **types, *name, *groupName, *type;
|
|
|
|
int side, scatter, number, active;
|
2015-12-12 19:24:23 +01:00
|
|
|
int i, numTypes, addFlags;
|
2015-12-12 18:12:25 +01:00
|
|
|
long flags;
|
2015-12-07 20:19:14 +01:00
|
|
|
float x, y;
|
|
|
|
|
|
|
|
if (node)
|
|
|
|
{
|
|
|
|
node = node->child;
|
|
|
|
|
|
|
|
while (node)
|
|
|
|
{
|
|
|
|
name = NULL;
|
|
|
|
groupName = NULL;
|
|
|
|
scatter = 1;
|
|
|
|
active = 1;
|
|
|
|
number = 1;
|
2015-12-12 18:12:25 +01:00
|
|
|
flags = -1;
|
2015-12-07 20:19:14 +01:00
|
|
|
|
|
|
|
types = toTypeArray(cJSON_GetObjectItem(node, "types")->valuestring, &numTypes);
|
|
|
|
side = lookup(cJSON_GetObjectItem(node, "side")->valuestring);
|
2016-02-21 08:48:22 +01:00
|
|
|
x = (cJSON_GetObjectItem(node, "x")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_WIDTH;
|
|
|
|
y = (cJSON_GetObjectItem(node, "y")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_HEIGHT;
|
2015-12-07 20:19:14 +01:00
|
|
|
|
|
|
|
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, "number"))
|
|
|
|
{
|
|
|
|
number = cJSON_GetObjectItem(node, "number")->valueint;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cJSON_GetObjectItem(node, "scatter"))
|
|
|
|
{
|
|
|
|
scatter = cJSON_GetObjectItem(node, "scatter")->valueint;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cJSON_GetObjectItem(node, "active"))
|
|
|
|
{
|
|
|
|
active = cJSON_GetObjectItem(node, "active")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-12-12 18:12:25 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "flags"))
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
flags = flagsToLong(cJSON_GetObjectItem(node, "flags")->valuestring, &addFlags);
|
2015-12-12 18:12:25 +01:00
|
|
|
}
|
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
for (i = 0 ; i < number ; i++)
|
|
|
|
{
|
|
|
|
type = types[rand() % numTypes];
|
|
|
|
|
|
|
|
e = spawnCapitalShip(type, x, y, side);
|
|
|
|
|
|
|
|
if (scatter > 1)
|
|
|
|
{
|
|
|
|
e->x += (rand() % scatter) - (rand() % scatter);
|
|
|
|
e->y += (rand() % scatter) - (rand() % scatter);
|
|
|
|
}
|
|
|
|
|
|
|
|
e->active = active;
|
|
|
|
|
|
|
|
if (name)
|
|
|
|
{
|
|
|
|
STRNCPY(e->name, name, MAX_NAME_LENGTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (groupName)
|
|
|
|
{
|
|
|
|
STRNCPY(e->groupName, groupName, MAX_NAME_LENGTH);
|
|
|
|
}
|
2015-12-12 18:12:25 +01:00
|
|
|
|
|
|
|
if (flags != -1)
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
if (addFlags)
|
|
|
|
{
|
|
|
|
e->flags |= flags;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
e->flags = flags;
|
|
|
|
|
|
|
|
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_WARN, "Flags for '%s' (%s) replaced", e->name, e->defName);
|
|
|
|
}
|
2015-12-12 18:12:25 +01:00
|
|
|
}
|
2015-12-18 13:02:45 +01:00
|
|
|
|
2015-12-30 19:43:47 +01:00
|
|
|
updateCapitalShipComponentProperties(e);
|
2015-12-07 20:19:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
node = node->next;
|
|
|
|
|
2015-12-20 12:58:35 +01:00
|
|
|
for (i = 0 ; i < numTypes ; i++)
|
|
|
|
{
|
|
|
|
free(types[i]);
|
|
|
|
}
|
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
free(types);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-26 18:27:43 +01:00
|
|
|
static void loadEntities(cJSON *node)
|
2015-10-27 08:24:17 +01:00
|
|
|
{
|
|
|
|
Entity *e;
|
2015-11-14 09:41:07 +01:00
|
|
|
char *name, *groupName;
|
2015-11-28 15:34:38 +01:00
|
|
|
int i, type, scatter, number, active;
|
2015-11-02 11:49:10 +01:00
|
|
|
float x, y;
|
2015-10-27 08:24:17 +01:00
|
|
|
|
|
|
|
if (node)
|
|
|
|
{
|
|
|
|
node = node->child;
|
|
|
|
|
|
|
|
while (node)
|
|
|
|
{
|
2015-11-26 18:41:05 +01:00
|
|
|
e = NULL;
|
2015-10-27 08:24:17 +01:00
|
|
|
type = lookup(cJSON_GetObjectItem(node, "type")->valuestring);
|
2016-02-21 08:48:22 +01:00
|
|
|
x = (cJSON_GetObjectItem(node, "x")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_WIDTH;
|
|
|
|
y = (cJSON_GetObjectItem(node, "y")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_HEIGHT;
|
2015-10-29 12:08:47 +01:00
|
|
|
name = NULL;
|
2015-11-14 09:41:07 +01:00
|
|
|
groupName = NULL;
|
2015-11-26 18:41:05 +01:00
|
|
|
number = 1;
|
2015-11-28 15:34:38 +01:00
|
|
|
active = 1;
|
2015-11-29 17:30:59 +01:00
|
|
|
scatter = 1;
|
2015-10-27 08:24:17 +01:00
|
|
|
|
2015-10-28 20:12:58 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "name"))
|
2015-10-27 08:24:17 +01:00
|
|
|
{
|
2015-10-28 20:12:58 +01:00
|
|
|
name = cJSON_GetObjectItem(node, "name")->valuestring;
|
|
|
|
}
|
|
|
|
|
2015-11-14 09:41:07 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "groupName"))
|
|
|
|
{
|
|
|
|
groupName = cJSON_GetObjectItem(node, "groupName")->valuestring;
|
|
|
|
}
|
|
|
|
|
2015-11-26 18:41:05 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "number"))
|
|
|
|
{
|
|
|
|
number = cJSON_GetObjectItem(node, "number")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-11-28 15:34:38 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "active"))
|
|
|
|
{
|
|
|
|
active = cJSON_GetObjectItem(node, "active")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-10-28 20:12:58 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "scatter"))
|
|
|
|
{
|
|
|
|
scatter = cJSON_GetObjectItem(node, "scatter")->valueint;
|
2015-10-27 08:24:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0 ; i < number ; i++)
|
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case ET_WAYPOINT:
|
|
|
|
e = spawnWaypoint();
|
|
|
|
break;
|
2015-11-26 18:41:05 +01:00
|
|
|
|
|
|
|
case ET_EXTRACTION_POINT:
|
|
|
|
e = spawnExtractionPoint();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("Error: Unhandled entity type: %s\n", cJSON_GetObjectItem(node, "type")->valuestring);
|
|
|
|
exit(1);
|
|
|
|
break;
|
2015-10-27 08:24:17 +01:00
|
|
|
}
|
2015-10-28 20:12:58 +01:00
|
|
|
|
|
|
|
if (name)
|
|
|
|
{
|
|
|
|
STRNCPY(e->name, name, MAX_NAME_LENGTH);
|
|
|
|
}
|
|
|
|
|
2015-11-14 09:41:07 +01:00
|
|
|
if (groupName)
|
|
|
|
{
|
|
|
|
STRNCPY(e->groupName, groupName, MAX_NAME_LENGTH);
|
|
|
|
}
|
|
|
|
|
2015-11-16 13:31:50 +01:00
|
|
|
e->x = x;
|
|
|
|
e->y = y;
|
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
if (scatter > 1)
|
|
|
|
{
|
|
|
|
e->x += (rand() % scatter) - (rand() % scatter);
|
|
|
|
e->y += (rand() % scatter) - (rand() % scatter);
|
|
|
|
}
|
2015-11-16 13:31:50 +01:00
|
|
|
|
2015-11-28 15:34:38 +01:00
|
|
|
e->active = active;
|
|
|
|
|
2015-11-16 13:31:50 +01:00
|
|
|
SDL_QueryTexture(e->texture, NULL, NULL, &e->w, &e->h);
|
|
|
|
}
|
|
|
|
|
|
|
|
node = node->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void loadItems(cJSON *node)
|
|
|
|
{
|
|
|
|
Entity *e;
|
|
|
|
char *name, *groupName, *type;
|
2015-12-12 19:24:23 +01:00
|
|
|
int i, scatter, number, active, addFlags;
|
2015-11-16 13:31:50 +01:00
|
|
|
long flags;
|
|
|
|
float x, y;
|
|
|
|
|
|
|
|
flags = -1;
|
|
|
|
scatter = 1;
|
|
|
|
|
|
|
|
if (node)
|
|
|
|
{
|
|
|
|
node = node->child;
|
|
|
|
|
|
|
|
while (node)
|
|
|
|
{
|
|
|
|
type = cJSON_GetObjectItem(node, "type")->valuestring;
|
2016-02-21 08:48:22 +01:00
|
|
|
x = (cJSON_GetObjectItem(node, "x")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_WIDTH;
|
|
|
|
y = (cJSON_GetObjectItem(node, "y")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_HEIGHT;
|
2015-11-16 13:31:50 +01:00
|
|
|
name = NULL;
|
|
|
|
groupName = NULL;
|
2015-11-26 18:41:05 +01:00
|
|
|
number = 1;
|
|
|
|
active = 1;
|
2015-11-16 13:31:50 +01:00
|
|
|
|
|
|
|
if (cJSON_GetObjectItem(node, "name"))
|
|
|
|
{
|
|
|
|
name = cJSON_GetObjectItem(node, "name")->valuestring;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cJSON_GetObjectItem(node, "groupName"))
|
|
|
|
{
|
|
|
|
groupName = cJSON_GetObjectItem(node, "groupName")->valuestring;
|
|
|
|
}
|
|
|
|
|
2015-11-26 18:41:05 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "number"))
|
|
|
|
{
|
|
|
|
number = cJSON_GetObjectItem(node, "number")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-11-16 13:31:50 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "scatter"))
|
|
|
|
{
|
|
|
|
scatter = cJSON_GetObjectItem(node, "scatter")->valueint;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cJSON_GetObjectItem(node, "flags"))
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
flags = flagsToLong(cJSON_GetObjectItem(node, "flags")->valuestring, &addFlags);
|
2015-11-16 13:31:50 +01:00
|
|
|
}
|
|
|
|
|
2015-11-26 18:41:05 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "active"))
|
|
|
|
{
|
|
|
|
active = cJSON_GetObjectItem(node, "active")->valueint;
|
|
|
|
}
|
|
|
|
|
2015-11-16 13:31:50 +01:00
|
|
|
for (i = 0 ; i < number ; i++)
|
|
|
|
{
|
|
|
|
e = spawnItem(type);
|
|
|
|
|
|
|
|
if (name)
|
|
|
|
{
|
|
|
|
STRNCPY(e->name, name, MAX_NAME_LENGTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (groupName)
|
|
|
|
{
|
|
|
|
STRNCPY(e->groupName, groupName, MAX_NAME_LENGTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags != -1)
|
|
|
|
{
|
2015-12-12 19:24:23 +01:00
|
|
|
if (addFlags)
|
|
|
|
{
|
|
|
|
e->flags |= flags;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
e->flags = flags;
|
|
|
|
|
|
|
|
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_WARN, "Flags for '%s' (%s) replaced", e->name, e->defName);
|
|
|
|
}
|
2015-11-16 13:31:50 +01:00
|
|
|
}
|
|
|
|
|
2015-10-27 08:24:17 +01:00
|
|
|
e->x = x;
|
|
|
|
e->y = y;
|
2015-11-26 18:41:05 +01:00
|
|
|
e->active = active;
|
2015-10-27 08:24:17 +01:00
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
if (scatter > 1)
|
|
|
|
{
|
|
|
|
e->x += (rand() % scatter) - (rand() % scatter);
|
|
|
|
e->y += (rand() % scatter) - (rand() % scatter);
|
|
|
|
}
|
2015-11-02 14:34:59 +01:00
|
|
|
|
|
|
|
SDL_QueryTexture(e->texture, NULL, NULL, &e->w, &e->h);
|
2015-10-27 08:24:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
node = node->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-22 13:42:04 +01:00
|
|
|
static void loadLocations(cJSON *node)
|
|
|
|
{
|
2015-12-22 18:58:18 +01:00
|
|
|
int active;
|
2015-12-22 13:42:04 +01:00
|
|
|
Location *l;
|
|
|
|
|
|
|
|
if (node)
|
|
|
|
{
|
|
|
|
node = node->child;
|
|
|
|
|
|
|
|
while (node)
|
|
|
|
{
|
2015-12-22 18:58:18 +01:00
|
|
|
active = 1;
|
|
|
|
|
2015-12-22 13:42:04 +01:00
|
|
|
l = malloc(sizeof(Location));
|
|
|
|
memset(l, 0, sizeof(Location));
|
|
|
|
battle.locationTail->next = l;
|
|
|
|
battle.locationTail = l;
|
|
|
|
|
|
|
|
STRNCPY(l->name, cJSON_GetObjectItem(node, "name")->valuestring, MAX_NAME_LENGTH);
|
2016-02-21 08:48:22 +01:00
|
|
|
l->x = (cJSON_GetObjectItem(node, "x")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_WIDTH;
|
|
|
|
l->y = (cJSON_GetObjectItem(node, "y")->valuedouble / BATTLE_AREA_CELLS) * BATTLE_AREA_HEIGHT;
|
|
|
|
|
2015-12-22 13:42:04 +01:00
|
|
|
l->size = cJSON_GetObjectItem(node, "size")->valueint;
|
|
|
|
|
2015-12-22 18:58:18 +01:00
|
|
|
if (cJSON_GetObjectItem(node, "active"))
|
|
|
|
{
|
|
|
|
active = cJSON_GetObjectItem(node, "active")->valueint;
|
|
|
|
}
|
|
|
|
|
2016-02-21 08:48:22 +01:00
|
|
|
l->x += (SCREEN_WIDTH / 2);
|
|
|
|
l->y += (SCREEN_HEIGHT / 2);
|
2015-12-22 18:58:18 +01:00
|
|
|
l->active = active;
|
2015-12-22 13:42:04 +01:00
|
|
|
|
|
|
|
node = node->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-07 20:19:14 +01:00
|
|
|
static char **toTypeArray(char *types, int *numTypes)
|
2015-10-25 13:56:24 +01:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2015-11-01 10:25:10 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-25 10:29:41 +01:00
|
|
|
Mission *getMission(char *filename)
|
2015-10-20 13:51:49 +02:00
|
|
|
{
|
2015-10-25 10:29:41 +01:00
|
|
|
StarSystem *starSystem;
|
2015-10-20 13:51:49 +02:00
|
|
|
Mission *mission;
|
|
|
|
|
2015-10-25 10:29:41 +01:00
|
|
|
for (starSystem = game.starSystemHead.next ; starSystem != NULL ; starSystem = starSystem->next)
|
2015-10-20 13:51:49 +02:00
|
|
|
{
|
2015-10-25 10:29:41 +01:00
|
|
|
for (mission = starSystem->missionHead.next ; mission != NULL ; mission = mission->next)
|
2015-10-20 13:51:49 +02:00
|
|
|
{
|
2015-10-25 10:29:41 +01:00
|
|
|
if (strcmp(mission->filename, filename) == 0)
|
|
|
|
{
|
|
|
|
return mission;
|
|
|
|
}
|
2015-10-20 13:51:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-10-29 09:03:05 +01:00
|
|
|
int isMissionAvailable(Mission *mission, Mission *prev)
|
|
|
|
{
|
2015-12-21 15:19:21 +01:00
|
|
|
return prev->completed && mission->requires <= game.completedMissions;
|
2015-10-29 09:03:05 +01:00
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
}
|